-
Notifications
You must be signed in to change notification settings - Fork 29
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
fb6b727
commit 3498c8e
Showing
7 changed files
with
236 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,64 @@ | ||
# Copyright 2024 Akretion (http://www.akretion.com). | ||
# @author Florian Mounier <[email protected]> | ||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). | ||
|
||
import logging | ||
from . import models | ||
from . import http | ||
from starlette.responses import JSONResponse | ||
from odoo import SUPERUSER_ID, api | ||
from odoo.addons.extendable.models.ir_http import IrHttp | ||
from odoo.addons.fastapi.fastapi_dispatcher import FastApiDispatcher | ||
from odoo.addons.fastapi.tests.common import FastAPITransactionCase | ||
from odoo.tests.common import SavepointCase | ||
|
||
_logger = logging.getLogger(__name__) | ||
|
||
|
||
# Use SavepointCase instead of TransactionCase (16.0 merge) | ||
# And use test mode to avoid deadlock in envioronment RLock | ||
class TestModeSavepointCase(SavepointCase): | ||
def setUp(self): | ||
super().setUp() | ||
self.registry.enter_test_mode(self.env.cr) | ||
|
||
def tearDown(self): | ||
self.registry.leave_test_mode() | ||
super().tearDown() | ||
|
||
@classmethod | ||
def _patch_app_to_handle_exception(cls, app): | ||
def handle_error(request, exc): | ||
_logger.error("Error in test request", exc_info=exc) | ||
|
||
def make_json_response(body, status, headers): | ||
|
||
response = JSONResponse(body, status_code=status) | ||
if headers: | ||
response.headers.update(headers) | ||
return response | ||
|
||
request.make_json_response = make_json_response | ||
return FastApiDispatcher(request).handle_error(exc) | ||
|
||
app.exception_handlers = {Exception: handle_error} | ||
|
||
|
||
FastAPITransactionCase.__bases__ = (TestModeSavepointCase,) | ||
|
||
|
||
@classmethod | ||
def _dispatch(cls): | ||
with cls._extendable_context_registry(): | ||
return super(IrHttp, cls)._dispatch() | ||
|
||
|
||
IrHttp._dispatch = _dispatch | ||
|
||
|
||
def post_init_hook(cr, registry): | ||
env = api.Environment(cr, SUPERUSER_ID, {}) | ||
# this is the trigger that sends notifications when jobs change | ||
_logger.info("Resyncing registries") | ||
endpoints_ids = env["fastapi.endpoint"].search([]).ids | ||
env["fastapi.endpoint"]._handle_registry_sync(endpoints_ids) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
# Copyright 2024 Akretion (http://www.akretion.com). | ||
# @author Florian Mounier <[email protected]> | ||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). | ||
|
||
{ | ||
"name": "Fastapi Backport", | ||
"summary": "Backport of FastAPI to Odoo 14.0", | ||
"version": "14.0.1.0.0", | ||
"author": " Akretion", | ||
"license": "AGPL-3", | ||
"depends": [ | ||
"sixteen_in_fourteen", | ||
"base_contextvars", | ||
"base_future_response", | ||
"fastapi", | ||
"pydantic", | ||
"extendable_fastapi", | ||
"extendable", | ||
], | ||
"post_init_hook": "post_init_hook", | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,110 @@ | ||
# Copyright 2023 ACSONE SA/NV | ||
# Copyright 2024 Akretion (http://www.akretion.com). | ||
# @author Florian Mounier <[email protected]> | ||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). | ||
|
||
import json | ||
import logging | ||
from functools import lru_cache | ||
|
||
import werkzeug.datastructures | ||
|
||
import odoo | ||
from odoo import http | ||
from odoo.tools import date_utils | ||
|
||
from odoo.addons.fastapi.fastapi_dispatcher import FastApiDispatcher | ||
|
||
_logger = logging.getLogger(__name__) | ||
|
||
|
||
class FastapiRootPaths: | ||
_root_paths_by_db = {} | ||
|
||
@classmethod | ||
def set_root_paths(cls, db, root_paths): | ||
cls._root_paths_by_db[db] = root_paths | ||
cls.is_fastapi_path.cache_clear() | ||
|
||
@classmethod | ||
@lru_cache(maxsize=1024) | ||
def is_fastapi_path(cls, db, path): | ||
return any( | ||
path.startswith(root_path) | ||
for root_path in cls._root_paths_by_db.get(db, []) | ||
) | ||
|
||
|
||
class FastapiRequest(http.WebRequest): | ||
_request_type = "fastapi" | ||
|
||
def __init__(self, *args): | ||
super().__init__(*args) | ||
self.params = {} | ||
self._dispatcher = FastApiDispatcher(self) | ||
|
||
def make_response(self, data, headers=None, cookies=None, status=200): | ||
"""Helper for non-HTML responses, or HTML responses with custom | ||
response headers or cookies. | ||
While handlers can just return the HTML markup of a page they want to | ||
send as a string if non-HTML data is returned they need to create a | ||
complete response object, or the returned data will not be correctly | ||
interpreted by the clients. | ||
:param basestring data: response body | ||
:param headers: HTTP headers to set on the response | ||
:type headers: ``[(name, value)]`` | ||
:param collections.abc.Mapping cookies: cookies to set on the client | ||
""" | ||
response = http.Response(data, status=status, headers=headers) | ||
if cookies: | ||
for k, v in cookies.items(): | ||
response.set_cookie(k, v) | ||
return response | ||
|
||
def make_json_response(self, data, headers=None, cookies=None, status=200): | ||
"""Helper for JSON responses, it json-serializes ``data`` and | ||
sets the Content-Type header accordingly if none is provided. | ||
:param data: the data that will be json-serialized into the response body | ||
:param int status: http status code | ||
:param List[(str, str)] headers: HTTP headers to set on the response | ||
:param collections.abc.Mapping cookies: cookies to set on the client | ||
:rtype: :class:`~odoo.http.Response` | ||
""" | ||
data = json.dumps(data, ensure_ascii=False, default=date_utils.json_default) | ||
|
||
headers = werkzeug.datastructures.Headers(headers) | ||
headers["Content-Length"] = len(data) | ||
if "Content-Type" not in headers: | ||
headers["Content-Type"] = "application/json; charset=utf-8" | ||
|
||
return self.make_response(data, headers.to_wsgi_list(), cookies, status) | ||
|
||
def dispatch(self): | ||
return self._dispatcher.dispatch(None, None) | ||
|
||
def _handle_exception(self, exception): | ||
_logger.exception( | ||
"Exception during fastapi request handling", exc_info=exception | ||
) | ||
return self._dispatcher.handle_error(exception) | ||
|
||
|
||
ori_get_request = http.root.__class__.get_request | ||
|
||
|
||
def get_request(self, httprequest): | ||
db = httprequest.session.db | ||
if db and odoo.service.db.exp_db_exist(db): | ||
# on the very first request processed by a worker, | ||
# registry is not loaded yet | ||
# so we enforce its loading here. | ||
odoo.registry(db) | ||
if FastapiRootPaths.is_fastapi_path(db, httprequest.path): | ||
return FastapiRequest(httprequest) | ||
return ori_get_request(self, httprequest) | ||
|
||
|
||
http.root.__class__.get_request = get_request |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
from . import fastapi_endpoint |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,33 @@ | ||
# Copyright 2024 Akretion (http://www.akretion.com). | ||
# @author Florian Mounier <[email protected]> | ||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). | ||
|
||
|
||
from odoo import api, models | ||
|
||
from ..http import FastapiRootPaths | ||
|
||
|
||
class FastapiEndpoint(models.Model): | ||
_inherit = "fastapi.endpoint" | ||
|
||
@api.model | ||
def _update_root_paths_registry(self): | ||
root_paths = self.env["fastapi.endpoint"].search([]).mapped("root_path") | ||
FastapiRootPaths.set_root_paths(self.env.cr.dbname, root_paths) | ||
|
||
def _register_hook(self): | ||
super()._register_hook() | ||
self._update_root_paths_registry() | ||
|
||
def _inverse_root_path(self): | ||
super()._inverse_root_path() | ||
self._update_root_paths_registry() | ||
|
||
@api.depends("root_path") | ||
def _compute_urls(self): | ||
base_url = self.env["ir.config_parameter"].sudo().get_param("web.base.url") | ||
for rec in self: | ||
rec.docs_url = f"{base_url}{rec.root_path}/docs" | ||
rec.redoc_url = f"{base_url}{rec.root_path}/redoc" | ||
rec.openapi_url = f"{base_url}{rec.root_path}/openapi.json" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
../../../../fastapi_backport |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
import setuptools | ||
|
||
setuptools.setup( | ||
setup_requires=['setuptools-odoo'], | ||
odoo_addon=True, | ||
) |