first commit

This commit is contained in:
root
2026-03-14 09:42:12 +00:00
commit 0adbd20c2c
10991 changed files with 1646955 additions and 0 deletions

View File

@@ -0,0 +1,184 @@
# This file is part of Tryton. The COPYRIGHT file at the top level of
# this repository contains the full copyright notices and license terms.
from oauthlib.oauth2 import BackendApplicationClient
from requests_oauthlib import OAuth2Session
from trytond.i18n import gettext
from trytond.model import (
MatchMixin, ModelSQL, ModelView, fields, sequence_ordered)
from trytond.pool import Pool, PoolMeta
from trytond.pyson import Eval
from .exceptions import UPSCredentialWarning
class CredentialUPS(sequence_ordered(), ModelSQL, ModelView, MatchMixin):
__name__ = 'carrier.credential.ups'
_rec_name = 'account_number'
company = fields.Many2One('company.company', 'Company')
client_id = fields.Char("Client ID", required=True)
client_secret = fields.Char("Client Secret", required=True)
account_number = fields.Char('Account Number', required=True)
use_metric = fields.Boolean('Use Metric')
use_international_forms = fields.Boolean("Use International Forms")
server = fields.Selection([
('testing', 'Testing'),
('production', 'Production'),
], 'Server')
@classmethod
def __register__(cls, module):
table_h = cls.__table_handler__(module)
super().__register__(module)
# Migration from 6.8: switch to OAuth
table_h.drop_column('user_id')
table_h.drop_column('password')
table_h.drop_column('license')
@classmethod
def default_use_metric(cls):
return True
@classmethod
def default_use_international_forms(cls):
return False
@classmethod
def default_server(cls):
return 'testing'
def get_token(self):
if self.server == 'production':
url = 'https://onlinetools.ups.com/security/v1/oauth/token'
else:
url = 'https://wwwcie.ups.com/security/v1/oauth/token'
client = BackendApplicationClient(client_id=self.client_id)
oauth = OAuth2Session(client=client)
token = oauth.fetch_token(
token_url=url, client_id=self.client_id,
client_secret=self.client_secret)
return token['access_token']
def get_shipment_url(self):
if self.server == 'production':
return 'https://onlinetools.ups.com/api/shipments/v2403/ship'
else:
return 'https://wwwcie.ups.com/api/shipments/v2403/ship'
@classmethod
def check_modification(
cls, mode, credentials, values=None, external=False):
pool = Pool()
Warning = pool.get('res.user.warning')
super().check_modification(
mode, credentials, values=values, external=external)
if (mode == 'write'
and external
and values.keys() & {
'client_id', 'client_secret', 'account_number'}):
warning_name = Warning.format('dpd_credential', credentials)
if Warning.check(warning_name):
raise UPSCredentialWarning(
warning_name,
gettext('stock_package_shipping_ups'
'.msg_ups_credential_modified'))
class Carrier(metaclass=PoolMeta):
__name__ = 'carrier'
ups_service_type = fields.Selection([
(None, ''),
('01', 'Next Day Air'),
('02', '2nd Day Air'),
('03', 'Ground'),
('07', 'Express'),
('08', 'Expedited'),
('11', 'UPS Standard'),
('12', '3 Days Select'),
('13', 'Next Day Air Saver'),
('14', 'UPS Next Day Air Early'),
('17', "UPS Worldwide Economy DDU"),
('54', 'Express Plus'),
('59', '2nd Day Air A.M.'),
('65', 'UPS Saver'),
('M2', 'First Class Mail'),
('M3', 'Priority Mail'),
('M4', 'Expedited Mail Innovations'),
('M5', 'Priority Mail Innovations'),
('M6', 'Economy Mail Innovations'),
('M7', "Mail Innovations (MI) Returns"),
('70', 'UPS Access Point Economy'),
('71', "UPS Worldwide Express Freight Midday"),
('72', "UPS Worldwide Economy DDP"),
('74', "UPS Express 12:00"),
('82', 'UPS Today Standard'),
('83', 'UPS Today Dedicated Courier'),
('84', 'UPS Today Intercity'),
('85', 'UPS Today Express'),
('86', 'UPS Today Express Saver'),
('96', 'UPS Worldwide Express Freight'),
], 'Service Type', sort=False, translate=False,
states={
'required': Eval('shipping_service') == 'ups',
'invisible': Eval('shipping_service') != 'ups',
})
ups_label_image_format = fields.Selection([
(None, ''),
('EPL', 'EPL2'),
('SPL', 'SPL'),
('ZPL', 'ZPL'),
('GIF', 'GIF'),
('STARPL', 'Star Printer'),
], 'Label Image Format', sort=False, translate=False,
states={
'required': Eval('shipping_service') == 'ups',
'invisible': Eval('shipping_service') != 'ups',
})
ups_label_height = fields.Selection([
(None, ''),
('6', '6'),
('8', '8'),
], 'Label Height', sort=False, translate=False,
states={
'required': ((Eval('shipping_service') == 'ups')
& (Eval('ups_label_image_format') != 'GIF')),
'invisible': ((Eval('shipping_service') != 'ups')
| (Eval('ups_label_image_format') == 'GIF')),
})
ups_notifications = fields.MultiSelection([
('5', "Quantum View In-transit"),
('6', "Quantum View Shop"),
('7', "Quantum View Exception"),
('8', "Quantum View Delivery"),
('2', "Return or Label Creation"),
('012', "Alternate Delivery Location"),
('013', "UPS Access Point Shipper"),
], "Notifications", sort=False,
states={
'invisible': Eval('shipping_service') != 'ups',
})
@classmethod
def __setup__(cls):
super().__setup__()
cls.shipping_service.selection.append(('ups', 'UPS'))
@classmethod
def view_attributes(cls):
return super().view_attributes() + [
("/form/separator[@id='ups']", 'states', {
'invisible': Eval('shipping_service') != 'ups',
}),
]
@property
def shipping_label_mimetype(self):
mimetype = super().shipping_label_mimetype
if self.shipping_service == 'ups':
if self.ups_label_image_format == 'GIF':
mimetype = 'image/gif'
return mimetype