mirror of
https://github.com/bringout/oca-ocb-accounting.git
synced 2026-04-20 23:42:00 +02:00
Initial commit: Accounting packages
This commit is contained in:
commit
4ef34c2317
2661 changed files with 1709616 additions and 0 deletions
|
|
@ -0,0 +1,174 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||||
|
||||
from datetime import datetime
|
||||
import json
|
||||
import logging
|
||||
|
||||
import requests
|
||||
from werkzeug import urls
|
||||
|
||||
from odoo import api, fields, models, _
|
||||
|
||||
_logger = logging.getLogger(__name__)
|
||||
|
||||
TIMEOUT = 20
|
||||
|
||||
DEFAULT_MICROSOFT_AUTH_ENDPOINT = 'https://login.microsoftonline.com/common/oauth2/v2.0/authorize'
|
||||
DEFAULT_MICROSOFT_TOKEN_ENDPOINT = 'https://login.microsoftonline.com/common/oauth2/v2.0/token'
|
||||
DEFAULT_MICROSOFT_GRAPH_ENDPOINT = 'https://graph.microsoft.com'
|
||||
|
||||
RESOURCE_NOT_FOUND_STATUSES = (204, 404)
|
||||
|
||||
class MicrosoftService(models.AbstractModel):
|
||||
_name = 'microsoft.service'
|
||||
_description = 'Microsoft Service'
|
||||
|
||||
def _get_calendar_scope(self):
|
||||
return 'offline_access openid Calendars.ReadWrite'
|
||||
|
||||
@api.model
|
||||
def _get_auth_endpoint(self):
|
||||
return self.env["ir.config_parameter"].sudo().get_param('microsoft_account.auth_endpoint', DEFAULT_MICROSOFT_AUTH_ENDPOINT)
|
||||
|
||||
@api.model
|
||||
def _get_token_endpoint(self):
|
||||
return self.env["ir.config_parameter"].sudo().get_param('microsoft_account.token_endpoint', DEFAULT_MICROSOFT_TOKEN_ENDPOINT)
|
||||
|
||||
@api.model
|
||||
def generate_refresh_token(self, service, authorization_code):
|
||||
""" Call Microsoft API to refresh the token, with the given authorization code
|
||||
:param service : the name of the microsoft service to actualize
|
||||
:param authorization_code : the code to exchange against the new refresh token
|
||||
:returns the new refresh token
|
||||
"""
|
||||
Parameters = self.env['ir.config_parameter'].sudo()
|
||||
client_id = Parameters.get_param('microsoft_%s_client_id' % service)
|
||||
client_secret = Parameters.get_param('microsoft_%s_client_secret' % service)
|
||||
redirect_uri = Parameters.get_param('microsoft_redirect_uri')
|
||||
|
||||
scope = self._get_calendar_scope()
|
||||
|
||||
# Get the Refresh Token From Microsoft And store it in ir.config_parameter
|
||||
headers = {"Content-type": "application/x-www-form-urlencoded"}
|
||||
data = {
|
||||
'client_id': client_id,
|
||||
'redirect_uri': redirect_uri,
|
||||
'client_secret': client_secret,
|
||||
'scope': scope,
|
||||
'grant_type': "refresh_token"
|
||||
}
|
||||
try:
|
||||
req = requests.post(self._get_token_endpoint(), data=data, headers=headers, timeout=TIMEOUT)
|
||||
req.raise_for_status()
|
||||
content = req.json()
|
||||
except requests.exceptions.RequestException as exc:
|
||||
error_msg = _("Something went wrong during your token generation. Maybe your Authorization Code is invalid or already expired")
|
||||
raise self.env['res.config.settings'].get_config_warning(error_msg) from exc
|
||||
|
||||
return content.get('refresh_token')
|
||||
|
||||
@api.model
|
||||
def _get_authorize_uri(self, from_url, service, scope):
|
||||
""" This method return the url needed to allow this instance of Odoo to access to the scope
|
||||
of gmail specified as parameters
|
||||
"""
|
||||
state = {
|
||||
'd': self.env.cr.dbname,
|
||||
's': service,
|
||||
'f': from_url
|
||||
}
|
||||
|
||||
get_param = self.env['ir.config_parameter'].sudo().get_param
|
||||
base_url = get_param('web.base.url', default='http://www.odoo.com?NoBaseUrl')
|
||||
client_id = get_param('microsoft_%s_client_id' % (service,), default=False)
|
||||
|
||||
encoded_params = urls.url_encode({
|
||||
'response_type': 'code',
|
||||
'client_id': client_id,
|
||||
'state': json.dumps(state),
|
||||
'scope': scope,
|
||||
'redirect_uri': base_url + '/microsoft_account/authentication',
|
||||
'access_type': 'offline'
|
||||
})
|
||||
return "%s?%s" % (self._get_auth_endpoint(), encoded_params)
|
||||
|
||||
@api.model
|
||||
def _get_microsoft_tokens(self, authorize_code, service):
|
||||
""" Call Microsoft API to exchange authorization code against token, with POST request, to
|
||||
not be redirected.
|
||||
"""
|
||||
get_param = self.env['ir.config_parameter'].sudo().get_param
|
||||
base_url = get_param('web.base.url', default='http://www.odoo.com?NoBaseUrl')
|
||||
client_id = get_param('microsoft_%s_client_id' % (service,), default=False)
|
||||
client_secret = get_param('microsoft_%s_client_secret' % (service,), default=False)
|
||||
scope = self._get_calendar_scope()
|
||||
|
||||
headers = {"content-type": "application/x-www-form-urlencoded"}
|
||||
data = {
|
||||
'code': authorize_code,
|
||||
'client_id': client_id,
|
||||
'client_secret': client_secret,
|
||||
'grant_type': 'authorization_code',
|
||||
'scope': scope,
|
||||
'redirect_uri': base_url + '/microsoft_account/authentication'
|
||||
}
|
||||
try:
|
||||
dummy, response, dummy = self._do_request(self._get_token_endpoint(), params=data, headers=headers, method='POST', preuri='')
|
||||
access_token = response.get('access_token')
|
||||
refresh_token = response.get('refresh_token')
|
||||
ttl = response.get('expires_in')
|
||||
return access_token, refresh_token, ttl
|
||||
except requests.HTTPError:
|
||||
error_msg = _("Something went wrong during your token generation. Maybe your Authorization Code is invalid")
|
||||
raise self.env['res.config.settings'].get_config_warning(error_msg)
|
||||
|
||||
@api.model
|
||||
def _do_request(self, uri, params=None, headers=None, method='POST', preuri=DEFAULT_MICROSOFT_GRAPH_ENDPOINT, timeout=TIMEOUT):
|
||||
""" Execute the request to Microsoft API. Return a tuple ('HTTP_CODE', 'HTTP_RESPONSE')
|
||||
:param uri : the url to contact
|
||||
:param params : dict or already encoded parameters for the request to make
|
||||
:param headers : headers of request
|
||||
:param method : the method to use to make the request
|
||||
:param preuri : pre url to prepend to param uri.
|
||||
"""
|
||||
if params is None:
|
||||
params = {}
|
||||
if headers is None:
|
||||
headers = {}
|
||||
|
||||
assert urls.url_parse(preuri + uri).host in [
|
||||
urls.url_parse(url).host for url in (DEFAULT_MICROSOFT_TOKEN_ENDPOINT, DEFAULT_MICROSOFT_GRAPH_ENDPOINT)
|
||||
]
|
||||
|
||||
_logger.debug("Uri: %s - Type : %s - Headers: %s - Params : %s !" % (uri, method, headers, params))
|
||||
|
||||
ask_time = fields.Datetime.now()
|
||||
try:
|
||||
if method.upper() in ('GET', 'DELETE'):
|
||||
res = requests.request(method.lower(), preuri + uri, headers=headers, params=params, timeout=timeout)
|
||||
elif method.upper() in ('POST', 'PATCH', 'PUT'):
|
||||
res = requests.request(method.lower(), preuri + uri, data=params, headers=headers, timeout=timeout)
|
||||
else:
|
||||
raise Exception(_('Method not supported [%s] not in [GET, POST, PUT, PATCH or DELETE]!', method))
|
||||
res.raise_for_status()
|
||||
status = res.status_code
|
||||
|
||||
if int(status) in RESOURCE_NOT_FOUND_STATUSES:
|
||||
response = {}
|
||||
else:
|
||||
# Some answers return empty content
|
||||
response = res.content and res.json() or {}
|
||||
|
||||
try:
|
||||
ask_time = datetime.strptime(res.headers.get('date'), "%a, %d %b %Y %H:%M:%S %Z")
|
||||
except:
|
||||
pass
|
||||
except requests.HTTPError as error:
|
||||
if error.response.status_code in RESOURCE_NOT_FOUND_STATUSES:
|
||||
status = error.response.status_code
|
||||
response = {}
|
||||
else:
|
||||
_logger.exception("Bad microsoft request : %s !", error.response.content)
|
||||
raise error
|
||||
return (status, response, ask_time)
|
||||
Loading…
Add table
Add a link
Reference in a new issue