diff --git a/frontik/app.py b/frontik/app.py index 63cb91dd8..8a7bc709b 100644 --- a/frontik/app.py +++ b/frontik/app.py @@ -7,7 +7,7 @@ from collections.abc import Callable from ctypes import c_bool, c_int from threading import Lock -from typing import Optional, Union +from typing import Awaitable, Optional, Union from aiokafka import AIOKafkaProducer from fastapi import FastAPI, HTTPException @@ -15,21 +15,44 @@ from http_client import options as http_client_options from http_client.balancing import RequestBalancerBuilder, Upstream from lxml import etree +from tornado import httputil import frontik.producers.json_producer import frontik.producers.xml_producer from frontik import integrations, media_types from frontik.debug import get_frontik_and_apps_versions from frontik.handler import PageHandler, get_current_handler +from frontik.handler_asgi import execute_page from frontik.integrations.statsd import StatsDClient, StatsDClientStub, create_statsd_client from frontik.options import options from frontik.process import WorkerState -from frontik.routing import router +from frontik.routing import import_all_pages, router from frontik.service_discovery import UpstreamManager +from frontik.util import check_request_id, generate_uniq_timestamp_request_id app_logger = logging.getLogger('app_logger') +class AsgiRouter: + async def __call__(self, scope, receive, send): + assert scope['type'] == 'http' + + if 'router' not in scope: + scope['router'] = self + + route = scope['route'] + scope['endpoint'] = route.endpoint + + await route.handle(scope, receive, send) + + +class FrontikAsgiApp(FastAPI): + def __init__(self) -> None: + super().__init__() + self.router = AsgiRouter() # type: ignore + self.http_client = None + + @router.get('/version', cls=PageHandler) async def get_version(handler: PageHandler = get_current_handler()) -> None: handler.set_header('Content-Type', 'text/xml') @@ -53,10 +76,8 @@ class DefaultConfig: def __init__(self, app_module_name: Optional[str] = None) -> None: self.start_time = time.time() - self.fastapi_app = FastAPI() - self.app_module_name: Optional[str] = app_module_name - if app_module_name is None: + if app_module_name is None: # for tests app_module = importlib.import_module(self.__class__.__module__) else: app_module = importlib.import_module(app_module_name) @@ -79,6 +100,23 @@ def __init__(self, app_module_name: Optional[str] = None) -> None: count_down_lock = multiprocessing.Lock() self.worker_state = WorkerState(init_workers_count_down, master_done, count_down_lock) # type: ignore + import_all_pages(app_module_name) + + self.ui_methods: dict = {} + self.ui_modules: dict = {} + self.settings: dict = {} + + self.app = FrontikAsgiApp() + + def __call__(self, tornado_request: httputil.HTTPServerRequest) -> Optional[Awaitable[None]]: + # for making more asgi, reimplement tornado.http1connection._server_request_loop and ._read_message + request_id = tornado_request.headers.get('X-Request-Id') or generate_uniq_timestamp_request_id() + if options.validate_request_id: + check_request_id(request_id) + + task = asyncio.create_task(execute_page(self, tornado_request, request_id, self.app)) + return task + def create_upstream_manager( self, upstreams: dict[str, Upstream], @@ -164,3 +202,6 @@ def get_current_status(self) -> dict[str, str]: def get_kafka_producer(self, producer_name: str) -> Optional[AIOKafkaProducer]: # pragma: no cover pass + + def log_request(self, tornado_handler: PageHandler) -> None: + pass diff --git a/frontik/auth.py b/frontik/auth.py index 671c8dc67..1f10a0284 100644 --- a/frontik/auth.py +++ b/frontik/auth.py @@ -4,9 +4,12 @@ import http.client from typing import TYPE_CHECKING, Optional +from tornado import httputil from tornado.escape import to_unicode from tornado.web import Finish +from frontik.options import options + if TYPE_CHECKING: from frontik.handler import PageHandler @@ -17,8 +20,8 @@ class DebugUnauthorizedError(Finish): pass -def passed_basic_auth(handler: PageHandler, login: Optional[str], passwd: Optional[str]) -> bool: - auth_header = handler.get_header('Authorization') +def passed_basic_auth(tornado_request: httputil.HTTPServerRequest, login: Optional[str], passwd: Optional[str]) -> bool: + auth_header = tornado_request.headers.get('Authorization') if auth_header and auth_header.startswith('Basic '): method, auth_b64 = auth_header.split(' ') try: @@ -30,21 +33,30 @@ def passed_basic_auth(handler: PageHandler, login: Optional[str], passwd: Option return False -def check_debug_auth(handler: PageHandler, login: Optional[str], password: Optional[str]) -> None: - """ - :type handler: tornado.web.RequestHandler - :return: None or tuple(http_code, headers) - """ - debug_auth_header = handler.get_header(DEBUG_AUTH_HEADER_NAME) +def check_debug_auth( + tornado_request: httputil.HTTPServerRequest, login: Optional[str], password: Optional[str] +) -> Optional[str]: + debug_auth_header = tornado_request.headers.get(DEBUG_AUTH_HEADER_NAME) if debug_auth_header is not None: debug_access = debug_auth_header == f'{login}:{password}' if not debug_access: - handler.set_header('WWW-Authenticate', f'{DEBUG_AUTH_HEADER_NAME}-Header realm="Secure Area"') - handler.set_status(http.client.UNAUTHORIZED) - handler.finish() + return f'{DEBUG_AUTH_HEADER_NAME}-Header realm="Secure Area"' else: - debug_access = passed_basic_auth(handler, login, password) + debug_access = passed_basic_auth(tornado_request, login, password) if not debug_access: - handler.set_header('WWW-Authenticate', 'Basic realm="Secure Area"') - handler.set_status(http.client.UNAUTHORIZED) - handler.finish() + return 'Basic realm="Secure Area"' + return None + + +def check_debug_auth_or_finish( + handler: PageHandler, login: Optional[str] = None, password: Optional[str] = None +) -> None: + if options.debug: + return + login = login or options.debug_login + password = password or options.debug_password + fail_header = check_debug_auth(handler.request, login, password) + if fail_header: + handler.set_header('WWW-Authenticate', fail_header) + handler.set_status(http.client.UNAUTHORIZED) + handler.finish() diff --git a/frontik/balancing_client.py b/frontik/balancing_client.py new file mode 100644 index 000000000..c5a830077 --- /dev/null +++ b/frontik/balancing_client.py @@ -0,0 +1,57 @@ +import time +from functools import partial +from typing import Annotated + +from fastapi import Depends, Request +from http_client import HttpClient, RequestBuilder +from http_client.request_response import USER_AGENT_HEADER + +from frontik import request_context +from frontik.auth import DEBUG_AUTH_HEADER_NAME +from frontik.debug import DEBUG_HEADER_NAME +from frontik.timeout_tracking import get_timeout_checker +from frontik.util import make_url + +OUTER_TIMEOUT_MS_HEADER = 'X-Outer-Timeout-Ms' + + +def modify_http_client_request(request: Request, balanced_request: RequestBuilder) -> None: + balanced_request.headers['x-request-id'] = request_context.get_request_id() + balanced_request.headers[OUTER_TIMEOUT_MS_HEADER] = f'{balanced_request.request_timeout * 1000:.0f}' + + outer_timeout = request.headers.get(OUTER_TIMEOUT_MS_HEADER.lower()) + if outer_timeout: + timeout_checker = get_timeout_checker( + request.headers.get(USER_AGENT_HEADER.lower()), + float(outer_timeout), + request['start_time'], + ) + timeout_checker.check(balanced_request) + + if request['pass_debug']: + balanced_request.headers[DEBUG_HEADER_NAME] = 'true' + + # debug_timestamp is added to avoid caching of debug responses + balanced_request.path = make_url(balanced_request.path, debug_timestamp=int(time.time())) + + for header_name in ('Authorization', DEBUG_AUTH_HEADER_NAME): + authorization = request.headers.get(header_name.lower()) + if authorization is not None: + balanced_request.headers[header_name] = authorization + + +def get_http_client(modify_request_hook=None): + def _get_http_client(request: Request) -> HttpClient: + hook = modify_request_hook or partial(modify_http_client_request, request) + + http_client = request['http_client_factory'].get_http_client( + modify_http_request_hook=hook, + debug_enabled=request['debug_enabled'], + ) + + return http_client + + return _get_http_client + + +HttpClientT = Annotated[HttpClient, Depends(get_http_client())] diff --git a/frontik/debug.py b/frontik/debug.py index 03d28e767..e32c019e7 100644 --- a/frontik/debug.py +++ b/frontik/debug.py @@ -21,17 +21,19 @@ import aiohttp import tornado -from fastapi import Request from lxml import etree from lxml.builder import E -from starlette.datastructures import Headers +from tornado import httputil from tornado.escape import to_unicode, utf8 from tornado.httputil import HTTPHeaders import frontik.util import frontik.xml_util from frontik import media_types, request_context +from frontik.auth import check_debug_auth from frontik.loggers import BufferedHandler +from frontik.options import options +from frontik.util import get_cookie_or_param_from_request from frontik.version import version as frontik_version from frontik.xml_util import dict_to_xml @@ -41,7 +43,6 @@ from http_client.request_response import RequestBuilder, RequestResult from frontik.app import FrontikApplication - from frontik.handler import PageHandler debug_log = logging.getLogger('frontik.debug') @@ -203,7 +204,7 @@ def _params_to_xml(url: str) -> etree.Element: return params -def _headers_to_xml(request_or_response_headers: dict | Headers) -> etree.Element: +def _headers_to_xml(request_or_response_headers: HTTPHeaders) -> etree.Element: headers = etree.Element('headers') for name, value in request_or_response_headers.items(): if name != 'Cookie': @@ -212,7 +213,7 @@ def _headers_to_xml(request_or_response_headers: dict | Headers) -> etree.Elemen return headers -def _cookies_to_xml(request_or_response_headers: dict) -> etree.Element: +def _cookies_to_xml(request_or_response_headers: HTTPHeaders) -> etree.Element: cookies = etree.Element('cookies') if 'Cookie' in request_or_response_headers: _cookies: SimpleCookie = SimpleCookie(request_or_response_headers['Cookie']) @@ -365,18 +366,39 @@ def _produce_one(self, record: logging.LogRecord) -> etree.Element: DEBUG_XSL = os.path.join(os.path.dirname(__file__), 'debug/debug.xsl') +def _data_to_chunk(data: Any, headers: HTTPHeaders) -> bytes: + result: bytes = b'' + if data is None: + return result + if isinstance(data, str): + result = data.encode('utf-8') + elif isinstance(data, dict): + chunk = json.dumps(data).replace(' None: + def __init__(self, application: FrontikApplication, debug_mode: DebugMode) -> None: self.application = application - self.request: Request = request + self.debug_mode = debug_mode def is_enabled(self) -> bool: - return getattr(self.request.state.handler, '_debug_enabled', False) + return self.debug_mode.enabled def is_inherited(self) -> bool: - return getattr(self.request.state.handler, '_debug_inherited', False) + return self.debug_mode.inherited + + def transform_chunk( + self, tornado_request: httputil.HTTPServerRequest, status_code: int, original_headers: HTTPHeaders, data: bytes + ) -> tuple[int, HTTPHeaders, bytes]: + chunk = _data_to_chunk(data, original_headers) - def transform_chunk(self, status_code: int, original_headers: dict, chunk: bytes) -> tuple[int, dict, bytes]: if not self.is_enabled(): return status_code, original_headers, chunk @@ -390,9 +412,9 @@ def transform_chunk(self, status_code: int, original_headers: dict, chunk: bytes debug_log_data = request_context.get_log_handler().produce_all() # type: ignore debug_log_data.set('code', str(int(status_code))) debug_log_data.set('handler-name', request_context.get_handler_name()) - debug_log_data.set('started', _format_number(self.request.state.start_time)) - debug_log_data.set('request-id', str(self.request.state.handler.request_id)) - debug_log_data.set('stages-total', _format_number((time.time() - self.request.state.start_time) * 1000)) + debug_log_data.set('started', _format_number(tornado_request._start_time)) + debug_log_data.set('request-id', str(tornado_request.request_id)) # type: ignore + debug_log_data.set('stages-total', _format_number((time.time() - tornado_request._start_time) * 1000)) try: debug_log_data.append(E.versions(_pretty_print_xml(get_frontik_and_apps_versions(self.application)))) @@ -408,10 +430,10 @@ def transform_chunk(self, status_code: int, original_headers: dict, chunk: bytes debug_log_data.append( E.request( - E.method(self.request.method), - _params_to_xml(str(self.request.url)), - _headers_to_xml(self.request.headers), - _cookies_to_xml(self.request.headers), # type: ignore + E.method(tornado_request.method), + _params_to_xml(str(tornado_request.uri)), + _headers_to_xml(tornado_request.headers), + _cookies_to_xml(tornado_request.headers), ), ) @@ -432,7 +454,7 @@ def transform_chunk(self, status_code: int, original_headers: dict, chunk: bytes upstream.set('bgcolor', bgcolor) upstream.set('fgcolor', fgcolor) - if not getattr(self.request.state.handler, '_debug_inherited', False): + if not self.debug_mode.inherited: try: transform = etree.XSLT(etree.parse(DEBUG_XSL)) log_document = utf8(str(transform(debug_log_data))) @@ -449,35 +471,43 @@ def transform_chunk(self, status_code: int, original_headers: dict, chunk: bytes else: log_document = etree.tostring(debug_log_data, encoding='UTF-8', xml_declaration=True) - return 200, wrap_headers, log_document + return 200, HTTPHeaders(wrap_headers), log_document class DebugMode: - def __init__(self, handler: PageHandler) -> None: - debug_value = frontik.util.get_cookie_or_url_param_value(handler, 'debug') - - self.mode_values = debug_value.split(',') if debug_value is not None else '' - self.inherited = handler.get_header(DEBUG_HEADER_NAME, None) - self.pass_debug: bool = False + def __init__(self, tornado_request: httputil.HTTPServerRequest) -> None: + self.debug_value = get_cookie_or_param_from_request(tornado_request, 'debug') + self.mode_values = self.debug_value.split(',') if self.debug_value is not None else '' + self.inherited = tornado_request.headers.get(DEBUG_HEADER_NAME, None) + self.pass_debug = False + self.enabled = False + self.profile_xslt = False + self.failed_auth_header = None if self.inherited: debug_log.debug('debug mode is inherited due to %s request header', DEBUG_HEADER_NAME) - handler._debug_inherited = True # type: ignore - if debug_value is not None or self.inherited: - handler.require_debug_access() + if self.debug_value is not None or self.inherited: + if options.debug: + self.on_auth_ok() + return - self.enabled = handler._debug_enabled = True # type: ignore - self.pass_debug = 'nopass' not in self.mode_values or bool(self.inherited) - self.profile_xslt = 'xslt' in self.mode_values + self.failed_auth_header = check_debug_auth(tornado_request, options.debug_login, options.debug_password) + if not self.failed_auth_header: + self.on_auth_ok() - request_context.set_log_handler(DebugBufferedHandler()) + def on_auth_ok(self) -> None: + self.enabled = True + self.pass_debug = 'nopass' not in self.mode_values or bool(self.inherited) + self.profile_xslt = 'xslt' in self.mode_values - if self.pass_debug: - debug_log.debug('%s header will be passed to all requests', DEBUG_HEADER_NAME) - else: - self.enabled = False - self.profile_xslt = False + request_context.set_log_handler(DebugBufferedHandler()) + + if self.pass_debug: + debug_log.debug('%s header will be passed to all requests', DEBUG_HEADER_NAME) + + def auth_failed(self) -> bool: + return self.failed_auth_header is not None def get_frontik_and_apps_versions(application: FrontikApplication) -> etree.Element: diff --git a/frontik/futures.py b/frontik/futures.py index d7151d1e9..cafe7dd66 100644 --- a/frontik/futures.py +++ b/frontik/futures.py @@ -1,8 +1,17 @@ +from __future__ import annotations + import asyncio import logging -from typing import Optional +import time +from functools import partial, wraps +from typing import TYPE_CHECKING, Optional from tornado.concurrent import Future +from tornado.ioloop import IOLoop + +if TYPE_CHECKING: + from collections.abc import Callable + from typing import Any async_logger = logging.getLogger('frontik.futures') @@ -14,33 +23,163 @@ class AbortAsyncGroup(Exception): # AsyncGroup will become legacy in future releases # It will be replaced with FutureGroup class AsyncGroup: - def __init__(self, name: Optional[str] = None) -> None: + """ + Grouping of several async requests and final callback in such way that final callback is invoked + after the last request is finished. + + If any callback throws an exception, all pending callbacks would be aborted and finish_cb + would not be automatically called. + """ + + def __init__(self, finish_cb: Callable, name: Optional[str] = None) -> None: + self._counter = 0 + self._finish_cb: Optional[Callable] = finish_cb self._finished = False self._name = name + self._future: Future = Future() + self._start_time = time.time() self._futures: list[Future] = [] - def add_future(self, future: Future) -> None: + def is_finished(self) -> bool: + return self._finished + + def abort(self) -> None: + async_logger.info('aborting %s', self) + self._finished = True + if not self._future.done(): + self._future.set_exception(AbortAsyncGroup()) + + def finish(self) -> None: if self._finished: - raise RuntimeError('finish group is finished') - self._futures.append(future) + async_logger.warning('trying to finish already finished %s', self) + return None + + self._finished = True + self._future.set_result(None) - async def finish(self) -> None: try: - await asyncio.gather(*self._futures) + if self._finish_cb is not None: + self._finish_cb() finally: - self._finished = True + # prevent possible cycle references + self._finish_cb = None - def done(self) -> bool: - return self._finished + return None - def pending(self) -> bool: - return not self._finished and len(self._futures) != 0 + def try_finish(self) -> None: + if self._counter == 0: + self.finish() - def abort(self) -> None: - for future in self._futures: - if not future.done(): - future.cancel() - self._finished = True + def try_finish_async(self): + """Executes finish_cb in next IOLoop iteration""" + if self._counter == 0: + IOLoop.current().add_callback(self.finish) + + def _inc(self) -> None: + if self._finished: + async_logger.info('ignoring adding callback in %s', self) + raise AbortAsyncGroup() + + self._counter += 1 + + def _dec(self) -> None: + self._counter -= 1 + + def add(self, intermediate_cb: Callable, exception_handler: Optional[Callable] = None) -> Callable: + self._inc() + + @wraps(intermediate_cb) + def new_cb(*args, **kwargs): + if self._finished: + async_logger.info('ignoring executing callback in %s', self) + return + + try: + self._dec() + intermediate_cb(*args, **kwargs) + except Exception as ex: + self.abort() + if exception_handler is not None: + exception_handler(ex) + else: + raise + + self.try_finish() + + return new_cb + + def add_notification(self) -> Callable: + self._inc() + + def new_cb(*args, **kwargs): + self._dec() + self.try_finish() + + return new_cb + + @staticmethod + def _handle_future(callback, future): + future.result() + callback() + + def add_future(self, future: Future) -> Future: + IOLoop.current().add_future(future, partial(self._handle_future, self.add_notification())) + self._futures.append(future) + return future + + def get_finish_future(self) -> Future: + return self._future + + def get_gathering_future(self) -> Future: + return asyncio.gather(*self._futures) def __str__(self): return f'AsyncGroup(name={self._name}, finished={self._finished})' + + +def future_fold( + future: Future, + result_mapper: Optional[Callable] = None, + exception_mapper: Optional[Callable] = None, +) -> Future: + """ + Creates a new future with result or exception processed by result_mapper and exception_mapper. + + If result_mapper or exception_mapper raises an exception, it will be set as an exception for the resulting future. + Any of the mappers can be None — then the result or exception is left as is. + """ + + res_future: Future = Future() + + def _process(func: Optional[Callable], value: Any) -> None: + try: + processed = func(value) if func is not None else value + except Exception as e: + res_future.set_exception(e) + return + res_future.set_result(processed) + + def _on_ready(wrapped_future): + exception = wrapped_future.exception() + if exception is not None: + if not callable(exception_mapper): + + def default_exception_func(error): + raise error + + _process(default_exception_func, exception) + else: + _process(exception_mapper, exception) + else: + _process(result_mapper, future.result()) + + IOLoop.current().add_future(future, callback=_on_ready) + return res_future + + +def future_map(future, func): + return future_fold(future, result_mapper=func) + + +def future_map_exception(future, func): + return future_fold(future, exception_mapper=func) diff --git a/frontik/handler.py b/frontik/handler.py index 25f02fe78..ed1803cf4 100644 --- a/frontik/handler.py +++ b/frontik/handler.py @@ -1,40 +1,41 @@ from __future__ import annotations import asyncio -import datetime import http.client -import json import logging import re -import sys import time from asyncio import Task from asyncio.futures import Future +from functools import wraps +from http import HTTPStatus from typing import TYPE_CHECKING, Any, Optional, Type, TypeVar, Union, overload -from fastapi import Depends, HTTPException, Request, Response +import tornado.web +from fastapi import Depends, Request +from fastapi.dependencies.utils import solve_dependencies from fastapi.routing import APIRoute from http_client.request_response import USER_AGENT_HEADER, FailFastError, RequestBuilder, RequestResult from pydantic import BaseModel, ValidationError -from starlette.datastructures import Headers, QueryParams -from tornado.httputil import format_timestamp, parse_body_arguments +from tornado import httputil +from tornado.httputil import HTTPHeaders, HTTPServerRequest +from tornado.ioloop import IOLoop +from tornado.web import Finish, RequestHandler import frontik.auth -import frontik.handler_active_limit import frontik.producers.json_producer import frontik.producers.xml_producer import frontik.util from frontik import media_types, request_context from frontik.auth import DEBUG_AUTH_HEADER_NAME -from frontik.debug import DEBUG_HEADER_NAME, DebugMode, DebugTransform +from frontik.debug import DEBUG_HEADER_NAME, DebugMode from frontik.futures import AbortAsyncGroup, AsyncGroup -from frontik.http_status import ALLOWED_STATUSES +from frontik.http_status import ALLOWED_STATUSES, CLIENT_CLOSED_REQUEST, NON_CRITICAL_BAD_GATEWAY from frontik.json_builder import FrontikJsonDecodeError, json_decode from frontik.loggers import CUSTOM_JSON_EXTRA, JSON_REQUESTS_LOGGER from frontik.loggers.stages import StagesLogger -from frontik.options import options from frontik.timeout_tracking import get_timeout_checker -from frontik.util import make_url +from frontik.util import gather_dict, make_url from frontik.validator import BaseValidationModel, Validators from frontik.version import version as frontik_version @@ -52,43 +53,28 @@ def __init__(self, wait_finish_group: bool = False) -> None: self.wait_finish_group = wait_finish_group -class HTTPErrorWithPostprocessors(HTTPException): +class HTTPErrorWithPostprocessors(tornado.web.HTTPError): pass -class TypedArgumentError(HTTPException): +class TypedArgumentError(tornado.web.HTTPError): pass -class JSONBodyParseError(HTTPException): +class JSONBodyParseError(tornado.web.HTTPError): def __init__(self) -> None: super().__init__(400, 'Failed to parse json in request body') -class DefaultValueError(HTTPException): - def __init__(self, arg_name: str) -> None: - super().__init__(400, 'Missing argument %s' % arg_name) - self.arg_name = arg_name - - -class FinishPageSignal(Exception): - def __init__(self, data: Any = None, *args: object) -> None: - super().__init__(*args) - self.data = data - - -class RedirectPageSignal(Exception): - def __init__(self, url: str, status: int, *args: object) -> None: +class DefaultValueError(Exception): + def __init__(self, *args: object) -> None: super().__init__(*args) - self.url = url - self.status = status _ARG_DEFAULT = object() MEDIA_TYPE_PARAMETERS_SEPARATOR_RE = r' *; *' OUTER_TIMEOUT_MS_HEADER = 'X-Outer-Timeout-Ms' _remove_control_chars_regex = re.compile(r'[\x00-\x08\x0e-\x1f]') -_T = TypeVar('_T') handler_logger = logging.getLogger('handler') @@ -105,81 +91,59 @@ def _fail_fast_policy(fail_fast: bool, waited: bool, host: str, path: str) -> bo return fail_fast -class PageHandler: +class PageHandler(RequestHandler): def __init__( self, application: FrontikApplication, - query_params: QueryParams, - cookie_params: dict[str, str], - header_params: Headers, - body_bytes: bytes, - request_start_time: float, - path: str, - path_params: dict, - remote_ip: str, - method: str, - ) -> None: # request: Request - self.application = application - self.query_params = query_params - self.cookie_params = cookie_params or {} - self.header_params: Headers = header_params - self.body_bytes = body_bytes - self.request_start_time = request_start_time - self.path = path - self.path_params = path_params - self.remote_ip = remote_ip - self.method = method - - self._json_body = None - self.body_arguments: dict[str, Any] = {} - self.files: dict = {} - self.parse_body_bytes() - + request: HTTPServerRequest, + route: APIRoute, + debug_mode: DebugMode, + path_params: dict[str, str], + ) -> None: + self.name = self.__class__.__name__ self.request_id: str = request_context.get_request_id() # type: ignore self.config = application.config self.log = handler_logger self.text: Any = None - self._finished = False + self.route = route + self.debug_mode = debug_mode + self.path_params = path_params + + super().__init__(application, request) # type: ignore - self.statsd_client: StatsDClient | StatsDClientStub + self.statsd_client: StatsDClient | StatsDClientStub = application.statsd_client for integration in application.available_integrations: integration.initialize_handler(self) - self.stages_logger = StagesLogger(request_start_time, self.statsd_client) + self.stages_logger = StagesLogger(request._start_time, self.statsd_client) - self._debug_access: Optional[bool] = None self._render_postprocessors: list = [] self._postprocessors: list = [] + self._mandatory_cookies: dict = {} + self._mandatory_headers = httputil.HTTPHeaders() + self._validation_model: type[BaseValidationModel | BaseModel] = BaseValidationModel self.timeout_checker = None - self.use_adaptive_strategy = False - outer_timeout = header_params.get(OUTER_TIMEOUT_MS_HEADER) + outer_timeout = request.headers.get(OUTER_TIMEOUT_MS_HEADER) if outer_timeout: self.timeout_checker = get_timeout_checker( - header_params.get(USER_AGENT_HEADER), + request.headers.get(USER_AGENT_HEADER), float(outer_timeout), - request_start_time, + request._start_time, ) - self._status = 200 - self._reason: Optional[str] = None + self.handler_result_future: Future[tuple[int, str, HTTPHeaders, bytes]] = Future() def __repr__(self): return f'{self.__module__}.{self.__class__.__name__}' def prepare(self) -> None: - self.resp_headers = get_default_headers() - self.resp_cookies: dict[str, dict] = {} - - self.finish_group = AsyncGroup(name='finish') - - self.active_limit = frontik.handler_active_limit.ActiveHandlersLimit(self.statsd_client) - - self.debug_mode = DebugMode(self) + self.application: FrontikApplication # type: ignore + self.finish_group = AsyncGroup(lambda: None, name='finish') self.json_producer = self.application.json.get_producer(self) self.json = self.json_producer.json @@ -190,15 +154,23 @@ def prepare(self) -> None: self._http_client: HttpClient = self.application.http_client_factory.get_http_client( self.modify_http_client_request, self.debug_mode.enabled, - self.use_adaptive_strategy, ) - # Simple getters and setters + self._handler_finished_notification = self.finish_group.add_notification() + + super().prepare() - def get_request_headers(self) -> Headers: - return self.header_params + def set_default_headers(self): + self._headers = httputil.HTTPHeaders({ + 'Server': f'Frontik/{frontik_version}', + 'X-Request-Id': self.request_id, + }) + + @property + def path(self) -> str: + return self.request.path - def get_path_argument(self, name, default=_ARG_DEFAULT): + def get_path_argument(self, name: str, default: Any = _ARG_DEFAULT) -> str: value = self.path_params.get(name, None) if value is None: if default is _ARG_DEFAULT: @@ -207,91 +179,49 @@ def get_path_argument(self, name, default=_ARG_DEFAULT): value = _remove_control_chars_regex.sub(' ', value) return value - def get_query_argument( - self, - name: str, - default: Union[str, _T] = _ARG_DEFAULT, # type: ignore - strip: bool = True, - ) -> Union[str, _T]: - args = self._get_arguments(name, strip=strip) - if not args: - if default is _ARG_DEFAULT: - raise DefaultValueError(name) - return default - return args[-1] - - def get_query_arguments(self, name: Optional[str] = None, strip: bool = True) -> Union[list[str], dict[str, str]]: - if name is None: - return self._get_all_query_arguments(strip) - return self._get_arguments(name, strip) - - def _get_all_query_arguments(self, strip: bool = True) -> dict[str, str]: - qargs_list = self.query_params.multi_items() - values = {} - for qarg_k, qarg_v in qargs_list: - v = _remove_control_chars_regex.sub(' ', qarg_v) - if strip: - v = v.strip() - values[qarg_k] = v - - return values - - def _get_arguments(self, name: str, strip: bool = True) -> list[str]: - qargs_list = self.query_params.multi_items() - values = [] - for qarg_k, qarg_v in qargs_list: - if qarg_k != name: - continue - - # Get rid of any weird control chars (unless decoding gave - # us bytes, in which case leave it alone) - v = _remove_control_chars_regex.sub(' ', qarg_v) - if strip: - v = v.strip() - values.append(v) - - return values + @overload + def get_header(self, param_name: str, default: None = None) -> Optional[str]: ... - def get_str_argument( - self, - name: str, - default: Any = _ARG_DEFAULT, - path_safe: bool = True, - **kwargs: Any, - ) -> Optional[Union[str, list[str]]]: - if path_safe: - return self.get_validated_argument(name, Validators.PATH_SAFE_STRING, default=default, **kwargs) - return self.get_validated_argument(name, Validators.STRING, default=default, **kwargs) + @overload + def get_header(self, param_name: str, default: str) -> str: ... - def get_int_argument( - self, - name: str, - default: Any = _ARG_DEFAULT, - **kwargs: Any, - ) -> Optional[Union[int, list[int]]]: - return self.get_validated_argument(name, Validators.INTEGER, default=default, **kwargs) + def get_header(self, param_name: str, default: Optional[str] = None) -> Optional[str]: + return self.request.headers.get(param_name.lower(), default) - def get_bool_argument( - self, - name: str, - default: Any = _ARG_DEFAULT, - **kwargs: Any, - ) -> Optional[Union[bool, list[bool]]]: - return self.get_validated_argument(name, Validators.BOOLEAN, default=default, **kwargs) + def decode_argument(self, value: bytes, name: Optional[str] = None) -> str: + try: + return super().decode_argument(value, name) + except (UnicodeError, tornado.web.HTTPError): + self.log.warning('cannot decode utf-8 query parameter, trying other charsets') - def get_float_argument( - self, - name: str, - default: Any = _ARG_DEFAULT, - **kwargs: Any, - ) -> Optional[Union[float, list[float]]]: - return self.get_validated_argument(name, Validators.FLOAT, default=default, **kwargs) + try: + return frontik.util.decode_string_from_charset(value) + except UnicodeError: + self.log.exception('cannot decode argument, ignoring invalid chars') + return value.decode('utf-8', 'ignore') + + def get_body_argument(self, name: str, default: Any = _ARG_DEFAULT, strip: bool = True) -> Optional[str]: + if self._get_request_mime_type(self.request) == media_types.APPLICATION_JSON: + if name not in self.json_body and default == _ARG_DEFAULT: + raise tornado.web.MissingArgumentError(name) + + result = self.json_body.get(name, default) + + if strip and isinstance(result, str): + return result.strip() + + return result + + if default == _ARG_DEFAULT: + return super().get_body_argument(name, strip=strip) + return super().get_body_argument(name, default, strip) def set_validation_model(self, model: type[Union[BaseValidationModel, BaseModel]]) -> None: if issubclass(model, BaseModel): self._validation_model = model else: - raise TypeError('model is not subclass of BaseClass') + msg = 'model is not subclass of BaseClass' + raise TypeError(msg) def get_validated_argument( self, @@ -318,9 +248,9 @@ def get_validated_argument( elif from_body: value = self.get_body_argument(name, validated_default, strip) elif array: - value = self.get_query_arguments(name, strip) + value = self.get_arguments(name, strip) else: - value = self.get_query_argument(name, validated_default, strip) + value = self.get_argument(name, validated_default, strip) try: params = {validator: value} @@ -332,238 +262,193 @@ def get_validated_argument( return validated_value - def get_body_arguments( - self, name: Optional[str] = None, strip: bool = True - ) -> Union[list[str], dict[str, list[str]]]: - if name is None: - return self._get_all_body_arguments(strip) - return self._get_body_arguments(name, strip) - - def _get_all_body_arguments(self, strip: bool) -> dict[str, list[str]]: - result: dict[str, list[str]] = {} - for key, values in self.body_arguments.items(): - result[key] = [] - for v in values: - s = self.decode_argument(v) - if isinstance(s, str): - s = _remove_control_chars_regex.sub(' ', s) - if strip: - s = s.strip() - result[key].append(s) - return result - - def get_body_argument(self, name: str, default: Any = _ARG_DEFAULT, strip: bool = True) -> Optional[str]: - if self._get_request_mime_type() == media_types.APPLICATION_JSON: - if name not in self.json_body and default is _ARG_DEFAULT: - raise DefaultValueError(name) - - result = self.json_body.get(name, default) - - if strip and isinstance(result, str): - return result.strip() + def get_str_argument( + self, + name: str, + default: Any = _ARG_DEFAULT, + path_safe: bool = True, + **kwargs: Any, + ) -> Optional[Union[str, list[str]]]: + if path_safe: + return self.get_validated_argument(name, Validators.PATH_SAFE_STRING, default=default, **kwargs) + return self.get_validated_argument(name, Validators.STRING, default=default, **kwargs) - return result + def get_int_argument( + self, + name: str, + default: Any = _ARG_DEFAULT, + **kwargs: Any, + ) -> Optional[Union[int, list[int]]]: + return self.get_validated_argument(name, Validators.INTEGER, default=default, **kwargs) - if default is _ARG_DEFAULT: - return self._get_body_argument(name, strip=strip) - return self._get_body_argument(name, default, strip) + def get_bool_argument( + self, + name: str, + default: Any = _ARG_DEFAULT, + **kwargs: Any, + ) -> Optional[Union[bool, list[bool]]]: + return self.get_validated_argument(name, Validators.BOOLEAN, default=default, **kwargs) - def _get_body_argument( + def get_float_argument( self, name: str, default: Any = _ARG_DEFAULT, - strip: bool = True, - ) -> Optional[str]: - args = self._get_body_arguments(name, strip=strip) - if not args: - if default is _ARG_DEFAULT: - raise DefaultValueError(name) - return default - return args[-1] - - def _get_body_arguments(self, name: str, strip: bool = True) -> list[str]: - values = [] - for v in self.body_arguments.get(name, []): - s = self.decode_argument(v, name=name) - if isinstance(s, str): - s = _remove_control_chars_regex.sub(' ', s) - if strip: - s = s.strip() - values.append(s) - return values - - def parse_body_bytes(self) -> None: - if self._get_request_mime_type() == media_types.APPLICATION_JSON: - return - else: - parse_body_arguments( - self.get_header('Content-Type', ''), - self.body_bytes, - self.body_arguments, - self.files, - self.header_params, # type: ignore - ) + **kwargs: Any, + ) -> Optional[Union[float, list[float]]]: + return self.get_validated_argument(name, Validators.FLOAT, default=default, **kwargs) + + def _get_request_mime_type(self, request: HTTPServerRequest) -> str: + content_type = request.headers.get('Content-Type', '') + return re.split(MEDIA_TYPE_PARAMETERS_SEPARATOR_RE, content_type)[0] + + def set_status(self, status_code: int, reason: Optional[str] = None) -> None: + status_code = status_code if status_code in ALLOWED_STATUSES else http.client.SERVICE_UNAVAILABLE + super().set_status(status_code, reason=reason) + + def redirect(self, url: str, *args: Any, allow_protocol_relative: bool = False, **kwargs: Any) -> None: + if not allow_protocol_relative and url.startswith('//'): + # A redirect with two initial slashes is a "protocol-relative" URL. + # This means the next path segment is treated as a hostname instead + # of a part of the path, making this effectively an open redirect. + # Reject paths starting with two slashes to prevent this. + # This is only reachable under certain configurations. + raise tornado.web.HTTPError(403, 'cannot redirect path with two initial slashes') + self.log.info('redirecting to: %s', url) + return super().redirect(url, *args, **kwargs) @property def json_body(self): - if self._json_body is None: + if not hasattr(self, '_json_body'): self._json_body = self._get_json_body() return self._json_body def _get_json_body(self) -> Any: try: - return json_decode(self.body_bytes) + return json_decode(self.request.body) except FrontikJsonDecodeError as _: raise JSONBodyParseError() - def decode_argument(self, value: bytes, name: Optional[str] = None) -> str: - try: - return value.decode('utf-8') - except UnicodeError: - self.log.warning('cannot decode utf-8 body parameter %s, trying other charsets', name) + @classmethod + def add_callback(cls, callback: Callable, *args: Any, **kwargs: Any) -> None: + IOLoop.current().add_callback(callback, *args, **kwargs) - try: - return frontik.util.decode_string_from_charset(value) - except UnicodeError: - self.log.exception('cannot decode body parameter %s, ignoring invalid chars', name) - return value.decode('utf-8', 'ignore') + @classmethod + def add_timeout(cls, deadline: float, callback: Callable, *args: Any, **kwargs: Any) -> Any: + return IOLoop.current().add_timeout(deadline, callback, *args, **kwargs) - @overload - def get_header(self, param_name: str, default: None = None) -> Optional[str]: ... + @staticmethod + def remove_timeout(timeout): + IOLoop.current().remove_timeout(timeout) - @overload - def get_header(self, param_name: str, default: str) -> str: ... + @classmethod + def add_future(cls, future: Future, callback: Callable) -> None: + IOLoop.current().add_future(future, callback) - def get_header(self, param_name: str, default: Optional[str] = None) -> Optional[str]: - return self.header_params.get(param_name.lower(), default) + # Requests handling - def set_header(self, k: str, v: str) -> None: - self.resp_headers[k] = v + async def my_execute(self) -> tuple[int, str, HTTPHeaders, bytes]: + try: + await super()._execute([], b'', b'') + except Exception as ex: + self._handle_request_exception(ex) + return await self.handler_result_future # status, reason, headers, chunk - def _get_request_mime_type(self) -> str: - content_type = self.get_header('Content-Type', '') - return re.split(MEDIA_TYPE_PARAMETERS_SEPARATOR_RE, content_type)[0] + async def get(self, *args, **kwargs): + await self._execute_page() - def clear_header(self, name: str) -> None: - if name in self.resp_headers: - del self.resp_headers[name] + async def post(self, *args, **kwargs): + await self._execute_page() - def clear_cookie(self, name: str, path: str = '/', domain: Optional[str] = None) -> None: - expires = datetime.datetime.now() - datetime.timedelta(days=365) - self.set_cookie(name, value='', expires=expires, path=path, domain=domain) + async def put(self, *args, **kwargs): + await self._execute_page() - def get_cookie(self, param_name: str, default: Optional[str]) -> Optional[str]: - return self.cookie_params.get(param_name, default) + async def delete(self, *args, **kwargs): + await self._execute_page() - def set_cookie( - self, - name: str, - value: Union[str, bytes], - domain: Optional[str] = None, - expires: Optional[Union[float, tuple, datetime.datetime]] = None, - path: str = '/', - expires_days: Optional[float] = None, - # Keyword-only args start here for historical reasons. - *, - max_age: Optional[int] = None, - httponly: bool = False, - secure: bool = False, - samesite: Optional[str] = None, - ) -> None: - name = str(name) - value = str(value) - if re.search(r'[\x00-\x20]', name + value): - # Don't let us accidentally inject bad stuff - raise ValueError('Invalid cookie %s: %s', name, value) - - if name in self.resp_cookies: - del self.resp_cookies[name] - self.resp_cookies[name] = {'value': value} - morsel = self.resp_cookies[name] - if domain: - morsel['domain'] = domain - if expires_days is not None and not expires: - expires = datetime.datetime.now() + datetime.timedelta(days=expires_days) - if expires: - morsel['expires'] = format_timestamp(expires) - if path: - morsel['path'] = path - if max_age: - # Note change from _ to -. - morsel['max_age'] = str(max_age) - if httponly: - # Note that SimpleCookie ignores the value here. The presense of an - # httponly (or secure) key is treated as true. - morsel['httponly'] = True - if secure: - morsel['secure'] = True - if samesite: - morsel['samesite'] = samesite + async def head(self, *args, **kwargs): + await self._execute_page() - # Requests handling + def options(self, *args, **kwargs): + self.return_405() - def require_debug_access(self, login: Optional[str] = None, passwd: Optional[str] = None) -> None: - if self._debug_access is None: - if options.debug: - debug_access = True - else: - check_login = login if login is not None else options.debug_login - check_passwd = passwd if passwd is not None else options.debug_password - frontik.auth.check_debug_auth(self, check_login, check_passwd) - debug_access = True + async def _execute_page(self) -> None: + self.stages_logger.commit_stage('prepare') - self._debug_access = debug_access + f_request = Request({ + 'type': 'http', + 'query_string': '', + 'headers': '', + 'handler': self, + }) - def set_status(self, status_code: int, reason: Optional[str] = None) -> None: - status_code = status_code if status_code in ALLOWED_STATUSES else http.client.SERVICE_UNAVAILABLE + values, errors, _, _, _ = await solve_dependencies( + request=f_request, dependant=self.route.dependant, body=None, dependency_overrides_provider=None + ) + if errors: + raise RuntimeError(f'dependency solving failed: {errors}') - self._status = status_code - self._reason = reason + assert self.route.dependant.call is not None + await self.route.dependant.call(**values) - def get_status(self) -> int: - return self._status + self._handler_finished_notification() + await self.finish_group.get_gathering_future() + await self.finish_group.get_finish_future() - def redirect(self, url: str, permanent: bool = False, status: Optional[int] = None) -> None: - if url.startswith('//'): - raise RuntimeError('403 cannot redirect path with two initial slashes') - self.log.info('redirecting to: %s', url) - if status is None: - status = 301 if permanent else 302 - else: - assert isinstance(status, int) - assert 300 <= status <= 399 - raise RedirectPageSignal(url, status) + render_result = await self._postprocess() + if render_result is not None: + self.write(render_result) - def finish(self, data: Optional[Union[str, bytes, dict]] = None) -> None: - raise FinishPageSignal(data) + def return_405(self) -> None: + allowed_methods = [name for name in ('get', 'post', 'put', 'delete') if f'{name}_page' in vars(self.__class__)] + self.set_header('Allow', ', '.join(allowed_methods)) + self.set_status(405) + self.finish() - async def get_page_fail_fast(self, request_result: RequestResult) -> tuple[int, dict, Any]: - return await self.__return_error(request_result.status_code, error_info={'is_fail_fast': True}) + def get_page_fail_fast(self, request_result: RequestResult) -> None: + self.__return_error(request_result.status_code, error_info={'is_fail_fast': True}) - async def post_page_fail_fast(self, request_result: RequestResult) -> tuple[int, dict, Any]: - return await self.__return_error(request_result.status_code, error_info={'is_fail_fast': True}) + def post_page_fail_fast(self, request_result: RequestResult) -> None: + self.__return_error(request_result.status_code, error_info={'is_fail_fast': True}) - async def put_page_fail_fast(self, request_result: RequestResult) -> tuple[int, dict, Any]: - return await self.__return_error(request_result.status_code, error_info={'is_fail_fast': True}) + def put_page_fail_fast(self, request_result: RequestResult) -> None: + self.__return_error(request_result.status_code, error_info={'is_fail_fast': True}) - async def delete_page_fail_fast(self, request_result: RequestResult) -> tuple[int, dict, Any]: - return await self.__return_error(request_result.status_code, error_info={'is_fail_fast': True}) + def delete_page_fail_fast(self, request_result: RequestResult) -> None: + self.__return_error(request_result.status_code, error_info={'is_fail_fast': True}) - async def __return_error(self, response_code: int, **kwargs: Any) -> tuple[int, dict, Any]: - return await self.send_error(response_code if 300 <= response_code < 500 else 502, **kwargs) + def __return_error(self, response_code: int, **kwargs: Any) -> None: + if not (300 <= response_code < 500 or response_code == NON_CRITICAL_BAD_GATEWAY): + response_code = HTTPStatus.BAD_GATEWAY + self.send_error(response_code, **kwargs) # Finish page def is_finished(self) -> bool: return self._finished - async def finish_with_postprocessors(self) -> tuple[int, dict, Any]: - if self.finish_group.pending(): - self.log.error('finish_with_postprocessors before finish group done') + def check_finished(self, callback: Callable) -> Callable: + @wraps(callback) + def wrapper(*args, **kwargs): + if self.is_finished(): + self.log.warning('page was already finished, %s ignored', callback) + else: + return callback(*args, **kwargs) + + return wrapper + + def finish_with_postprocessors(self) -> None: + if not self.finish_group.get_finish_future().done(): self.finish_group.abort() - content = await self._postprocess() - return self.get_status(), self.resp_headers, content + def _cb(future: Future) -> None: + if (ex := future.exception()) is not None: + self.log.error('postprocess failed %s', ex) + self.set_status(500) + self.finish() + if future.result() is not None: + self.finish(future.result()) + + asyncio.create_task(self._postprocess()).add_done_callback(_cb) def run_task(self: PageHandler, coro: Coroutine) -> Task: task = asyncio.create_task(coro) @@ -600,42 +485,40 @@ async def _postprocess(self) -> Any: ) return postprocessed_result - def on_finish(self, status: int) -> None: - self.stages_logger.commit_stage('flush') - self.stages_logger.flush_stages(status) - - async def handle_request_exception(self, ex: BaseException) -> tuple[int, dict, Any]: - if isinstance(ex, FinishPageSignal): - chunk = _data_to_chunk(ex.data, self.resp_headers) - return self.get_status(), self.resp_headers, chunk + def on_connection_close(self): + with request_context.request_context(self.request_id): + super().on_connection_close() - if isinstance(ex, RedirectPageSignal): - self.set_header('Location', ex.url) - return ex.status, self.resp_headers, None + self.finish_group.abort() + self.set_status(CLIENT_CLOSED_REQUEST, 'Client closed the connection: aborting request') - if isinstance(ex, FinishWithPostprocessors): - if ex.wait_finish_group: - await self.finish_group.finish() - return await self.finish_with_postprocessors() + self.stages_logger.commit_stage('page') + self.stages_logger.flush_stages(self.get_status()) - if isinstance(ex, HTTPErrorWithPostprocessors): - self.set_status(ex.status_code) - return await self.finish_with_postprocessors() + self.finish() - if isinstance(ex, HTTPException): - self.resp_cookies = {} - if ex.headers is None: - ex.headers = {'Content-Type': media_types.TEXT_PLAIN} + def on_finish(self): + self.stages_logger.commit_stage('flush') + self.stages_logger.flush_stages(self.get_status()) - self.log.error('HTTPException with code: %s, reason: %s', ex.status_code, ex.detail) + def _handle_request_exception(self, e: BaseException) -> None: + if isinstance(e, AbortAsyncGroup): + self.log.info('page was aborted, skipping postprocessing') + return - if hasattr(self, 'write_error'): - return await self.write_error(ex.status_code, exc_info=sys.exc_info()) + if isinstance(e, FinishWithPostprocessors): + if e.wait_finish_group: + self._handler_finished_notification() + self.add_future(self.finish_group.get_finish_future(), lambda _: self.finish_with_postprocessors()) + else: + self.finish_with_postprocessors() + return - return build_error_data(ex.status_code, ex.detail) + if self._finished and not isinstance(e, Finish): + return - if isinstance(ex, FailFastError): - request = ex.failed_result.request + if isinstance(e, FailFastError): + request = e.failed_result.request if self.log.isEnabledFor(logging.WARNING): _max_uri_length = 24 @@ -646,37 +529,133 @@ async def handle_request_exception(self, ex: BaseException) -> tuple[int, dict, if request.name: request_name = f'{request_name} ({request.name})' - self.log.error( + self.log.warning( 'FailFastError: request %s failed with %s code', request_name, - ex.failed_result.status_code, + e.failed_result.status_code, ) - error_method_name = f'{self.method.lower()}_page_fail_fast' - method = getattr(self, error_method_name, None) - if callable(method): - return await method(ex.failed_result) - else: - return await self.__return_error(ex.failed_result.status_code, error_info={'is_fail_fast': True}) + try: + error_method_name = f'{self.request.method.lower()}_page_fail_fast' # type: ignore + method = getattr(self, error_method_name, None) + if callable(method): + method(e.failed_result) + else: + self.__return_error(e.failed_result.status_code, error_info={'is_fail_fast': True}) + + except Exception as exc: + super()._handle_request_exception(exc) else: - raise ex + super()._handle_request_exception(e) - async def send_error(self, status_code: int = 500, **kwargs: Any) -> tuple[int, dict, Any]: + def send_error(self, status_code: int = 500, **kwargs: Any) -> None: + """`send_error` is adapted to support `write_error` that can call + `finish` asynchronously. + """ self.stages_logger.commit_stage('page') - self._reason = kwargs.get('reason') + if self._headers_written: + super().send_error(status_code, **kwargs) + return + + reason = kwargs.get('reason') + if 'exc_info' in kwargs: + exception = kwargs['exc_info'][1] + if isinstance(exception, tornado.web.HTTPError) and exception.reason: + reason = exception.reason + else: + exception = None + + if not isinstance(exception, HTTPErrorWithPostprocessors): + self.clear() + + self.set_status(status_code, reason=reason) + + try: + self.write_error(status_code, **kwargs) + except Exception: + self.log.exception('Uncaught exception in write_error') + if not self._finished: + self.finish() + + def write_error(self, status_code: int = 500, **kwargs: Any) -> None: + """ + `write_error` can call `finish` asynchronously if HTTPErrorWithPostprocessors is raised. + """ + exception = kwargs['exc_info'][1] if 'exc_info' in kwargs else None + + if isinstance(exception, HTTPErrorWithPostprocessors): + self.finish_with_postprocessors() + return + + self.set_header('Content-Type', media_types.TEXT_HTML) + super().write_error(status_code, **kwargs) + + def finish(self, chunk: Optional[Union[str, bytes, dict]] = None) -> Future[None]: + self.stages_logger.commit_stage('postprocess') + for name, value in self._mandatory_headers.items(): + self.set_header(name, value) + + for args, kwargs in self._mandatory_cookies.values(): + try: + self.set_cookie(*args, **kwargs) + except ValueError: + self.set_status(http.client.BAD_REQUEST) + + if self._status_code in (204, 304) or (100 <= self._status_code < 200): + self._write_buffer = [] + chunk = None + + finish_future = super().finish(chunk) + return finish_future + + def flush(self, include_footers: bool = False) -> Future[None]: + assert self.request.connection is not None + chunk = b''.join(self._write_buffer) + self._write_buffer = [] + self._headers_written = True + + if self.request.method == 'HEAD': + chunk = b'' - self.set_status(status_code, reason=self._reason) - return build_error_data(status_code, self._reason) + if hasattr(self, '_new_cookie'): + for cookie in self._new_cookie.values(): + self.add_header('Set-Cookie', cookie.OutputString(None)) - def cleanup(self) -> None: - self._finished = True - if hasattr(self, 'active_limit'): - self.active_limit.release() + self.handler_result_future.set_result((self._status_code, self._reason, self._headers, chunk)) + + future = Future() # type: Future[None] + future.set_result(None) + return future # postprocessors + def set_mandatory_header(self, name: str, value: str) -> None: + self._mandatory_headers[name] = value + + def set_mandatory_cookie( + self, + name: str, + value: str, + domain: Optional[str] = None, + expires: Optional[str] = None, + path: str = '/', + expires_days: Optional[int] = None, + **kwargs: Any, + ) -> None: + self._mandatory_cookies[name] = ((name, value, domain, expires, path, expires_days), kwargs) + + def clear_header(self, name: str) -> None: + if name in self._mandatory_headers: + del self._mandatory_headers[name] + super().clear_header(name) + + def clear_cookie(self, name: str, path: str = '/', domain: Optional[str] = None) -> None: # type: ignore + if name in self._mandatory_cookies: + del self._mandatory_cookies[name] + super().clear_cookie(name, path=path, domain=domain) + async def _run_postprocessors(self, postprocessors: list) -> bool: for p in postprocessors: if asyncio.iscoroutinefunction(p): @@ -714,7 +693,7 @@ def add_postprocessor(self, postprocessor: Any) -> None: async def _generic_producer(self): self.log.debug('finishing plaintext') - if self.resp_headers.get('Content-Type') is None: + if self._headers.get('Content-Type') is None: self.set_header('Content-Type', media_types.TEXT_HTML) return self.text, None @@ -732,7 +711,6 @@ def set_template(self, filename: str) -> None: def modify_http_client_request(self, balanced_request: RequestBuilder) -> None: balanced_request.headers['x-request-id'] = request_context.get_request_id() - balanced_request.headers[OUTER_TIMEOUT_MS_HEADER] = f'{balanced_request.request_timeout * 1000:.0f}' if self.timeout_checker is not None: @@ -745,10 +723,13 @@ def modify_http_client_request(self, balanced_request: RequestBuilder) -> None: balanced_request.path = make_url(balanced_request.path, debug_timestamp=int(time.time())) for header_name in ('Authorization', DEBUG_AUTH_HEADER_NAME): - authorization = self.get_header(header_name) + authorization = self.request.headers.get(header_name) if authorization is not None: balanced_request.headers[header_name] = authorization + def group(self, futures: dict) -> Task: + return self.run_task(gather_dict(coro_dict=futures)) + def get_url( self, host: str, @@ -967,7 +948,7 @@ def _execute_http_client_method( client_method: Callable, waited: bool, ) -> Future[RequestResult]: - if waited and (self.is_finished() or self.finish_group.done()): + if waited and (self.is_finished() or self.finish_group.is_finished()): handler_logger.info( 'attempted to make waited http request to %s %s in finished handler, ignoring', host, @@ -975,7 +956,7 @@ def _execute_http_client_method( ) future: Future = Future() - future.set_exception(AbortAsyncGroup('attempted to make waited http request is finished handler')) + future.set_exception(AbortAsyncGroup()) return future future = client_method() @@ -985,22 +966,23 @@ def _execute_http_client_method( return future - def log_request(self, request: Request) -> None: - request_time = int(1000.0 * (time.time() - self.request_start_time)) - extra = { - 'ip': request.client.host if request.client else None, - 'rid': request_context.get_request_id(), - 'status': self.get_status(), - 'time': request_time, - 'method': request.method, - 'uri': request.url.path, - } - handler_name = request_context.get_handler_name() - if handler_name: - extra['controller'] = handler_name +def log_request(tornado_request: httputil.HTTPServerRequest, status_code: int) -> None: + request_time = int(1000.0 * tornado_request.request_time()) + extra = { + 'ip': tornado_request.remote_ip, + 'rid': request_context.get_request_id(), + 'status': status_code, + 'time': request_time, + 'method': tornado_request.method, + 'uri': tornado_request.uri, + } + + handler_name = request_context.get_handler_name() + if handler_name: + extra['controller'] = handler_name - JSON_REQUESTS_LOGGER.info('', extra={CUSTOM_JSON_EXTRA: extra}) + JSON_REQUESTS_LOGGER.info('', extra={CUSTOM_JSON_EXTRA: extra}) PageHandlerT = TypeVar('PageHandlerT', bound=PageHandler) @@ -1008,7 +990,7 @@ def log_request(self, request: Request) -> None: def get_current_handler(_: Union[PageHandlerT, Type[PageHandler]] = PageHandler) -> PageHandlerT: async def handler_getter(request: Request) -> PageHandlerT: - return request.state.handler + return request['handler'] return Depends(handler_getter) @@ -1019,84 +1001,3 @@ def get_default_headers() -> dict[str, str]: 'Server': f'Frontik/{frontik_version}', 'X-Request-Id': request_id, } - - -def build_error_data(status_code: int = 500, message: Optional[str] = 'Internal Server Error') -> tuple[int, dict, Any]: - headers = get_default_headers() - headers['Content-Type'] = media_types.TEXT_HTML - content = f'{status_code}: {message}{status_code}: {message}' - return status_code, headers, content - - -def _data_to_chunk(data: Any, headers: dict) -> bytes: - result: bytes = b'' - if data is None: - return result - if isinstance(data, str): - result = data.encode('utf-8') - elif isinstance(data, dict): - chunk = json.dumps(data).replace(' Response: - handler: PageHandler = request.state.handler - - try: - request_context.set_handler_name(f'{route.endpoint.__module__}.{route.endpoint.__name__}') - - handler.prepare() - handler.stages_logger.commit_stage('prepare') - _response = await call_next(request) - - await handler.finish_group.finish() - handler.stages_logger.commit_stage('page') - - content = await handler._postprocess() - headers = handler.resp_headers - status = handler.get_status() - - handler.stages_logger.commit_stage('postprocess') - - except Exception as ex: - try: - status, headers, content = await handler.handle_request_exception(ex) - except Exception as exc: - handler_logger.error('request processing has failed: %s', exc) - if getattr(handler, '_debug_enabled', False): - status, headers, content = build_error_data() - elif hasattr(handler, 'write_error'): - status, headers, content = await handler.write_error(exc_info=sys.exc_info()) - else: - raise - - finally: - handler.cleanup() - - if status in (204, 304) or (100 <= status < 200): - for h in ('Content-Encoding', 'Content-Language', 'Content-Type'): - if h in headers: - headers.pop(h) - content = None - - if getattr(handler, '_debug_enabled', False): - chunk = _data_to_chunk(content, headers) - debug_transform = DebugTransform(request.app.frontik_app, request) - status, headers, content = debug_transform.transform_chunk(status, headers, chunk) - - response = Response(status_code=status, headers=headers, content=content) - - for key, values in handler.resp_cookies.items(): - response.set_cookie(key, **values) - - handler.finish_group.abort() - handler.log_request(request) - handler.on_finish(status) - - return response diff --git a/frontik/handler_active_limit.py b/frontik/handler_active_limit.py index 0f45e1561..889346bb7 100644 --- a/frontik/handler_active_limit.py +++ b/frontik/handler_active_limit.py @@ -1,9 +1,8 @@ from __future__ import annotations import logging -from typing import TYPE_CHECKING - -from fastapi import HTTPException +from contextlib import contextmanager +from typing import TYPE_CHECKING, Iterator, Union from frontik.options import options @@ -18,14 +17,13 @@ class ActiveHandlersLimit: high_watermark_ratio = 0.75 def __init__(self, statsd_client: StatsDClient | StatsDClientStub) -> None: - self._acquired = False + self.acquired = False self._statsd_client = statsd_client self._high_watermark = int(options.max_active_handlers * self.high_watermark_ratio) if ActiveHandlersLimit.count > options.max_active_handlers: handlers_count_logger.warning('dropping request: too many active handlers (%s)', ActiveHandlersLimit.count) - - raise HTTPException(503) + return elif ActiveHandlersLimit.count > self._high_watermark: handlers_count_logger.warning( @@ -38,13 +36,22 @@ def __init__(self, statsd_client: StatsDClient | StatsDClientStub) -> None: self.acquire() def acquire(self) -> None: - if not self._acquired: + if not self.acquired: ActiveHandlersLimit.count += 1 - self._acquired = True + self.acquired = True self._statsd_client.gauge('handler.active_count', ActiveHandlersLimit.count) def release(self) -> None: - if self._acquired: + if self.acquired: ActiveHandlersLimit.count -= 1 - self._acquired = False + self.acquired = False self._statsd_client.gauge('handler.active_count', ActiveHandlersLimit.count) + + +@contextmanager +def request_limiter(statsd_client: Union[StatsDClient, StatsDClientStub]) -> Iterator: + active_limit = ActiveHandlersLimit(statsd_client) + try: + yield active_limit.acquired + finally: + active_limit.release() diff --git a/frontik/handler_asgi.py b/frontik/handler_asgi.py new file mode 100644 index 000000000..4d9fb5c79 --- /dev/null +++ b/frontik/handler_asgi.py @@ -0,0 +1,193 @@ +from __future__ import annotations + +import http.client +import logging +from typing import TYPE_CHECKING, Any, Callable, Optional + +from fastapi.routing import APIRoute +from tornado import httputil +from tornado.httputil import HTTPHeaders + +from frontik import media_types, request_context +from frontik.debug import DebugMode, DebugTransform +from frontik.handler import PageHandler, get_default_headers, log_request +from frontik.handler_active_limit import request_limiter +from frontik.json_builder import JsonBuilder +from frontik.routing import find_route, get_allowed_methods + +if TYPE_CHECKING: + from frontik.app import FrontikApplication, FrontikAsgiApp + +CHARSET = 'utf-8' +log = logging.getLogger('handler') + + +async def execute_page( + frontik_app: FrontikApplication, tornado_request: httputil.HTTPServerRequest, request_id: str, app: FrontikAsgiApp +) -> None: + with request_context.request_context(request_id), request_limiter(frontik_app.statsd_client) as accepted: + log.info('requested url: %s', tornado_request.uri) + tornado_request.request_id = request_id # type: ignore + assert tornado_request.method is not None + route, page_cls, path_params = find_route(tornado_request.path, tornado_request.method) + + debug_mode = DebugMode(tornado_request) + data: bytes + + if not accepted: + status, reason, headers, data = make_not_accepted_response() + elif debug_mode.auth_failed(): + assert debug_mode.failed_auth_header is not None + status, reason, headers, data = make_debug_auth_failed_response(debug_mode.failed_auth_header) + elif route is None: + status, reason, headers, data = make_not_found_response(frontik_app, tornado_request.path) + else: + request_context.set_handler_name(f'{route.endpoint.__module__}.{route.endpoint.__name__}') + + if page_cls is not None: + status, reason, headers, data = await legacy_process_request( + frontik_app, tornado_request, route, page_cls, path_params, debug_mode + ) + else: + result = {'headers': get_default_headers()} + scope, receive, send = convert_tornado_request_to_asgi( + frontik_app, tornado_request, route, path_params, debug_mode, result + ) + await app(scope, receive, send) + + status = result['status'] + reason = httputil.responses.get(status, 'Unknown') + headers = HTTPHeaders(result['headers']) + data = result['data'] + + if not scope['json_builder'].is_empty(): + if data != b'null': + raise RuntimeError('Cant have return and json.put at the same time') + + headers['Content-Type'] = media_types.APPLICATION_JSON + data = scope['json_builder'].to_bytes() + headers['Content-Length'] = str(len(data)) + + if debug_mode.enabled: + debug_transform = DebugTransform(frontik_app, debug_mode) + status, headers, data = debug_transform.transform_chunk(tornado_request, status, headers, data) + reason = httputil.responses.get(status, 'Unknown') + + log_request(tornado_request, status) + + assert tornado_request.connection is not None + tornado_request.connection.set_close_callback(None) # type: ignore + + start_line = httputil.ResponseStartLine('', status, reason) + future = tornado_request.connection.write_headers(start_line, headers, data) + tornado_request.connection.finish() + await future + + +def make_not_found_response(frontik_app: FrontikApplication, path: str) -> tuple[int, str, HTTPHeaders, bytes]: + allowed_methods = get_allowed_methods(path) + + if allowed_methods: + status = 405 + headers = get_default_headers() + headers['Allow'] = ', '.join(allowed_methods) + data = b'' + elif hasattr(frontik_app, 'application_404_handler'): + status, headers, data = frontik_app.application_404_handler() + else: + status, headers, data = build_error_data(404, 'Not Found') + + reason = httputil.responses.get(status, 'Unknown') + return status, reason, HTTPHeaders(headers), data + + +def make_debug_auth_failed_response(auth_header: str) -> tuple[int, str, HTTPHeaders, bytes]: + status = http.client.UNAUTHORIZED + reason = httputil.responses.get(status, 'Unknown') + headers = get_default_headers() + headers['WWW-Authenticate'] = auth_header + + return status, reason, HTTPHeaders(headers), b'' + + +def make_not_accepted_response() -> tuple[int, str, HTTPHeaders, bytes]: + status = http.client.SERVICE_UNAVAILABLE + reason = httputil.responses.get(status, 'Unknown') + headers = get_default_headers() + return status, reason, HTTPHeaders(headers), b'' + + +def build_error_data( + status_code: int = 500, message: Optional[str] = 'Internal Server Error' +) -> tuple[int, dict, bytes]: + headers = get_default_headers() + headers['Content-Type'] = media_types.TEXT_HTML + data = f'{status_code}: {message}{status_code}: {message}'.encode() + return status_code, headers, data + + +async def legacy_process_request( + frontik_app: FrontikApplication, + tornado_request: httputil.HTTPServerRequest, + route: APIRoute, + page_cls: type[PageHandler], + path_params: dict[str, str], + debug_mode: DebugMode, +) -> tuple[int, str, HTTPHeaders, bytes]: + handler: PageHandler = page_cls(frontik_app, tornado_request, route, debug_mode, path_params) + return await handler.my_execute() + + +def convert_tornado_request_to_asgi( + frontik_app: FrontikApplication, + tornado_request: httputil.HTTPServerRequest, + route: APIRoute, + path_params: dict[str, str], + debug_mode: DebugMode, + result: dict[str, Any], +) -> tuple[dict, Callable, Callable]: + headers = [ + (header.encode(CHARSET).lower(), value.encode(CHARSET)) + for header in tornado_request.headers + for value in tornado_request.headers.get_list(header) + ] + + json_builder = JsonBuilder() + + scope = { + 'type': tornado_request.protocol, + 'http_version': tornado_request.version, + 'path': tornado_request.path, + 'method': tornado_request.method, + 'query_string': tornado_request.query.encode(CHARSET), + 'headers': headers, + 'client': (tornado_request.remote_ip, 0), + 'route': route, + 'path_params': path_params, + 'http_client_factory': frontik_app.http_client_factory, + 'debug_enabled': debug_mode.enabled, + 'pass_debug': debug_mode.pass_debug, + 'start_time': tornado_request._start_time, + 'json_builder': json_builder, + } + + async def receive(): + return { + 'body': tornado_request.body, + 'type': 'http.request', + 'more_body': False, + } + + async def send(data): + if data['type'] == 'http.response.start': + result['status'] = data['status'] + for h in data['headers']: + if len(h) == 2: + result['headers'][h[0].decode(CHARSET)] = h[1].decode(CHARSET) + elif data['type'] == 'http.response.body': + assert isinstance(data['body'], bytes) + result['data'] = data['body'] + else: + raise RuntimeError(f'Unsupported response type "{data["type"]}" for asgi app') + + return scope, receive, send diff --git a/frontik/integrations/sentry.py b/frontik/integrations/sentry.py index cb29341a6..7887c946c 100644 --- a/frontik/integrations/sentry.py +++ b/frontik/integrations/sentry.py @@ -3,6 +3,7 @@ from typing import TYPE_CHECKING, Optional import sentry_sdk +from http_client.request_response import FailFastError from sentry_sdk.integrations.aiohttp import AioHttpIntegration from sentry_sdk.integrations.atexit import AtexitIntegration from sentry_sdk.integrations.dedupe import DedupeIntegration @@ -11,6 +12,8 @@ from sentry_sdk.integrations.logging import LoggingIntegration from sentry_sdk.integrations.modules import ModulesIntegration from sentry_sdk.integrations.stdlib import StdlibIntegration +from sentry_sdk.integrations.tornado import TornadoIntegration +from tornado.web import HTTPError from frontik.integrations import Integration, integrations_logger from frontik.options import options @@ -35,6 +38,7 @@ def initialize_app(self, app: FrontikApplication) -> Optional[Future]: DedupeIntegration(), ModulesIntegration(), StdlibIntegration(), + TornadoIntegration(), ] if options.sentry_exception_integration: @@ -54,6 +58,7 @@ def initialize_app(self, app: FrontikApplication) -> Optional[Future]: traces_sample_rate=options.sentry_traces_sample_rate, in_app_include=list(filter(None, options.sentry_in_app_include.split(','))), profiles_sample_rate=options.sentry_profiles_sample_rate, + ignore_errors=[HTTPError, FailFastError], ) return None diff --git a/frontik/integrations/statsd.py b/frontik/integrations/statsd.py index 9631958a6..01c4be340 100644 --- a/frontik/integrations/statsd.py +++ b/frontik/integrations/statsd.py @@ -27,7 +27,7 @@ def initialize_app(self, app: FrontikApplication) -> Optional[Future]: return None def initialize_handler(self, handler): - handler.statsd_client = self.statsd_client + pass def _convert_tag(name: str, value: Any) -> str: diff --git a/frontik/integrations/telemetry.py b/frontik/integrations/telemetry.py index e5165414e..d77471169 100644 --- a/frontik/integrations/telemetry.py +++ b/frontik/integrations/telemetry.py @@ -5,22 +5,21 @@ from typing import TYPE_CHECKING, Optional from urllib.parse import urlparse -import opentelemetry.instrumentation.fastapi from http_client import client_request_context, response_status_code_context from http_client.options import options as http_client_options from opentelemetry import trace from opentelemetry.exporter.otlp.proto.grpc.trace_exporter import OTLPSpanExporter -from opentelemetry.instrumentation import aiohttp_client, fastapi +from opentelemetry.instrumentation import aiohttp_client, tornado from opentelemetry.propagate import set_global_textmap from opentelemetry.sdk.resources import Resource -from opentelemetry.sdk.trace import IdGenerator, ReadableSpan, TracerProvider +from opentelemetry.sdk.trace import IdGenerator, TracerProvider +from opentelemetry.sdk.trace import Span as SpanImpl from opentelemetry.sdk.trace.export import BatchSpanProcessor from opentelemetry.sdk.trace.sampling import ParentBased, TraceIdRatioBased from opentelemetry.semconv.resource import ResourceAttributes from opentelemetry.semconv.trace import SpanAttributes -from opentelemetry.trace import SpanKind from opentelemetry.trace.propagation.tracecontext import TraceContextTextMapPropagator -from starlette.types import Scope +from opentelemetry.util.http import ExcludeList from frontik import request_context from frontik.integrations import Integration, integrations_logger @@ -32,46 +31,39 @@ import aiohttp from http_client.request_response import RequestBuilder from opentelemetry.trace import Span + from opentelemetry.util import types from frontik.app import FrontikApplication log = logging.getLogger('telemetry') +# change log-level, because mainly detach context produce exception on Tornado 5. Will be deleted, when up Tornado to 6 +logging.getLogger('opentelemetry.context').setLevel(logging.CRITICAL) set_global_textmap(TraceContextTextMapPropagator()) - -class FrontikSpanProcessor(BatchSpanProcessor): - def on_end(self, span: ReadableSpan) -> None: - if ( - span.kind == SpanKind.INTERNAL - and span.attributes - and ( - span.attributes.get('type', None) - in ('http.request', 'http.response.start', 'http.disconnect', 'http.response.body') - ) - ): - return - super().on_end(span=span) - - -def monkey_patch_route_details(scope: Scope) -> tuple: - route = scope['path'] - span_name = route or scope.get('method', '') - attributes = {} - if route: - attributes[SpanAttributes.HTTP_ROUTE] = route - return span_name, attributes +tornado._excluded_urls = ExcludeList([*list(tornado._excluded_urls._excluded_urls), '/status']) +excluded_span_attributes = ['tornado.handler'] class TelemetryIntegration(Integration): def __init__(self): self.aiohttp_instrumentor = aiohttp_client.AioHttpClientInstrumentor() + self.tornado_instrumentor = tornado.TornadoInstrumentor() + TelemetryIntegration.patch_span_impl() + + @staticmethod + def patch_span_impl() -> None: + set_attribute = SpanImpl.set_attribute + + def patched_set_attribute(self: SpanImpl, key: str, value: types.AttributeValue) -> None: + if key not in excluded_span_attributes: + return set_attribute(self, key, value) + + SpanImpl.set_attribute = patched_set_attribute # type: ignore def initialize_app(self, app: FrontikApplication) -> Optional[Future]: if not options.opentelemetry_enabled: return None - opentelemetry.instrumentation.fastapi._get_route_details = monkey_patch_route_details - integrations_logger.info('start telemetry') resource = Resource( @@ -90,15 +82,11 @@ def initialize_app(self, app: FrontikApplication) -> Optional[Future]: sampler=ParentBased(TraceIdRatioBased(options.opentelemetry_sampler_ratio)), ) - provider.add_span_processor(FrontikSpanProcessor(otlp_exporter)) + provider.add_span_processor(BatchSpanProcessor(otlp_exporter)) trace.set_tracer_provider(provider) self.aiohttp_instrumentor.instrument(request_hook=_client_request_hook, response_hook=_client_response_hook) - - fastapi.FastAPIInstrumentor.instrument_app( - app.fastapi_app, server_request_hook=_server_request_hook, excluded_urls='/status' - ) - + self.tornado_instrumentor.instrument(server_request_hook=_server_request_hook) return None def deinitialize_app(self, app: FrontikApplication) -> Optional[Future]: @@ -107,15 +95,21 @@ def deinitialize_app(self, app: FrontikApplication) -> Optional[Future]: integrations_logger.info('stop telemetry') self.aiohttp_instrumentor.uninstrument() - fastapi.FastAPIInstrumentor.uninstrument_app(app.fastapi_app) + self.tornado_instrumentor.uninstrument() return None def initialize_handler(self, handler): pass -def _server_request_hook(span: Span, scope: dict) -> None: - span.set_attribute(SpanAttributes.HTTP_TARGET, scope['path']) +def _server_request_hook(span, handler): + if (handler_name := request_context.get_handler_name()) is not None: + method_path, method_name = handler_name.rsplit('.', 1) + span.update_name(f'{method_path}.{method_name}') + span.set_attribute(SpanAttributes.CODE_FUNCTION, method_name) + span.set_attribute(SpanAttributes.CODE_NAMESPACE, method_path) + + span.set_attribute(SpanAttributes.HTTP_TARGET, handler.request.uri) def _client_request_hook(span: Span, params: aiohttp.TraceRequestStartParams) -> None: diff --git a/frontik/json_builder.py b/frontik/json_builder.py index bdf940cea..9c98054e4 100644 --- a/frontik/json_builder.py +++ b/frontik/json_builder.py @@ -1,9 +1,10 @@ from __future__ import annotations import logging -from typing import Any, Callable, Optional, Union +from typing import Annotated, Any, Callable, Optional, Union import orjson +from fastapi import Depends, Request from pydantic import BaseModel from tornado.concurrent import Future @@ -60,8 +61,12 @@ def _encode_value(value: Any) -> Any: raise TypeError +def json_encode_bytes(obj: Any, default: Callable = _encode_value) -> bytes: + return orjson.dumps(obj, default=default, option=orjson.OPT_NON_STR_KEYS) + + def json_encode(obj: Any, default: Callable = _encode_value) -> str: - return orjson.dumps(obj, default=default, option=orjson.OPT_NON_STR_KEYS).decode('utf-8') + return json_encode_bytes(obj, default).decode('utf-8') def json_decode(value: Union[str, bytes]) -> Any: @@ -120,3 +125,13 @@ def to_string(self) -> str: return json_encode(self._concat_chunks()) return json_encode(self._concat_chunks(), default=self._encoder) + + def to_bytes(self) -> bytes: + return json_encode_bytes(self._concat_chunks()) + + +def get_json_builder(request: Request) -> JsonBuilder: + return request['json_builder'] + + +JsonBuilderT = Annotated[JsonBuilder, Depends(get_json_builder)] diff --git a/frontik/producers/json_producer.py b/frontik/producers/json_producer.py index 4881e0772..96f24c253 100644 --- a/frontik/producers/json_producer.py +++ b/frontik/producers/json_producer.py @@ -10,6 +10,7 @@ from tornado.escape import to_unicode from frontik import json_builder, media_types +from frontik.auth import check_debug_auth_or_finish from frontik.options import options from frontik.producers import ProducerFactory from frontik.util import get_abs_path, get_cookie_or_url_param_value @@ -39,7 +40,7 @@ def __init__( def __call__(self): if get_cookie_or_url_param_value(self.handler, 'notpl') is not None: - self.handler.require_debug_access() + check_debug_auth_or_finish(self.handler) self.log.debug('ignoring templating because notpl parameter is passed') return self._finish_with_json() @@ -108,7 +109,7 @@ async def _finish_with_template(self) -> tuple[Optional[str], None]: msg = 'Cannot apply template, no Jinja2 environment configured' raise Exception(msg) - if self.handler.resp_headers.get('Content-Type', None) is None: + if self.handler._headers.get('Content-Type') is None: self.handler.set_header('Content-Type', media_types.TEXT_HTML) try: @@ -141,7 +142,7 @@ async def _finish_with_template(self) -> tuple[Optional[str], None]: async def _finish_with_json(self) -> tuple[str, None]: self.log.debug('finishing without templating') - if self.handler.resp_headers.get('Content-Type', None) is None: + if self.handler._headers.get('Content-Type') is None: self.handler.set_header('Content-Type', media_types.APPLICATION_JSON) return self.json.to_string(), None diff --git a/frontik/producers/xml_producer.py b/frontik/producers/xml_producer.py index 86144943d..dbf18782e 100644 --- a/frontik/producers/xml_producer.py +++ b/frontik/producers/xml_producer.py @@ -14,6 +14,7 @@ import frontik.doc import frontik.util from frontik import file_cache, media_types +from frontik.auth import check_debug_auth_or_finish from frontik.options import options from frontik.producers import ProducerFactory from frontik.util import get_abs_path @@ -49,7 +50,7 @@ def __init__( def __call__(self): if any(frontik.util.get_cookie_or_url_param_value(self.handler, p) is not None for p in ('noxsl', 'notpl')): - self.handler.require_debug_access() + check_debug_auth_or_finish(self.handler) self.log.debug('ignoring XSLT because noxsl/notpl parameter is passed') return self._finish_with_xml(escape_xmlns=True) @@ -76,7 +77,7 @@ def set_xsl(self, filename: str) -> None: async def _finish_with_xslt(self) -> tuple[Optional[str], Optional[list[Any]]]: self.log.debug('finishing with XSLT') - if self.handler.resp_headers.get('Content-Type', None) is None: + if self.handler._headers.get('Content-Type') is None: self.handler.set_header('Content-Type', media_types.TEXT_HTML) def job(): @@ -127,7 +128,7 @@ def get_xsl_log() -> str: async def _finish_with_xml(self, escape_xmlns: bool = False) -> tuple[bytes, None]: self.log.debug('finishing without XSLT') - if self.handler.resp_headers.get('Content-Type', None) is None: + if self.handler._headers.get('Content-Type') is None: self.handler.set_header('Content-Type', media_types.APPLICATION_XML) if escape_xmlns: diff --git a/frontik/routing.py b/frontik/routing.py index 7e605b870..2a4efce41 100644 --- a/frontik/routing.py +++ b/frontik/routing.py @@ -1,20 +1,18 @@ +from __future__ import annotations + import importlib import logging import pkgutil import re -import time from collections.abc import Generator from pathlib import Path -from typing import Any, Callable, MutableSequence, Optional, Type, Union +from typing import TYPE_CHECKING, Any, Callable, MutableSequence, Optional, Type, Union -from fastapi import APIRouter, Request, Response +from fastapi import APIRouter from fastapi.routing import APIRoute -from starlette.middleware.base import BaseHTTPMiddleware -from frontik import request_context -from frontik.handler import PageHandler, build_error_data, get_default_headers, process_request -from frontik.options import options -from frontik.util import check_request_id, generate_uniq_timestamp_request_id +if TYPE_CHECKING: + from frontik.handler import PageHandler routing_logger = logging.getLogger('frontik.routing') @@ -30,23 +28,23 @@ def __init__(self, *args: Any, **kwargs: Any) -> None: self._cls: Optional[Type[PageHandler]] = None self._path: Optional[str] = None - def get(self, path: str, cls: Type[PageHandler] = PageHandler, **kwargs: Any) -> Callable: + def get(self, path: str, cls: Optional[Type[PageHandler]] = None, **kwargs: Any) -> Callable: self._path, self._cls = path, cls return super().get(path, **kwargs) - def post(self, path: str, cls: Type[PageHandler] = PageHandler, **kwargs: Any) -> Callable: + def post(self, path: str, cls: Optional[Type[PageHandler]] = None, **kwargs: Any) -> Callable: self._path, self._cls = path, cls return super().post(path, **kwargs) - def put(self, path: str, cls: Type[PageHandler] = PageHandler, **kwargs: Any) -> Callable: + def put(self, path: str, cls: Optional[Type[PageHandler]] = None, **kwargs: Any) -> Callable: self._path, self._cls = path, cls return super().put(path, **kwargs) - def delete(self, path: str, cls: Type[PageHandler] = PageHandler, **kwargs: Any) -> Callable: + def delete(self, path: str, cls: Optional[Type[PageHandler]] = None, **kwargs: Any) -> Callable: self._path, self._cls = path, cls return super().delete(path, **kwargs) - def head(self, path: str, cls: Type[PageHandler] = PageHandler, **kwargs: Any) -> Callable: + def head(self, path: str, cls: Optional[Type[PageHandler]] = None, **kwargs: Any) -> Callable: self._path, self._cls = path, cls return super().head(path, **kwargs) @@ -69,23 +67,23 @@ def __init__(self, *args: Any, **kwargs: Any) -> None: self._cls: Optional[Type[PageHandler]] = None self._path: Optional[str] = None - def get(self, path: str, cls: Type[PageHandler] = PageHandler, **kwargs: Any) -> Callable: + def get(self, path: str, cls: Optional[Type[PageHandler]] = None, **kwargs: Any) -> Callable: self._path, self._cls = path, cls return super().get(path, **kwargs) - def post(self, path: str, cls: Type[PageHandler] = PageHandler, **kwargs: Any) -> Callable: + def post(self, path: str, cls: Optional[Type[PageHandler]] = None, **kwargs: Any) -> Callable: self._path, self._cls = path, cls return super().post(path, **kwargs) - def put(self, path: str, cls: Type[PageHandler] = PageHandler, **kwargs: Any) -> Callable: + def put(self, path: str, cls: Optional[Type[PageHandler]] = None, **kwargs: Any) -> Callable: self._path, self._cls = path, cls return super().put(path, **kwargs) - def delete(self, path: str, cls: Type[PageHandler] = PageHandler, **kwargs: Any) -> Callable: + def delete(self, path: str, cls: Optional[Type[PageHandler]] = None, **kwargs: Any) -> Callable: self._path, self._cls = path, cls return super().delete(path, **kwargs) - def head(self, path: str, cls: Type[PageHandler] = PageHandler, **kwargs: Any) -> Callable: + def head(self, path: str, cls: Optional[Type[PageHandler]] = None, **kwargs: Any) -> Callable: self._path, self._cls = path, cls return super().head(path, **kwargs) @@ -136,85 +134,37 @@ def import_all_pages(app_module: Optional[str]) -> None: routers.extend((router, regex_router)) -def _get_remote_ip(request: Request) -> str: - ip = request.headers.get('X-Real-Ip', None) or request.headers.get('X-Forwarded-For', None) - if ip is None and request.client: - ip = request.client.host - return ip or '' +def _find_regex_route( + path: str, method: str +) -> Union[tuple[APIRoute, Type[PageHandler], dict], tuple[None, None, None]]: + for pattern, route, cls in _regex_mapping: + match = pattern.match(path) + if match and next(iter(route.methods), None) == method: + return route, cls, match.groupdict() + return None, None, None -def _setup_page_handler(request: Request, cls: Type[PageHandler]) -> None: - # create legacy PageHandler and put to request - handler = cls( - request.app.frontik_app, - request.query_params, - request.cookies, - request.headers, - request.state.body_bytes, - request.state.start_time, - request.url.path, - request.state.path_params, - _get_remote_ip(request), - request.method, - ) - request.state.handler = handler +def find_route(path: str, method: str) -> tuple[APIRoute, type, dict]: + route: APIRoute + route, page_cls = _plain_routes.get((path, method), (None, None)) + path_params: dict = {} + if route is None: + route, page_cls, path_params = _find_regex_route(path, method) -def _find_regex_route(request: Request) -> Union[tuple[APIRoute, Type[PageHandler], dict], tuple[None, None, None]]: - for pattern, route, cls in _regex_mapping: - match = pattern.match(request.url.path) - if match and next(iter(route.methods), None) == request.method: - return route, cls, match.groupdict() + if route is None: + routing_logger.error('match for request url %s "%s" not found', method, path) + return None, None, None - return None, None, None + return route, page_cls, path_params -def make_not_found_response(frontik_app, path): +def get_allowed_methods(path: str) -> list[str]: allowed_methods = [] for method in ('GET', 'POST', 'PUT', 'DELETE', 'HEAD'): route, page_cls = _plain_routes.get((path, method), (None, None)) if route is not None: allowed_methods.append(method) - if allowed_methods: - status = 405 - headers = get_default_headers() - headers['Allow'] = ', '.join(allowed_methods) - content = b'' - elif hasattr(frontik_app, 'application_404_handler'): - status, headers, content = frontik_app.application_404_handler() - else: - status, headers, content = build_error_data(404, 'Not Found') - - return Response(status_code=status, headers=headers, content=content) - - -class RoutingMiddleware(BaseHTTPMiddleware): - async def dispatch(self, request: Request, _ignored_call_next: Callable) -> Response: - request.state.start_time = time.time() - - routing_logger.info('requested url: %s', request.url.path) - - request_id = request.headers.get('X-Request-Id') or generate_uniq_timestamp_request_id() - if options.validate_request_id: - check_request_id(request_id) - - with request_context.request_context(request_id): - route: APIRoute - route, page_cls = _plain_routes.get((request.url.path, request.method), (None, None)) - request.state.path_params = {} - - if route is None: - route, page_cls, path_params = _find_regex_route(request) - request.state.path_params = path_params - - if route is None: - routing_logger.error('match for request url %s "%s" not found', request.method, request.url.path) - return make_not_found_response(request.app.frontik_app, request.url.path) - - request.state.body_bytes = await request.body() - _setup_page_handler(request, page_cls) - - response = await process_request(request, route.get_route_handler(), route) - return response + return allowed_methods diff --git a/frontik/server.py b/frontik/server.py index 808b9ae84..3657eba07 100644 --- a/frontik/server.py +++ b/frontik/server.py @@ -4,28 +4,23 @@ import importlib import logging import signal -import socket import sys from asyncio import Future -from collections.abc import Awaitable, Coroutine +from collections.abc import Coroutine from concurrent.futures import ThreadPoolExecutor -from datetime import timedelta from functools import partial from threading import Lock from typing import Any, Callable, Optional, Union -import anyio import tornado.autoreload -import uvicorn from http_client.balancing import Upstream -from starlette.middleware import Middleware +from tornado.httpserver import HTTPServer from frontik.app import FrontikApplication from frontik.config_parser import parse_configs from frontik.loggers import MDC from frontik.options import options from frontik.process import fork_workers -from frontik.routing import RoutingMiddleware, import_all_pages, routers log = logging.getLogger('server') @@ -109,79 +104,28 @@ def _run_worker(app: FrontikApplication) -> None: loop = asyncio.get_event_loop() executor = ThreadPoolExecutor(options.common_executor_pool_size) loop.set_default_executor(executor) - init_task = loop.create_task(_init_app(app)) + initialize_application_task = loop.create_task(_init_app(app)) - def initialize_application_task_result_handler(task): - if task.exception(): + def initialize_application_task_result_handler(future): + if future.exception(): loop.stop() - init_task.add_done_callback(initialize_application_task_result_handler) + initialize_application_task.add_done_callback(initialize_application_task_result_handler) loop.run_forever() + # to raise init exception if any + initialize_application_task.result() - if init_task.done() and init_task.exception(): - raise RuntimeError('worker failed') from init_task.exception() - -async def periodic_task(callback: Callable, check_timedelta: timedelta) -> None: - while True: - await asyncio.sleep(check_timedelta.total_seconds()) - callback() - - -def bind_socket(host: str, port: int) -> socket.socket: - sock = socket.socket(family=socket.AF_INET) - sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) - sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEPORT, 1) - - try: - sock.bind((host, port)) - except OSError as exc: - log.error(exc) - sys.exit(1) - - sock.set_inheritable(True) - return sock - - -def run_server(frontik_app: FrontikApplication, sock: Optional[socket.socket] = None) -> Awaitable: +def run_server(app: FrontikApplication) -> None: """Starts Frontik server for an application""" loop = asyncio.get_event_loop() log.info('starting server on %s:%s', options.host, options.port) - - anyio.to_thread.run_sync = anyio_noop - import_all_pages(frontik_app.app_module_name) - fastapi_app = frontik_app.fastapi_app - setattr(fastapi_app, 'frontik_app', frontik_app) - for router in routers: - fastapi_app.include_router(router) - - # because on idx=0 we have OpenTelemetryMiddleware - fastapi_app.user_middleware.insert(1, Middleware(RequestCancelledMiddleware)) - fastapi_app.user_middleware.insert(1, Middleware(RoutingMiddleware)) # should be last, because it ignores call_next - - config = uvicorn.Config( - fastapi_app, - host=options.host, - port=options.port, - log_level='critical', - loop='none', - log_config=None, - access_log=False, - server_header=False, - lifespan='off', - ) - server = uvicorn.Server(config) + http_server = HTTPServer(app, xheaders=options.xheaders) + http_server.bind(options.port, options.host, reuse_port=options.reuse_port) + http_server.start() if options.autoreload: - check_timedelta = timedelta(milliseconds=500) - modify_times: dict[str, float] = {} - reload = partial(tornado.autoreload._reload_on_update, modify_times) - - server_task = asyncio.gather(server._serve(), periodic_task(reload, check_timedelta)) - else: - if sock is None: - sock = bind_socket(options.host, options.port) - server_task = loop.create_task(server._serve([sock])) # type: ignore + tornado.autoreload.start(1000) def worker_sigterm_handler(_signum, _frame): log.info('requested shutdown, shutting down server on %s:%d', options.host, options.port) @@ -189,42 +133,35 @@ def worker_sigterm_handler(_signum, _frame): loop.call_soon_threadsafe(server_stop) def server_stop(): - log.info('going down in %s seconds', options.stop_timeout) + deinit_task = loop.create_task(_deinit_app(app)) + http_server.stop() - def ioloop_stop(_deinit_task): - if loop.is_running(): - log.info('stopping IOLoop') - loop.stop() - log.info('stopped') + if loop.is_running(): + log.info('going down in %s seconds', options.stop_timeout) - deinit_task = loop.create_task(_deinit_app(frontik_app, server)) - deinit_task.add_done_callback(ioloop_stop) + def ioloop_stop(_deinit_task): + if loop.is_running(): + log.info('stopping IOLoop') + loop.stop() + log.info('stopped') + + deinit_task.add_done_callback(ioloop_stop) signal.signal(signal.SIGTERM, worker_sigterm_handler) signal.signal(signal.SIGINT, worker_sigterm_handler) - return server_task - async def _init_app(frontik_app: FrontikApplication) -> None: await frontik_app.init() - server_task = run_server(frontik_app) + run_server(frontik_app) log.info('Successfully inited application %s', frontik_app.app_name) with frontik_app.worker_state.count_down_lock: frontik_app.worker_state.init_workers_count_down.value -= 1 log.info('worker is up, remaining workers = %s', frontik_app.worker_state.init_workers_count_down.value) - await server_task - - -async def kill_server(app: FrontikApplication, server: uvicorn.Server) -> None: - await asyncio.sleep(options.stop_timeout) - if app.http_client is not None: - await app.http_client.client_session.close() - server.should_exit = True -async def _deinit_app(app: FrontikApplication, server: uvicorn.Server) -> None: - deinit_futures: list[Optional[Union[Future, Coroutine]]] = [kill_server(app, server)] +async def _deinit_app(app: FrontikApplication) -> None: + deinit_futures: list[Optional[Union[Future, Coroutine]]] = [] deinit_futures.extend([integration.deinitialize_app(app) for integration in app.available_integrations]) app.upstream_manager.deregister_service_and_close() diff --git a/frontik/testing.py b/frontik/testing.py index 221bf28cb..3d1245aa2 100644 --- a/frontik/testing.py +++ b/frontik/testing.py @@ -10,6 +10,7 @@ from http_client.request_response import RequestBuilder, RequestResult from lxml import etree from tornado.escape import utf8 +from tornado.httpserver import HTTPServer from tornado_mock.httpclient import patch_http_client, set_stub from yarl import URL @@ -17,8 +18,7 @@ from frontik.loggers import bootstrap_core_logging from frontik.media_types import APPLICATION_JSON, APPLICATION_PROTOBUF, APPLICATION_XML, TEXT_PLAIN from frontik.options import options -from frontik.server import bind_socket, run_server -from frontik.util import make_url, safe_template +from frontik.util import bind_socket, make_url, safe_template class FrontikTestBase: @@ -40,13 +40,13 @@ async def _run_server(self, frontik_app): await frontik_app.init() - async def _server_coro() -> None: - await run_server(frontik_app, sock) + http_server = HTTPServer(frontik_app) + http_server.add_sockets([sock]) - server_task = asyncio.create_task(_server_coro()) yield - server_task.cancel() - await asyncio.wait_for(frontik_app.http_client.client_session.close(), timeout=5) + + http_server.stop() + await asyncio.wait_for(http_server.close_all_connections(), timeout=5) @pytest.fixture(scope='class') def with_tornado_mocks(self): diff --git a/frontik/timeout_tracking.py b/frontik/timeout_tracking.py index 9dbf52edf..8272354e5 100644 --- a/frontik/timeout_tracking.py +++ b/frontik/timeout_tracking.py @@ -112,7 +112,7 @@ def check(self, request: RequestBuilder) -> None: def get_timeout_checker( outer_caller: Optional[str], outer_timeout_ms: float, - time_since_outer_request_start_ms_supplier: float, + request_start_time: float, *, threshold_ms: float = 100, ) -> TimeoutChecker: @@ -120,6 +120,6 @@ def get_timeout_checker( return TimeoutChecker( outer_caller, outer_timeout_ms, - time_since_outer_request_start_ms_supplier, + request_start_time, threshold_ms=threshold_ms, ) diff --git a/frontik/util.py b/frontik/util.py index 400088c7e..9154ea2d2 100644 --- a/frontik/util.py +++ b/frontik/util.py @@ -6,13 +6,16 @@ import os.path import random import re +import socket +import sys from string import Template from typing import TYPE_CHECKING, Optional -from urllib.parse import parse_qs, urlencode +from urllib.parse import urlencode from uuid import uuid4 from http_client.util import any_to_bytes, any_to_unicode, to_unicode from tornado.escape import utf8 +from tornado.web import httputil if TYPE_CHECKING: from typing import Any @@ -82,7 +85,19 @@ def choose_boundary(): def get_cookie_or_url_param_value(handler: PageHandler, param_name: str) -> Optional[str]: - return handler.get_query_argument(param_name, handler.get_cookie(param_name, None)) + return handler.get_argument(param_name, handler.get_cookie(param_name, None)) + + +def get_cookie_or_param_from_request(tornado_request: httputil.HTTPServerRequest, param_name: str) -> Optional[str]: + query = tornado_request.query_arguments.get(param_name) + if query: + return query[-1].decode() + + cookie = tornado_request.cookies.get('debug', None) + if cookie: + return cookie.value + + return None def reverse_regex_named_groups(pattern: str, *args: Any, **kwargs: Any) -> str: @@ -149,11 +164,18 @@ async def gather_dict(coro_dict: dict) -> dict: return dict(zip(coro_dict.keys(), results)) -def tornado_parse_qs_bytes( - qs: bytes, keep_blank_values: bool = False, strict_parsing: bool = False -) -> dict[str, list[bytes]]: - result = parse_qs(qs.decode('latin1'), keep_blank_values, strict_parsing, encoding='latin1', errors='strict') - encoded = {} - for key, values in result.items(): - encoded[key] = [item.encode('latin1') for item in values] - return encoded +def bind_socket(host: str, port: int) -> socket.socket: + sock = socket.socket(family=socket.AF_INET) + sock.setblocking(False) + sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) + sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEPORT, 1) + + try: + sock.bind((host, port)) + except OSError as exc: + logger.error(exc) + sys.exit(1) + + sock.set_inheritable(True) + sock.listen() + return sock diff --git a/poetry.lock b/poetry.lock index 00abaa333..2275693c1 100644 --- a/poetry.lock +++ b/poetry.lock @@ -156,17 +156,17 @@ zstd = ["zstandard"] [[package]] name = "aioresponses" -version = "0.7.4" +version = "0.7.6" description = "Mock out requests made by ClientSession from aiohttp package" optional = false python-versions = "*" files = [ - {file = "aioresponses-0.7.4-py2.py3-none-any.whl", hash = "sha256:1160486b5ea96fcae6170cf2bdef029b9d3a283b7dbeabb3d7f1182769bfb6b7"}, - {file = "aioresponses-0.7.4.tar.gz", hash = "sha256:9b8c108b36354c04633bad0ea752b55d956a7602fe3e3234b939fc44af96f1d8"}, + {file = "aioresponses-0.7.6-py2.py3-none-any.whl", hash = "sha256:d2c26defbb9b440ea2685ec132e90700907fd10bcca3e85ec2f157219f0d26f7"}, + {file = "aioresponses-0.7.6.tar.gz", hash = "sha256:f795d9dbda2d61774840e7e32f5366f45752d1adc1b74c9362afd017296c7ee1"}, ] [package.dependencies] -aiohttp = ">=2.0.0,<4.0.0" +aiohttp = ">=3.3.0,<4.0.0" [[package]] name = "aiosignal" @@ -261,17 +261,6 @@ tests = ["attrs[tests-no-zope]", "zope-interface"] tests-mypy = ["mypy (>=1.6)", "pytest-mypy-plugins"] tests-no-zope = ["attrs[tests-mypy]", "cloudpickle", "hypothesis", "pympler", "pytest (>=4.3.0)", "pytest-xdist[psutil]"] -[[package]] -name = "backoff" -version = "2.2.1" -description = "Function decoration for backoff and retry" -optional = false -python-versions = ">=3.7,<4.0" -files = [ - {file = "backoff-2.2.1-py3-none-any.whl", hash = "sha256:63579f9a0628e06278f7e47b7d7d5b6ce20dc65c5e96a6f3ca99a6adca0396e8"}, - {file = "backoff-2.2.1.tar.gz", hash = "sha256:03f829f5bb1923180821643f8753b0502c3b682293992485b0eef2807afa5cba"}, -] - [[package]] name = "cachetools" version = "5.3.3" @@ -319,20 +308,6 @@ files = [ [package.extras] unicode-backport = ["unicodedata2"] -[[package]] -name = "click" -version = "8.1.7" -description = "Composable command line interface toolkit" -optional = false -python-versions = ">=3.7" -files = [ - {file = "click-8.1.7-py3-none-any.whl", hash = "sha256:ae74fb96c20a0277a1d615f1e4d73c8414f5a98db8b799a7931d1582f3390c28"}, - {file = "click-8.1.7.tar.gz", hash = "sha256:ca9853ad459e787e2192211578cc907e7594e294c7ccc834310722b41b9ca6de"}, -] - -[package.dependencies] -colorama = {version = "*", markers = "platform_system == \"Windows\""} - [[package]] name = "colorama" version = "0.4.6" @@ -408,18 +383,18 @@ all = ["email-validator (>=2.0.0)", "httpx (>=0.23.0)", "itsdangerous (>=1.1.0)" [[package]] name = "filelock" -version = "3.15.1" +version = "3.15.4" description = "A platform independent file lock." optional = false python-versions = ">=3.8" files = [ - {file = "filelock-3.15.1-py3-none-any.whl", hash = "sha256:71b3102950e91dfc1bb4209b64be4dc8854f40e5f534428d8684f953ac847fac"}, - {file = "filelock-3.15.1.tar.gz", hash = "sha256:58a2549afdf9e02e10720eaa4d4470f56386d7a6f72edd7d0596337af8ed7ad8"}, + {file = "filelock-3.15.4-py3-none-any.whl", hash = "sha256:6ca1fffae96225dab4c6eaf1c4f4f28cd2568d3ec2a44e15a08520504de468e7"}, + {file = "filelock-3.15.4.tar.gz", hash = "sha256:2207938cbc1844345cb01a5a95524dae30f0ce089eba5b00378295a17e3e90cb"}, ] [package.extras] docs = ["furo (>=2023.9.10)", "sphinx (>=7.2.6)", "sphinx-autodoc-typehints (>=1.25.2)"] -testing = ["covdefaults (>=2.3)", "coverage (>=7.3.2)", "diff-cover (>=8.0.1)", "pytest (>=7.4.3)", "pytest-asyncio (>=0.21)", "pytest-cov (>=4.1)", "pytest-mock (>=3.12)", "pytest-timeout (>=2.2)"] +testing = ["covdefaults (>=2.3)", "coverage (>=7.3.2)", "diff-cover (>=8.0.1)", "pytest (>=7.4.3)", "pytest-asyncio (>=0.21)", "pytest-cov (>=4.1)", "pytest-mock (>=3.12)", "pytest-timeout (>=2.2)", "virtualenv (>=20.26.2)"] typing = ["typing-extensions (>=4.8)"] [[package]] @@ -510,17 +485,17 @@ files = [ [[package]] name = "googleapis-common-protos" -version = "1.63.1" +version = "1.63.2" description = "Common protobufs used in Google APIs" optional = false python-versions = ">=3.7" files = [ - {file = "googleapis-common-protos-1.63.1.tar.gz", hash = "sha256:c6442f7a0a6b2a80369457d79e6672bb7dcbaab88e0848302497e3ec80780a6a"}, - {file = "googleapis_common_protos-1.63.1-py2.py3-none-any.whl", hash = "sha256:0e1c2cdfcbc354b76e4a211a35ea35d6926a835cba1377073c4861db904a1877"}, + {file = "googleapis-common-protos-1.63.2.tar.gz", hash = "sha256:27c5abdffc4911f28101e635de1533fb4cfd2c37fbaa9174587c799fac90aa87"}, + {file = "googleapis_common_protos-1.63.2-py2.py3-none-any.whl", hash = "sha256:27a2499c7e8aff199665b22741997e485eccc8645aa9176c7c988e6fae507945"}, ] [package.dependencies] -protobuf = ">=3.19.5,<3.20.0 || >3.20.0,<3.20.1 || >3.20.1,<4.21.1 || >4.21.1,<4.21.2 || >4.21.2,<4.21.3 || >4.21.3,<4.21.4 || >4.21.4,<4.21.5 || >4.21.5,<6.0.0.dev0" +protobuf = ">=3.20.2,<4.21.1 || >4.21.1,<4.21.2 || >4.21.2,<4.21.3 || >4.21.3,<4.21.4 || >4.21.4,<4.21.5 || >4.21.5,<6.0.0.dev0" [package.extras] grpc = ["grpcio (>=1.44.0,<2.0.0.dev0)"] @@ -583,17 +558,6 @@ files = [ [package.extras] protobuf = ["grpcio-tools (>=1.64.1)"] -[[package]] -name = "h11" -version = "0.14.0" -description = "A pure-Python, bring-your-own-I/O implementation of HTTP/1.1" -optional = false -python-versions = ">=3.7" -files = [ - {file = "h11-0.14.0-py3-none-any.whl", hash = "sha256:e3fe4ac4b851c468cc8363d500db52c2ead036020723024a109d37346efaa761"}, - {file = "h11-0.14.0.tar.gz", hash = "sha256:8f19fbbe99e72420ff35c00b27a34cb9937e902a8b810e2c88300c6f0a3b699d"}, -] - [[package]] name = "http-client" version = "2.1.13" @@ -612,8 +576,8 @@ yarl = "1.9.2" [package.source] type = "git" url = "https://github.com/hhru/balancing-http-client.git" -reference = "2.1.13" -resolved_reference = "9f503bf815262df536ebe63bf60396045e6d6271" +reference = "HH-220770" +resolved_reference = "321e07babb06b62387dfb9aa348ddbec10b759ac" [[package]] name = "idna" @@ -628,22 +592,22 @@ files = [ [[package]] name = "importlib-metadata" -version = "6.0.1" +version = "7.1.0" description = "Read metadata from Python packages" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "importlib_metadata-6.0.1-py3-none-any.whl", hash = "sha256:1543daade821c89b1c4a55986c326f36e54f2e6ca3bad96be4563d0acb74dcd4"}, - {file = "importlib_metadata-6.0.1.tar.gz", hash = "sha256:950127d57e35a806d520817d3e92eec3f19fdae9f0cd99da77a407c5aabefba3"}, + {file = "importlib_metadata-7.1.0-py3-none-any.whl", hash = "sha256:30962b96c0c223483ed6cc7280e7f0199feb01a0e40cfae4d4450fc6fab1f570"}, + {file = "importlib_metadata-7.1.0.tar.gz", hash = "sha256:b78938b926ee8d5f020fc4772d487045805a55ddbad2ecf21c6d60938dc7fcd2"}, ] [package.dependencies] zipp = ">=0.5" [package.extras] -docs = ["furo", "jaraco.packaging (>=9)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-lint"] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-lint"] perf = ["ipython"] -testing = ["flake8 (<5)", "flufl.flake8", "importlib-resources (>=1.3)", "packaging", "pyfakefs", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=1.3)", "pytest-flake8", "pytest-mypy (>=0.9.1)", "pytest-perf (>=0.9.2)"] +testing = ["flufl.flake8", "importlib-resources (>=1.3)", "jaraco.test (>=5.4)", "packaging", "pyfakefs", "pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-mypy", "pytest-perf (>=0.9.2)", "pytest-ruff (>=0.2.1)"] [[package]] name = "iniconfig" @@ -1020,51 +984,62 @@ files = [ [[package]] name = "opentelemetry-api" -version = "1.17.0" +version = "1.25.0" description = "OpenTelemetry Python API" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "opentelemetry_api-1.17.0-py3-none-any.whl", hash = "sha256:b41d9b2a979607b75d2683b9bbf97062a683d190bc696969fb2122fa60aeaabc"}, - {file = "opentelemetry_api-1.17.0.tar.gz", hash = "sha256:3480fcf6b783be5d440a226a51db979ccd7c49a2e98d1c747c991031348dcf04"}, + {file = "opentelemetry_api-1.25.0-py3-none-any.whl", hash = "sha256:757fa1aa020a0f8fa139f8959e53dec2051cc26b832e76fa839a6d76ecefd737"}, + {file = "opentelemetry_api-1.25.0.tar.gz", hash = "sha256:77c4985f62f2614e42ce77ee4c9da5fa5f0bc1e1821085e9a47533a9323ae869"}, ] [package.dependencies] deprecated = ">=1.2.6" -importlib-metadata = ">=6.0.0,<6.1.0" -setuptools = ">=16.0" +importlib-metadata = ">=6.0,<=7.1" + +[[package]] +name = "opentelemetry-exporter-otlp-proto-common" +version = "1.25.0" +description = "OpenTelemetry Protobuf encoding" +optional = false +python-versions = ">=3.8" +files = [ + {file = "opentelemetry_exporter_otlp_proto_common-1.25.0-py3-none-any.whl", hash = "sha256:15637b7d580c2675f70246563363775b4e6de947871e01d0f4e3881d1848d693"}, + {file = "opentelemetry_exporter_otlp_proto_common-1.25.0.tar.gz", hash = "sha256:c93f4e30da4eee02bacd1e004eb82ce4da143a2f8e15b987a9f603e0a85407d3"}, +] + +[package.dependencies] +opentelemetry-proto = "1.25.0" [[package]] name = "opentelemetry-exporter-otlp-proto-grpc" -version = "1.17.0" +version = "1.25.0" description = "OpenTelemetry Collector Protobuf over gRPC Exporter" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "opentelemetry_exporter_otlp_proto_grpc-1.17.0-py3-none-any.whl", hash = "sha256:192d781b668a74edb49152b8b5f4f7e25bcb4307a9cf4b2dfcf87e68feac98bd"}, - {file = "opentelemetry_exporter_otlp_proto_grpc-1.17.0.tar.gz", hash = "sha256:f01476ae89484bc6210e50d7a4d93c293b3a12aff562253b94f588a85af13f70"}, + {file = "opentelemetry_exporter_otlp_proto_grpc-1.25.0-py3-none-any.whl", hash = "sha256:3131028f0c0a155a64c430ca600fd658e8e37043cb13209f0109db5c1a3e4eb4"}, + {file = "opentelemetry_exporter_otlp_proto_grpc-1.25.0.tar.gz", hash = "sha256:c0b1661415acec5af87625587efa1ccab68b873745ca0ee96b69bb1042087eac"}, ] [package.dependencies] -backoff = {version = ">=1.10.0,<3.0.0", markers = "python_version >= \"3.7\""} +deprecated = ">=1.2.6" googleapis-common-protos = ">=1.52,<2.0" grpcio = ">=1.0.0,<2.0.0" opentelemetry-api = ">=1.15,<2.0" -opentelemetry-proto = "1.17.0" -opentelemetry-sdk = ">=1.17.0,<1.18.0" - -[package.extras] -test = ["pytest-grpc"] +opentelemetry-exporter-otlp-proto-common = "1.25.0" +opentelemetry-proto = "1.25.0" +opentelemetry-sdk = ">=1.25.0,<1.26.0" [[package]] name = "opentelemetry-instrumentation" -version = "0.38b0" +version = "0.46b0" description = "Instrumentation Tools & Auto Instrumentation for OpenTelemetry Python" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "opentelemetry_instrumentation-0.38b0-py3-none-any.whl", hash = "sha256:48eed87e5db9d2cddd57a8ea359bd15318560c0ffdd80d90a5fc65816e15b7f4"}, - {file = "opentelemetry_instrumentation-0.38b0.tar.gz", hash = "sha256:3dbe93248eec7652d5725d3c6d2f9dd048bb8fda6b0505aadbc99e51638d833c"}, + {file = "opentelemetry_instrumentation-0.46b0-py3-none-any.whl", hash = "sha256:89cd721b9c18c014ca848ccd11181e6b3fd3f6c7669e35d59c48dc527408c18b"}, + {file = "opentelemetry_instrumentation-0.46b0.tar.gz", hash = "sha256:974e0888fb2a1e01c38fbacc9483d024bb1132aad92d6d24e2e5543887a7adda"}, ] [package.dependencies] @@ -1074,79 +1049,96 @@ wrapt = ">=1.0.0,<2.0.0" [[package]] name = "opentelemetry-instrumentation-aiohttp-client" -version = "0.38b0" +version = "0.46b0" description = "OpenTelemetry aiohttp client instrumentation" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "opentelemetry_instrumentation_aiohttp_client-0.38b0-py3-none-any.whl", hash = "sha256:093987f5c96518ac6999eb7480af168655bc3538752ae67d4d9a5807eaad1ee0"}, - {file = "opentelemetry_instrumentation_aiohttp_client-0.38b0.tar.gz", hash = "sha256:9c3e637e742b5d8e5c8a76fae4f3812dde5e58f85598d119abd0149cb1c82ec0"}, + {file = "opentelemetry_instrumentation_aiohttp_client-0.46b0-py3-none-any.whl", hash = "sha256:e0562fbabaf5cf6dd39a391827386f33d0b12edb4c8a6b6f0c361cbc2fa0b6b8"}, + {file = "opentelemetry_instrumentation_aiohttp_client-0.46b0.tar.gz", hash = "sha256:18c9cf8631cd6fe75376a84c6a1190f87085d184e92d4bbbdcd64a535e3a7e22"}, ] [package.dependencies] opentelemetry-api = ">=1.12,<2.0" -opentelemetry-instrumentation = "0.38b0" -opentelemetry-semantic-conventions = "0.38b0" -opentelemetry-util-http = "0.38b0" +opentelemetry-instrumentation = "0.46b0" +opentelemetry-semantic-conventions = "0.46b0" +opentelemetry-util-http = "0.46b0" wrapt = ">=1.0.0,<2.0.0" [package.extras] instruments = ["aiohttp (>=3.0,<4.0)"] -test = ["opentelemetry-instrumentation-aiohttp-client[instruments]"] [[package]] name = "opentelemetry-instrumentation-asgi" -version = "0.38b0" +version = "0.46b0" description = "ASGI instrumentation for OpenTelemetry" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "opentelemetry_instrumentation_asgi-0.38b0-py3-none-any.whl", hash = "sha256:c5bba11505008a3cd1b2c42b72f85f3f4f5af50ab931eddd0b01bde376dc5971"}, - {file = "opentelemetry_instrumentation_asgi-0.38b0.tar.gz", hash = "sha256:32d1034c253de6048d0d0166b304f9125267ca9329e374202ebe011a206eba53"}, + {file = "opentelemetry_instrumentation_asgi-0.46b0-py3-none-any.whl", hash = "sha256:f13c55c852689573057837a9500aeeffc010c4ba59933c322e8f866573374759"}, + {file = "opentelemetry_instrumentation_asgi-0.46b0.tar.gz", hash = "sha256:02559f30cf4b7e2a737ab17eb52aa0779bcf4cc06573064f3e2cb4dcc7d3040a"}, ] [package.dependencies] asgiref = ">=3.0,<4.0" opentelemetry-api = ">=1.12,<2.0" -opentelemetry-instrumentation = "0.38b0" -opentelemetry-semantic-conventions = "0.38b0" -opentelemetry-util-http = "0.38b0" +opentelemetry-instrumentation = "0.46b0" +opentelemetry-semantic-conventions = "0.46b0" +opentelemetry-util-http = "0.46b0" [package.extras] instruments = ["asgiref (>=3.0,<4.0)"] -test = ["opentelemetry-instrumentation-asgi[instruments]", "opentelemetry-test-utils (==0.38b0)"] [[package]] name = "opentelemetry-instrumentation-fastapi" -version = "0.38b0" +version = "0.46b0" description = "OpenTelemetry FastAPI Instrumentation" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "opentelemetry_instrumentation_fastapi-0.38b0-py3-none-any.whl", hash = "sha256:91139586732e437b1c3d5cf838dc5be910bce27b4b679612112be03fcc4fa2aa"}, - {file = "opentelemetry_instrumentation_fastapi-0.38b0.tar.gz", hash = "sha256:8946fd414084b305ad67556a1907e2d4a497924d023effc5ea3b4b1b0c55b256"}, + {file = "opentelemetry_instrumentation_fastapi-0.46b0-py3-none-any.whl", hash = "sha256:e0f5d150c6c36833dd011f0e6ef5ede6d7406c1aed0c7c98b2d3b38a018d1b33"}, + {file = "opentelemetry_instrumentation_fastapi-0.46b0.tar.gz", hash = "sha256:928a883a36fc89f9702f15edce43d1a7104da93d740281e32d50ffd03dbb4365"}, ] [package.dependencies] opentelemetry-api = ">=1.12,<2.0" -opentelemetry-instrumentation = "0.38b0" -opentelemetry-instrumentation-asgi = "0.38b0" -opentelemetry-semantic-conventions = "0.38b0" -opentelemetry-util-http = "0.38b0" +opentelemetry-instrumentation = "0.46b0" +opentelemetry-instrumentation-asgi = "0.46b0" +opentelemetry-semantic-conventions = "0.46b0" +opentelemetry-util-http = "0.46b0" [package.extras] instruments = ["fastapi (>=0.58,<1.0)"] -test = ["httpx (>=0.22,<1.0)", "opentelemetry-instrumentation-fastapi[instruments]", "opentelemetry-test-utils (==0.38b0)", "requests (>=2.23,<3.0)"] + +[[package]] +name = "opentelemetry-instrumentation-tornado" +version = "0.46b0" +description = "Tornado instrumentation for OpenTelemetry" +optional = false +python-versions = ">=3.8" +files = [ + {file = "opentelemetry_instrumentation_tornado-0.46b0-py3-none-any.whl", hash = "sha256:e0c933087a9fa74c1918a3a971ba09903762e6d30a3a0e9998c261cc11a96fa9"}, + {file = "opentelemetry_instrumentation_tornado-0.46b0.tar.gz", hash = "sha256:3369a20c57eb9ee6846de11b192403a2c25a1a56c5ab66e03178d2b6c10bddc2"}, +] + +[package.dependencies] +opentelemetry-api = ">=1.12,<2.0" +opentelemetry-instrumentation = "0.46b0" +opentelemetry-semantic-conventions = "0.46b0" +opentelemetry-util-http = "0.46b0" + +[package.extras] +instruments = ["tornado (>=5.1.1)"] [[package]] name = "opentelemetry-proto" -version = "1.17.0" +version = "1.25.0" description = "OpenTelemetry Python Proto" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "opentelemetry_proto-1.17.0-py3-none-any.whl", hash = "sha256:c7c0f748668102598e84ca4d51975f87ebf66865aa7469fc2c5e8bdaab813e93"}, - {file = "opentelemetry_proto-1.17.0.tar.gz", hash = "sha256:8501fdc3bc76c03a2ed11603a4d9fce6e5a97eeaebd7a20ad84bba7bd79cc9f8"}, + {file = "opentelemetry_proto-1.25.0-py3-none-any.whl", hash = "sha256:f07e3341c78d835d9b86665903b199893befa5e98866f63d22b00d0b7ca4972f"}, + {file = "opentelemetry_proto-1.25.0.tar.gz", hash = "sha256:35b6ef9dc4a9f7853ecc5006738ad40443701e52c26099e197895cbda8b815a3"}, ] [package.dependencies] @@ -1154,41 +1146,43 @@ protobuf = ">=3.19,<5.0" [[package]] name = "opentelemetry-sdk" -version = "1.17.0" +version = "1.25.0" description = "OpenTelemetry Python SDK" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "opentelemetry_sdk-1.17.0-py3-none-any.whl", hash = "sha256:07424cbcc8c012bc120ed573d5443e7322f3fb393512e72866c30111070a8c37"}, - {file = "opentelemetry_sdk-1.17.0.tar.gz", hash = "sha256:99bb9a787006774f865a4b24f8179900347d03a214c362a6cb70191f77dd6132"}, + {file = "opentelemetry_sdk-1.25.0-py3-none-any.whl", hash = "sha256:d97ff7ec4b351692e9d5a15af570c693b8715ad78b8aafbec5c7100fe966b4c9"}, + {file = "opentelemetry_sdk-1.25.0.tar.gz", hash = "sha256:ce7fc319c57707ef5bf8b74fb9f8ebdb8bfafbe11898410e0d2a761d08a98ec7"}, ] [package.dependencies] -opentelemetry-api = "1.17.0" -opentelemetry-semantic-conventions = "0.38b0" -setuptools = ">=16.0" +opentelemetry-api = "1.25.0" +opentelemetry-semantic-conventions = "0.46b0" typing-extensions = ">=3.7.4" [[package]] name = "opentelemetry-semantic-conventions" -version = "0.38b0" +version = "0.46b0" description = "OpenTelemetry Semantic Conventions" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "opentelemetry_semantic_conventions-0.38b0-py3-none-any.whl", hash = "sha256:b0ba36e8b70bfaab16ee5a553d809309cc11ff58aec3d2550d451e79d45243a7"}, - {file = "opentelemetry_semantic_conventions-0.38b0.tar.gz", hash = "sha256:37f09e47dd5fc316658bf9ee9f37f9389b21e708faffa4a65d6a3de484d22309"}, + {file = "opentelemetry_semantic_conventions-0.46b0-py3-none-any.whl", hash = "sha256:6daef4ef9fa51d51855d9f8e0ccd3a1bd59e0e545abe99ac6203804e36ab3e07"}, + {file = "opentelemetry_semantic_conventions-0.46b0.tar.gz", hash = "sha256:fbc982ecbb6a6e90869b15c1673be90bd18c8a56ff1cffc0864e38e2edffaefa"}, ] +[package.dependencies] +opentelemetry-api = "1.25.0" + [[package]] name = "opentelemetry-util-http" -version = "0.38b0" +version = "0.46b0" description = "Web util for OpenTelemetry" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "opentelemetry_util_http-0.38b0-py3-none-any.whl", hash = "sha256:8e5f0451eeb5307b2c628dd799886adc5e113fb13a7207c29c672e8d168eabd8"}, - {file = "opentelemetry_util_http-0.38b0.tar.gz", hash = "sha256:85eb032b6129c4d7620583acf574e99fe2e73c33d60e256b54af436f76ceb5ae"}, + {file = "opentelemetry_util_http-0.46b0-py3-none-any.whl", hash = "sha256:8dc1949ce63caef08db84ae977fdc1848fe6dc38e6bbaad0ae3e6ecd0d451629"}, + {file = "opentelemetry_util_http-0.46b0.tar.gz", hash = "sha256:03b6e222642f9c7eae58d9132343e045b50aca9761fcb53709bd2b663571fdf6"}, ] [[package]] @@ -1431,22 +1425,22 @@ typing-extensions = ">=4.6.0,<4.7.0 || >4.7.0" [[package]] name = "pyproject-api" -version = "1.6.1" +version = "1.7.1" description = "API to interact with the python pyproject.toml based projects" optional = false python-versions = ">=3.8" files = [ - {file = "pyproject_api-1.6.1-py3-none-any.whl", hash = "sha256:4c0116d60476b0786c88692cf4e325a9814965e2469c5998b830bba16b183675"}, - {file = "pyproject_api-1.6.1.tar.gz", hash = "sha256:1817dc018adc0d1ff9ca1ed8c60e1623d5aaca40814b953af14a9cf9a5cae538"}, + {file = "pyproject_api-1.7.1-py3-none-any.whl", hash = "sha256:2dc1654062c2b27733d8fd4cdda672b22fe8741ef1dde8e3a998a9547b071eeb"}, + {file = "pyproject_api-1.7.1.tar.gz", hash = "sha256:7ebc6cd10710f89f4cf2a2731710a98abce37ebff19427116ff2174c9236a827"}, ] [package.dependencies] -packaging = ">=23.1" +packaging = ">=24.1" tomli = {version = ">=2.0.1", markers = "python_version < \"3.11\""} [package.extras] -docs = ["furo (>=2023.8.19)", "sphinx (<7.2)", "sphinx-autodoc-typehints (>=1.24)"] -testing = ["covdefaults (>=2.3)", "pytest (>=7.4)", "pytest-cov (>=4.1)", "pytest-mock (>=3.11.1)", "setuptools (>=68.1.2)", "wheel (>=0.41.2)"] +docs = ["furo (>=2024.5.6)", "sphinx-autodoc-typehints (>=2.2.1)"] +testing = ["covdefaults (>=2.3)", "pytest (>=8.2.2)", "pytest-cov (>=5)", "pytest-mock (>=3.14)", "setuptools (>=70.1)"] [[package]] name = "pytest" @@ -1562,13 +1556,13 @@ files = [ [[package]] name = "sentry-sdk" -version = "2.1.1" +version = "2.7.0" description = "Python client for Sentry (https://sentry.io)" optional = false python-versions = ">=3.6" files = [ - {file = "sentry_sdk-2.1.1-py2.py3-none-any.whl", hash = "sha256:99aeb78fb76771513bd3b2829d12613130152620768d00cd3e45ac00cb17950f"}, - {file = "sentry_sdk-2.1.1.tar.gz", hash = "sha256:95d8c0bb41c8b0bc37ab202c2c4a295bb84398ee05f4cdce55051cd75b926ec1"}, + {file = "sentry_sdk-2.7.0-py2.py3-none-any.whl", hash = "sha256:db9594c27a4d21c1ebad09908b1f0dc808ef65c2b89c1c8e7e455143262e37c1"}, + {file = "sentry_sdk-2.7.0.tar.gz", hash = "sha256:d846a211d4a0378b289ced3c434480945f110d0ede00450ba631fc2852e7a0d4"}, ] [package.dependencies] @@ -1590,7 +1584,7 @@ django = ["django (>=1.8)"] falcon = ["falcon (>=1.4)"] fastapi = ["fastapi (>=0.79.0)"] flask = ["blinker (>=1.1)", "flask (>=0.11)", "markupsafe"] -grpcio = ["grpcio (>=1.21.1)"] +grpcio = ["grpcio (>=1.21.1)", "protobuf (>=3.8.0)"] httpx = ["httpx (>=0.16.0)"] huey = ["huey (>=2)"] huggingface-hub = ["huggingface-hub (>=0.22)"] @@ -1598,7 +1592,7 @@ langchain = ["langchain (>=0.0.210)"] loguru = ["loguru (>=0.5)"] openai = ["openai (>=1.0.0)", "tiktoken (>=0.3.0)"] opentelemetry = ["opentelemetry-distro (>=0.35b0)"] -opentelemetry-experimental = ["opentelemetry-distro (>=0.40b0,<1.0)", "opentelemetry-instrumentation-aiohttp-client (>=0.40b0,<1.0)", "opentelemetry-instrumentation-django (>=0.40b0,<1.0)", "opentelemetry-instrumentation-fastapi (>=0.40b0,<1.0)", "opentelemetry-instrumentation-flask (>=0.40b0,<1.0)", "opentelemetry-instrumentation-requests (>=0.40b0,<1.0)", "opentelemetry-instrumentation-sqlite3 (>=0.40b0,<1.0)", "opentelemetry-instrumentation-urllib (>=0.40b0,<1.0)"] +opentelemetry-experimental = ["opentelemetry-instrumentation-aio-pika (==0.46b0)", "opentelemetry-instrumentation-aiohttp-client (==0.46b0)", "opentelemetry-instrumentation-aiopg (==0.46b0)", "opentelemetry-instrumentation-asgi (==0.46b0)", "opentelemetry-instrumentation-asyncio (==0.46b0)", "opentelemetry-instrumentation-asyncpg (==0.46b0)", "opentelemetry-instrumentation-aws-lambda (==0.46b0)", "opentelemetry-instrumentation-boto (==0.46b0)", "opentelemetry-instrumentation-boto3sqs (==0.46b0)", "opentelemetry-instrumentation-botocore (==0.46b0)", "opentelemetry-instrumentation-cassandra (==0.46b0)", "opentelemetry-instrumentation-celery (==0.46b0)", "opentelemetry-instrumentation-confluent-kafka (==0.46b0)", "opentelemetry-instrumentation-dbapi (==0.46b0)", "opentelemetry-instrumentation-django (==0.46b0)", "opentelemetry-instrumentation-elasticsearch (==0.46b0)", "opentelemetry-instrumentation-falcon (==0.46b0)", "opentelemetry-instrumentation-fastapi (==0.46b0)", "opentelemetry-instrumentation-flask (==0.46b0)", "opentelemetry-instrumentation-grpc (==0.46b0)", "opentelemetry-instrumentation-httpx (==0.46b0)", "opentelemetry-instrumentation-jinja2 (==0.46b0)", "opentelemetry-instrumentation-kafka-python (==0.46b0)", "opentelemetry-instrumentation-logging (==0.46b0)", "opentelemetry-instrumentation-mysql (==0.46b0)", "opentelemetry-instrumentation-mysqlclient (==0.46b0)", "opentelemetry-instrumentation-pika (==0.46b0)", "opentelemetry-instrumentation-psycopg (==0.46b0)", "opentelemetry-instrumentation-psycopg2 (==0.46b0)", "opentelemetry-instrumentation-pymemcache (==0.46b0)", "opentelemetry-instrumentation-pymongo (==0.46b0)", "opentelemetry-instrumentation-pymysql (==0.46b0)", "opentelemetry-instrumentation-pyramid (==0.46b0)", "opentelemetry-instrumentation-redis (==0.46b0)", "opentelemetry-instrumentation-remoulade (==0.46b0)", "opentelemetry-instrumentation-requests (==0.46b0)", "opentelemetry-instrumentation-sklearn (==0.46b0)", "opentelemetry-instrumentation-sqlalchemy (==0.46b0)", "opentelemetry-instrumentation-sqlite3 (==0.46b0)", "opentelemetry-instrumentation-starlette (==0.46b0)", "opentelemetry-instrumentation-system-metrics (==0.46b0)", "opentelemetry-instrumentation-threading (==0.46b0)", "opentelemetry-instrumentation-tornado (==0.46b0)", "opentelemetry-instrumentation-tortoiseorm (==0.46b0)", "opentelemetry-instrumentation-urllib (==0.46b0)", "opentelemetry-instrumentation-urllib3 (==0.46b0)", "opentelemetry-instrumentation-wsgi (==0.46b0)"] pure-eval = ["asttokens", "executing", "pure-eval"] pymongo = ["pymongo (>=3.1)"] pyspark = ["pyspark (>=2.4.4)"] @@ -1612,18 +1606,18 @@ tornado = ["tornado (>=5)"] [[package]] name = "setuptools" -version = "70.0.0" +version = "70.1.1" description = "Easily download, build, install, upgrade, and uninstall Python packages" optional = false python-versions = ">=3.8" files = [ - {file = "setuptools-70.0.0-py3-none-any.whl", hash = "sha256:54faa7f2e8d2d11bcd2c07bed282eef1046b5c080d1c32add737d7b5817b1ad4"}, - {file = "setuptools-70.0.0.tar.gz", hash = "sha256:f211a66637b8fa059bb28183da127d4e86396c991a942b028c6650d4319c3fd0"}, + {file = "setuptools-70.1.1-py3-none-any.whl", hash = "sha256:a58a8fde0541dab0419750bcc521fbdf8585f6e5cb41909df3a472ef7b81ca95"}, + {file = "setuptools-70.1.1.tar.gz", hash = "sha256:937a48c7cdb7a21eb53cd7f9b59e525503aa8abaf3584c730dc5f7a5bec3a650"}, ] [package.extras] docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "pyproject-hooks (!=1.1)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"] -testing = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "importlib-metadata", "ini2toml[lite] (>=0.14)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "mypy (==1.9)", "packaging (>=23.2)", "pip (>=19.1)", "pyproject-hooks (!=1.1)", "pytest (>=6,!=8.1.1)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-home (>=0.5)", "pytest-mypy", "pytest-perf", "pytest-ruff (>=0.2.1)", "pytest-subprocess", "pytest-timeout", "pytest-xdist (>=3)", "tomli", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] +testing = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "importlib-metadata", "ini2toml[lite] (>=0.14)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "jaraco.test", "mypy (==1.10.0)", "packaging (>=23.2)", "pip (>=19.1)", "pyproject-hooks (!=1.1)", "pytest (>=6,!=8.1.1)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-home (>=0.5)", "pytest-mypy", "pytest-perf", "pytest-ruff (>=0.3.2)", "pytest-subprocess", "pytest-timeout", "pytest-xdist (>=3)", "tomli", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] [[package]] name = "six" @@ -1678,22 +1672,22 @@ files = [ [[package]] name = "tornado" -version = "6.3.2" +version = "6.3.3" description = "Tornado is a Python web framework and asynchronous networking library, originally developed at FriendFeed." optional = false python-versions = ">= 3.8" files = [ - {file = "tornado-6.3.2-cp38-abi3-macosx_10_9_universal2.whl", hash = "sha256:c367ab6c0393d71171123ca5515c61ff62fe09024fa6bf299cd1339dc9456829"}, - {file = "tornado-6.3.2-cp38-abi3-macosx_10_9_x86_64.whl", hash = "sha256:b46a6ab20f5c7c1cb949c72c1994a4585d2eaa0be4853f50a03b5031e964fc7c"}, - {file = "tornado-6.3.2-cp38-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c2de14066c4a38b4ecbbcd55c5cc4b5340eb04f1c5e81da7451ef555859c833f"}, - {file = "tornado-6.3.2-cp38-abi3-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:05615096845cf50a895026f749195bf0b10b8909f9be672f50b0fe69cba368e4"}, - {file = "tornado-6.3.2-cp38-abi3-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5b17b1cf5f8354efa3d37c6e28fdfd9c1c1e5122f2cb56dac121ac61baa47cbe"}, - {file = "tornado-6.3.2-cp38-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:29e71c847a35f6e10ca3b5c2990a52ce38b233019d8e858b755ea6ce4dcdd19d"}, - {file = "tornado-6.3.2-cp38-abi3-musllinux_1_1_i686.whl", hash = "sha256:834ae7540ad3a83199a8da8f9f2d383e3c3d5130a328889e4cc991acc81e87a0"}, - {file = "tornado-6.3.2-cp38-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:6a0848f1aea0d196a7c4f6772197cbe2abc4266f836b0aac76947872cd29b411"}, - {file = "tornado-6.3.2-cp38-abi3-win32.whl", hash = "sha256:7efcbcc30b7c654eb6a8c9c9da787a851c18f8ccd4a5a3a95b05c7accfa068d2"}, - {file = "tornado-6.3.2-cp38-abi3-win_amd64.whl", hash = "sha256:0c325e66c8123c606eea33084976c832aa4e766b7dff8aedd7587ea44a604cdf"}, - {file = "tornado-6.3.2.tar.gz", hash = "sha256:4b927c4f19b71e627b13f3db2324e4ae660527143f9e1f2e2fb404f3a187e2ba"}, + {file = "tornado-6.3.3-cp38-abi3-macosx_10_9_universal2.whl", hash = "sha256:502fba735c84450974fec147340016ad928d29f1e91f49be168c0a4c18181e1d"}, + {file = "tornado-6.3.3-cp38-abi3-macosx_10_9_x86_64.whl", hash = "sha256:805d507b1f588320c26f7f097108eb4023bbaa984d63176d1652e184ba24270a"}, + {file = "tornado-6.3.3-cp38-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1bd19ca6c16882e4d37368e0152f99c099bad93e0950ce55e71daed74045908f"}, + {file = "tornado-6.3.3-cp38-abi3-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7ac51f42808cca9b3613f51ffe2a965c8525cb1b00b7b2d56828b8045354f76a"}, + {file = "tornado-6.3.3-cp38-abi3-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:71a8db65160a3c55d61839b7302a9a400074c9c753040455494e2af74e2501f2"}, + {file = "tornado-6.3.3-cp38-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:ceb917a50cd35882b57600709dd5421a418c29ddc852da8bcdab1f0db33406b0"}, + {file = "tornado-6.3.3-cp38-abi3-musllinux_1_1_i686.whl", hash = "sha256:7d01abc57ea0dbb51ddfed477dfe22719d376119844e33c661d873bf9c0e4a16"}, + {file = "tornado-6.3.3-cp38-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:9dc4444c0defcd3929d5c1eb5706cbe1b116e762ff3e0deca8b715d14bf6ec17"}, + {file = "tornado-6.3.3-cp38-abi3-win32.whl", hash = "sha256:65ceca9500383fbdf33a98c0087cb975b2ef3bfb874cb35b8de8740cf7f41bd3"}, + {file = "tornado-6.3.3-cp38-abi3-win_amd64.whl", hash = "sha256:22d3c2fa10b5793da13c807e6fc38ff49a4f6e1e3868b0a6f4164768bb8e20f5"}, + {file = "tornado-6.3.3.tar.gz", hash = "sha256:e7d8db41c0181c80d76c982aacc442c0783a2c54d6400fe028954201a2e032fe"}, ] [[package]] @@ -1779,25 +1773,6 @@ h2 = ["h2 (>=4,<5)"] socks = ["pysocks (>=1.5.6,!=1.5.7,<2.0)"] zstd = ["zstandard (>=0.18.0)"] -[[package]] -name = "uvicorn" -version = "0.29.0" -description = "The lightning-fast ASGI server." -optional = false -python-versions = ">=3.8" -files = [ - {file = "uvicorn-0.29.0-py3-none-any.whl", hash = "sha256:2c2aac7ff4f4365c206fd773a39bf4ebd1047c238f8b8268ad996829323473de"}, - {file = "uvicorn-0.29.0.tar.gz", hash = "sha256:6a69214c0b6a087462412670b3ef21224fa48cae0e452b5883e8e8bdfdd11dd0"}, -] - -[package.dependencies] -click = ">=7.0" -h11 = ">=0.8" -typing-extensions = {version = ">=4.0", markers = "python_version < \"3.11\""} - -[package.extras] -standard = ["colorama (>=0.4)", "httptools (>=0.5.0)", "python-dotenv (>=0.13)", "pyyaml (>=5.1)", "uvloop (>=0.14.0,!=0.15.0,!=0.15.1)", "watchfiles (>=0.13)", "websockets (>=10.4)"] - [[package]] name = "virtualenv" version = "20.26.0" @@ -2006,4 +1981,4 @@ testing = ["aioresponses", "tornado-httpclient-mock"] [metadata] lock-version = "2.0" python-versions = "~=3.9" -content-hash = "20a463510ece2dcca3a46f3ff16a08e8b41035ad75ecb22571bab9ccedeeb5a1" +content-hash = "8e817515cbfc6ddb7dd667b91b6f6a52392f710ab0f17b5b537fde1e9639de37" diff --git a/pyproject.toml b/pyproject.toml index 7ef840b3d..dc43139a1 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -20,21 +20,21 @@ aiohttp = '3.8.3' jinja2 = '3.1.2' lxml = '4.9.2' pydantic = '^2.3.0' -tornado = '6.3.2' +tornado = '6.3.3' orjson = '*' -http-client = {git = 'https://github.com/hhru/balancing-http-client.git', tag = '2.1.13'} +http-client = {git = 'https://github.com/hhru/balancing-http-client.git', branch = 'HH-220770'} python-consul2-hh = {git = 'https://github.com/hhru/python-consul2', tag = 'v0.2.10'} -opentelemetry-sdk = '1.17.0' -opentelemetry-api = '1.17.0' -opentelemetry-exporter-otlp-proto-grpc = '1.17.0' -opentelemetry-instrumentation-fastapi = '0.38b0' # check monkey patches on update -opentelemetry-instrumentation-aiohttp-client = '0.38b0' +opentelemetry-sdk = '1.25.0' +opentelemetry-api = '1.25.0' +opentelemetry-exporter-otlp-proto-grpc = '1.25.0' +opentelemetry-instrumentation-fastapi = '0.46b0' +opentelemetry-instrumentation-aiohttp-client = '0.46b0' +opentelemetry-instrumentation-tornado = '0.46b0' fastapi = '0.105.0' aiokafka = '0.8.1' -sentry-sdk = '2.1.1' -aioresponses = '0.7.4' +sentry-sdk = '2.7.0' +aioresponses = '0.7.6' tornado-httpclient-mock = '0.2.3' -uvicorn = '0.29.0' # check server_run on update [tool.poetry.group.test.dependencies] pytest = '8.1.1' diff --git a/tests/projects/balancer_app/pages/__init__.py b/tests/projects/balancer_app/pages/__init__.py index d95c7b5bb..22796e835 100644 --- a/tests/projects/balancer_app/pages/__init__.py +++ b/tests/projects/balancer_app/pages/__init__.py @@ -1,5 +1,5 @@ -from fastapi import HTTPException from http_client.balancing import Upstream +from tornado.web import HTTPError from frontik.handler import PageHandler @@ -9,18 +9,18 @@ def check_all_servers_occupied(handler: PageHandler, name: str) -> None: servers = handler.application.upstream_manager.get_upstreams().get(name, noop_upstream).servers if any(server.current_requests == 0 for server in servers): - raise HTTPException(500, 'some servers are ignored') + raise HTTPError(500, 'some servers are ignored') def check_all_requests_done(handler: PageHandler, name: str) -> None: servers = handler.application.upstream_manager.get_upstreams().get(name, noop_upstream).servers if any(server.current_requests != 0 for server in servers): - raise HTTPException(500, 'some servers have unfinished requests') + raise HTTPError(500, 'some servers have unfinished requests') def check_all_servers_were_occupied(handler: PageHandler, name: str) -> None: servers = handler.application.upstream_manager.get_upstreams().get(name, noop_upstream).servers if any(server.current_requests != 0 for server in servers): - raise HTTPException(500, 'some servers are ignored') + raise HTTPError(500, 'some servers are ignored') if any(server.stat_requests == 0 for server in servers): - raise HTTPException(500, 'some servers are ignored') + raise HTTPError(500, 'some servers are ignored') diff --git a/tests/projects/balancer_app/pages/different_datacenter.py b/tests/projects/balancer_app/pages/different_datacenter.py index 9189e4d5a..4df600090 100644 --- a/tests/projects/balancer_app/pages/different_datacenter.py +++ b/tests/projects/balancer_app/pages/different_datacenter.py @@ -1,6 +1,6 @@ -from fastapi import HTTPException from http_client.balancing import Upstream from http_client.request_response import NoAvailableServerException +from tornado.web import HTTPError from frontik import media_types from frontik.handler import PageHandler, get_current_handler @@ -21,7 +21,7 @@ async def get_page(handler=get_current_handler()): result = await handler.post_url('different_datacenter', handler.path) for server in upstream.servers: if server.stat_requests != 0: - raise HTTPException(500) + raise HTTPError(500) if result.exc is not None and isinstance(result.exc, NoAvailableServerException): handler.text = 'no backend available' diff --git a/tests/projects/balancer_app/pages/profile_with_retry.py b/tests/projects/balancer_app/pages/profile_with_retry.py index 2ba559a78..414758955 100644 --- a/tests/projects/balancer_app/pages/profile_with_retry.py +++ b/tests/projects/balancer_app/pages/profile_with_retry.py @@ -1,5 +1,5 @@ -from fastapi import HTTPException from http_client.balancing import Upstream, UpstreamConfig +from tornado.web import HTTPError from frontik import media_types from frontik.handler import PageHandler, get_current_handler @@ -25,7 +25,7 @@ async def get_page(handler=get_current_handler()): result = await handler.put_url('profile_with_retry', handler.path, profile='profile_with_retry') if result.failed or result.data is None: - raise HTTPException(500) + raise HTTPError(500) handler.text = result.data diff --git a/tests/projects/balancer_app/pages/retry_connect.py b/tests/projects/balancer_app/pages/retry_connect.py index c5d87a79b..14947108e 100644 --- a/tests/projects/balancer_app/pages/retry_connect.py +++ b/tests/projects/balancer_app/pages/retry_connect.py @@ -1,5 +1,5 @@ -from fastapi import HTTPException from http_client.balancing import Upstream +from tornado.web import HTTPError from frontik import media_types from frontik.handler import PageHandler, get_current_handler @@ -30,7 +30,7 @@ async def get_page(handler: PageHandler = get_current_handler()) -> None: for result in results: if result.failed or result.data is None: - raise HTTPException(500) + raise HTTPError(500) handler.text = handler.text + result.data diff --git a/tests/projects/balancer_app/pages/retry_connect_timeout.py b/tests/projects/balancer_app/pages/retry_connect_timeout.py index 7c69a6f5f..82806e56f 100644 --- a/tests/projects/balancer_app/pages/retry_connect_timeout.py +++ b/tests/projects/balancer_app/pages/retry_connect_timeout.py @@ -1,5 +1,5 @@ -from fastapi import HTTPException from http_client.balancing import Upstream +from tornado.web import HTTPError from frontik import media_types from frontik.handler import PageHandler, get_current_handler @@ -26,7 +26,7 @@ async def get_page(handler: PageHandler = get_current_handler()) -> None: for result in results: if result.error or result.data is None: - raise HTTPException(500) + raise HTTPError(500) handler.text = handler.text + result.data diff --git a/tests/projects/balancer_app/pages/retry_error.py b/tests/projects/balancer_app/pages/retry_error.py index f215a6a05..a25b0e011 100644 --- a/tests/projects/balancer_app/pages/retry_error.py +++ b/tests/projects/balancer_app/pages/retry_error.py @@ -1,5 +1,5 @@ -from fastapi import HTTPException from http_client.balancing import Upstream +from tornado.web import HTTPError from frontik import media_types from frontik.handler import PageHandler, get_current_handler @@ -29,7 +29,7 @@ async def get_page(handler=get_current_handler()): for result in results: if result.error or result.data is None: - raise HTTPException(500) + raise HTTPError(500) handler.text = handler.text + result.data diff --git a/tests/projects/balancer_app/pages/retry_non_idempotent_503.py b/tests/projects/balancer_app/pages/retry_non_idempotent_503.py index 3a5e66046..3426492c2 100644 --- a/tests/projects/balancer_app/pages/retry_non_idempotent_503.py +++ b/tests/projects/balancer_app/pages/retry_non_idempotent_503.py @@ -1,5 +1,5 @@ -from fastapi import HTTPException from http_client.balancing import Upstream, UpstreamConfig +from tornado.web import HTTPError from frontik import media_types from frontik.handler import PageHandler, get_current_handler @@ -30,11 +30,11 @@ async def get_page(handler=get_current_handler()): ) if res1.error or res1.data is None: - raise HTTPException(500) + raise HTTPError(500) handler.text = res1.data if res2.status_code != 503: - raise HTTPException(500) + raise HTTPError(500) check_all_requests_done(handler, 'retry_non_idempotent_503') check_all_requests_done(handler, 'do_not_retry_non_idempotent_503') diff --git a/tests/projects/balancer_app/pages/retry_on_timeout.py b/tests/projects/balancer_app/pages/retry_on_timeout.py index e9a46310c..517d921ed 100644 --- a/tests/projects/balancer_app/pages/retry_on_timeout.py +++ b/tests/projects/balancer_app/pages/retry_on_timeout.py @@ -1,5 +1,5 @@ -from fastapi import HTTPException from http_client.balancing import Upstream +from tornado.web import HTTPError from frontik import media_types from frontik.handler import PageHandler, get_current_handler @@ -26,7 +26,7 @@ async def get_page(handler=get_current_handler()): ) if result.error or result.data is None: - raise HTTPException(500) + raise HTTPError(500) handler.text = result.data diff --git a/tests/projects/balancer_app/pages/speculative_retry.py b/tests/projects/balancer_app/pages/speculative_retry.py index e37f3dd18..e71b623fb 100644 --- a/tests/projects/balancer_app/pages/speculative_retry.py +++ b/tests/projects/balancer_app/pages/speculative_retry.py @@ -1,5 +1,5 @@ -from fastapi import HTTPException from http_client.balancing import Upstream +from tornado.web import HTTPError from frontik import media_types from frontik.handler import PageHandler, get_current_handler @@ -26,7 +26,7 @@ async def get_page(handler=get_current_handler()): ) if result.failed or result.data is None: - raise HTTPException(500) + raise HTTPError(500) handler.text = result.data diff --git a/tests/projects/broken_balancer_app/pages/no_retry_error.py b/tests/projects/broken_balancer_app/pages/no_retry_error.py index 9d61f30d1..ddec3d87a 100644 --- a/tests/projects/broken_balancer_app/pages/no_retry_error.py +++ b/tests/projects/broken_balancer_app/pages/no_retry_error.py @@ -1,4 +1,4 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler from frontik.routing import router @@ -6,4 +6,4 @@ @router.post('/no_retry_error', cls=PageHandler) async def post_page(): - raise HTTPException(500, 'something went wrong, no retry') + raise HTTPError(500, 'something went wrong, no retry') diff --git a/tests/projects/broken_balancer_app/pages/profile_with_retry.py b/tests/projects/broken_balancer_app/pages/profile_with_retry.py index f142f28fb..2d826e4b5 100644 --- a/tests/projects/broken_balancer_app/pages/profile_with_retry.py +++ b/tests/projects/broken_balancer_app/pages/profile_with_retry.py @@ -1,4 +1,4 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler from frontik.routing import router @@ -6,4 +6,4 @@ @router.put('/profile_with_retry', cls=PageHandler) async def put_page(): - raise HTTPException(503, 'broken') + raise HTTPError(503, 'broken') diff --git a/tests/projects/broken_balancer_app/pages/profile_without_retry.py b/tests/projects/broken_balancer_app/pages/profile_without_retry.py index 0fc5d8aa6..e44051a15 100644 --- a/tests/projects/broken_balancer_app/pages/profile_without_retry.py +++ b/tests/projects/broken_balancer_app/pages/profile_without_retry.py @@ -1,4 +1,4 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler from frontik.routing import router @@ -6,4 +6,4 @@ @router.put('/profile_without_retry', cls=PageHandler) async def put_page(): - raise HTTPException(503, 'broken') + raise HTTPError(503, 'broken') diff --git a/tests/projects/broken_balancer_app/pages/retry_connect.py b/tests/projects/broken_balancer_app/pages/retry_connect.py index a2fd8194c..a54fb1b6f 100644 --- a/tests/projects/broken_balancer_app/pages/retry_connect.py +++ b/tests/projects/broken_balancer_app/pages/retry_connect.py @@ -1,4 +1,4 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler from frontik.routing import router @@ -6,4 +6,4 @@ @router.post('/retry_connect', cls=PageHandler) async def post_page(): - raise HTTPException(503, 'broken, retry') + raise HTTPError(503, 'broken, retry') diff --git a/tests/projects/broken_balancer_app/pages/retry_error.py b/tests/projects/broken_balancer_app/pages/retry_error.py index d56b7cfdd..7d4c3c964 100644 --- a/tests/projects/broken_balancer_app/pages/retry_error.py +++ b/tests/projects/broken_balancer_app/pages/retry_error.py @@ -1,4 +1,4 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler from frontik.routing import router @@ -6,4 +6,4 @@ @router.put('/retry_error', cls=PageHandler) async def put_page(): - raise HTTPException(503, 'broken, retry') + raise HTTPError(503, 'broken, retry') diff --git a/tests/projects/broken_balancer_app/pages/retry_non_idempotent_503.py b/tests/projects/broken_balancer_app/pages/retry_non_idempotent_503.py index db8271f54..8866443f9 100644 --- a/tests/projects/broken_balancer_app/pages/retry_non_idempotent_503.py +++ b/tests/projects/broken_balancer_app/pages/retry_non_idempotent_503.py @@ -1,4 +1,4 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler from frontik.routing import router @@ -6,4 +6,4 @@ @router.post('/retry_non_idempotent_503', cls=PageHandler) async def post_page(): - raise HTTPException(503, 'broken, retry') + raise HTTPError(503, 'broken, retry') diff --git a/tests/projects/broken_balancer_app/pages/speculative_no_retry.py b/tests/projects/broken_balancer_app/pages/speculative_no_retry.py index d1cdc766f..e4bd29b05 100644 --- a/tests/projects/broken_balancer_app/pages/speculative_no_retry.py +++ b/tests/projects/broken_balancer_app/pages/speculative_no_retry.py @@ -1,6 +1,6 @@ import asyncio -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler from frontik.routing import router @@ -9,4 +9,4 @@ @router.post('/speculative_no_retry', cls=PageHandler) async def post_page(): await asyncio.sleep(0.8) - raise HTTPException(500, 'broken') + raise HTTPError(500, 'broken') diff --git a/tests/projects/broken_balancer_app/pages/speculative_retry.py b/tests/projects/broken_balancer_app/pages/speculative_retry.py index 21e13e257..6ba5d6d5e 100644 --- a/tests/projects/broken_balancer_app/pages/speculative_retry.py +++ b/tests/projects/broken_balancer_app/pages/speculative_retry.py @@ -1,6 +1,6 @@ import asyncio -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler from frontik.routing import router @@ -9,4 +9,4 @@ @router.put('/speculative_retry', cls=PageHandler) async def put_page(): await asyncio.sleep(0.8) - raise HTTPException(503, 'broken, retry') + raise HTTPError(503, 'broken, retry') diff --git a/tests/projects/no_debug_app/pages/basic_auth.py b/tests/projects/no_debug_app/pages/basic_auth.py index 431a893f8..d0c652d9e 100644 --- a/tests/projects/no_debug_app/pages/basic_auth.py +++ b/tests/projects/no_debug_app/pages/basic_auth.py @@ -1,8 +1,9 @@ +from frontik.auth import check_debug_auth_or_finish from frontik.handler import PageHandler, get_current_handler from frontik.routing import router @router.get('/basic_auth', cls=PageHandler) -async def get_page(handler=get_current_handler()): - handler.require_debug_access('user', 'god') +async def get_page(handler: PageHandler = get_current_handler()) -> None: + check_debug_auth_or_finish(handler, 'user', 'god') handler.json.put({'authenticated': True}) diff --git a/tests/projects/re_app/pages/id_param.py b/tests/projects/re_app/pages/id_param.py index 899563c0c..adbbe5018 100644 --- a/tests/projects/re_app/pages/id_param.py +++ b/tests/projects/re_app/pages/id_param.py @@ -5,6 +5,6 @@ @regex_router.get('/id/(?P[^/]+)', cls=PageHandler) -async def get_page(handler=get_current_handler()): +async def get_page(handler: PageHandler = get_current_handler()) -> None: handler.set_xsl('id_param.xsl') handler.doc.put(etree.Element('id', value=handler.get_path_argument('id', 'wrong'))) diff --git a/tests/projects/test_app/pages/api/2/store.py b/tests/projects/test_app/pages/api/2/store.py index d60695bd8..561bfd144 100644 --- a/tests/projects/test_app/pages/api/2/store.py +++ b/tests/projects/test_app/pages/api/2/store.py @@ -11,7 +11,7 @@ class Page(PageHandler): @router.post('/api/2/envelope/', cls=Page) async def post_page(handler: Page = get_current_handler()): - messages = gzip.decompress(handler.body_bytes).decode('utf8') + messages = gzip.decompress(handler.request.body).decode('utf8') for message in messages.split('\n'): if message == '': diff --git a/tests/projects/test_app/pages/arguments.py b/tests/projects/test_app/pages/arguments.py index 0418534e2..9c7b45e45 100644 --- a/tests/projects/test_app/pages/arguments.py +++ b/tests/projects/test_app/pages/arguments.py @@ -1,15 +1,7 @@ -from fastapi import Request - from frontik.handler import PageHandler, get_current_handler from frontik.routing import router -from frontik.util import tornado_parse_qs_bytes @router.get('/arguments', cls=PageHandler) -async def get_page(request: Request, handler: PageHandler = get_current_handler()) -> None: - if handler.get_bool_argument('enc', False): - qs = tornado_parse_qs_bytes(request.scope['query_string']) - param = qs.get('param', [])[0] - handler.json.put({'тест': handler.decode_argument(param)}) - else: - handler.json.put({'тест': handler.get_query_argument('param')}) +async def get_page(handler: PageHandler = get_current_handler()) -> None: + handler.json.put({'тест': handler.get_argument('param')}) diff --git a/tests/projects/test_app/pages/async_group/group.py b/tests/projects/test_app/pages/async_group/group.py index 7f7e01ef8..abc3577f7 100644 --- a/tests/projects/test_app/pages/async_group/group.py +++ b/tests/projects/test_app/pages/async_group/group.py @@ -1,29 +1,27 @@ from typing import Any -from fastapi import Request - from frontik.handler import PageHandler, get_current_handler from frontik.routing import router from frontik.util import gather_dict @router.get('/async_group/group', cls=PageHandler) -async def get_page(request: Request, handler: PageHandler = get_current_handler()) -> None: +async def get_page(handler: PageHandler = get_current_handler()) -> None: fail_callback = handler.get_query_argument('fail_callback', 'false') == 'true' fail_request = handler.get_query_argument('fail_request', 'false') == 'true' async def task() -> Any: - request_result = await handler.post_url(request.headers.get('host', ''), handler.path + '?data=2') + request_result = await handler.post_url(handler.request.headers.get('host', ''), handler.path + '?data=2') if fail_callback: msg = "I'm dying!" raise Exception(msg) return request_result.data data = await gather_dict({ - '1': handler.post_url(request.headers.get('host', ''), handler.path + '?data=1'), + '1': handler.post_url(handler.request.headers.get('host', ''), handler.path + '?data=1'), '2': task(), '3': handler.post_url( - request.headers.get('host', ''), + handler.request.headers.get('host', ''), handler.path, data={'data': '3' if not fail_request else None}, parse_on_error=False, @@ -31,7 +29,9 @@ async def task() -> Any: }) handler.json.put(data) - result = await gather_dict({'4': handler.post_url(request.headers.get('host', ''), handler.path + '?data=4')}) + result = await gather_dict({ + '4': handler.post_url(handler.request.headers.get('host', ''), handler.path + '?data=4') + }) handler.json.put({'future_callback_result': result['4'].data['4']}) handler.json.put({'final_callback_called': True}) diff --git a/tests/projects/test_app/pages/async_group/not_waited_failed_requests.py b/tests/projects/test_app/pages/async_group/not_waited_failed_requests.py index 2ccf68bd1..719d46f38 100644 --- a/tests/projects/test_app/pages/async_group/not_waited_failed_requests.py +++ b/tests/projects/test_app/pages/async_group/not_waited_failed_requests.py @@ -1,5 +1,3 @@ -from fastapi import Request - from frontik.handler import PageHandler, get_current_handler from frontik.routing import router @@ -14,13 +12,14 @@ def _record_failed_request(self, data: dict) -> None: @router.get('/async_group/not_waited_failed_requests', cls=Page) -async def get_page(request: Request, handler: Page = get_current_handler()) -> None: +async def get_page(handler: Page = get_current_handler()) -> None: if not handler.data: + host = handler.request.headers.get('host', '') # HTTP request with waited=False and fail_fast=True should not influence responses to client - await handler.head_url(request.headers.get('host', ''), handler.path, waited=False, fail_fast=True) - await handler.post_url(request.headers.get('host', ''), handler.path, waited=False, fail_fast=True) - await handler.put_url(request.headers.get('host', ''), handler.path, waited=False, fail_fast=True) - await handler.delete_url(request.headers.get('host', ''), handler.path, waited=False, fail_fast=True) + await handler.head_url(host, handler.path, waited=False, fail_fast=True) + await handler.post_url(host, handler.path, waited=False, fail_fast=True) + await handler.put_url(host, handler.path, waited=False, fail_fast=True) + await handler.delete_url(host, handler.path, waited=False, fail_fast=True) handler.json.put({'get': True}) else: diff --git a/tests/projects/test_app/pages/async_group/not_waited_requests.py b/tests/projects/test_app/pages/async_group/not_waited_requests.py index 8421c4d73..1991d78d2 100644 --- a/tests/projects/test_app/pages/async_group/not_waited_requests.py +++ b/tests/projects/test_app/pages/async_group/not_waited_requests.py @@ -1,7 +1,5 @@ import asyncio -from fastapi import Request - from frontik.handler import AbortAsyncGroup, PageHandler, get_current_handler from frontik.routing import router @@ -24,10 +22,10 @@ def record_request(self, data: dict) -> None: @router.get('/async_group/not_waited_requests', cls=Page) -async def get_page(request: Request, handler: Page = get_current_handler()) -> None: +async def get_page(handler: Page = get_current_handler()) -> None: if not handler.data: handler.json.put({'get': True}) - asyncio.create_task(handler.coro(request.headers.get('host', ''))) + asyncio.create_task(handler.coro(handler.request.headers.get('host', ''))) else: while not all(x in handler.data for x in ('post_made', 'delete_cancelled')): await asyncio.sleep(0.05) diff --git a/tests/projects/test_app/pages/broken_workflow.py b/tests/projects/test_app/pages/broken_workflow.py index 6670d15a1..27352ef2e 100644 --- a/tests/projects/test_app/pages/broken_workflow.py +++ b/tests/projects/test_app/pages/broken_workflow.py @@ -1,4 +1,4 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler, get_current_handler from frontik.routing import router @@ -11,7 +11,7 @@ async def get_page(handler=get_current_handler()): @handler.check_finished def cb(*args, **kw): - raise HTTPException(400) + raise HTTPError(400) results = await gather_list( handler.get_url(f'http://localhost:{port}', '/page/simple/'), diff --git a/tests/projects/test_app/pages/fail_fast/__init__.py b/tests/projects/test_app/pages/fail_fast/__init__.py index 24f253032..069f98113 100644 --- a/tests/projects/test_app/pages/fail_fast/__init__.py +++ b/tests/projects/test_app/pages/fail_fast/__init__.py @@ -10,13 +10,14 @@ async def get_page_preprocessor(handler: PageHandler = get_current_handler()) -> class Page(PageHandler): - async def get_page_fail_fast(self, failed_future): - if self.get_query_argument('exception_in_fail_fast', 'false') == 'true': - raise Exception('Exception in fail_fast') + def get_page_fail_fast(self, failed_future): + if self.get_argument('exception_in_fail_fast', 'false') == 'true': + msg = 'Exception in fail_fast' + raise Exception(msg) self.json.replace({'fail_fast': True}) self.set_status(403) - return await self.finish_with_postprocessors() + self.finish_with_postprocessors() @router.get('/fail_fast', cls=Page, dependencies=[Depends(get_page_preprocessor)]) diff --git a/tests/projects/test_app/pages/fail_fast/fail_fast_without_done.py b/tests/projects/test_app/pages/fail_fast/fail_fast_without_done.py index ea917c5a0..aef672cf0 100644 --- a/tests/projects/test_app/pages/fail_fast/fail_fast_without_done.py +++ b/tests/projects/test_app/pages/fail_fast/fail_fast_without_done.py @@ -1,12 +1,12 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler, get_current_handler from frontik.routing import router class Page(PageHandler): - async def get_page_fail_fast(self, failed_future): - raise HTTPException(401) + def get_page_fail_fast(self, failed_future): + raise HTTPError(401) @router.get('/fail_fast/fail_fast_without_done', cls=Page) @@ -16,4 +16,4 @@ async def get_page(handler=get_current_handler()): @router.post('/fail_fast/fail_fast_without_done', cls=Page) async def post_page(): - raise HTTPException(403) + raise HTTPError(403) diff --git a/tests/projects/test_app/pages/fail_fast/with_postprocessors.py b/tests/projects/test_app/pages/fail_fast/with_postprocessors.py index 9de9b4e3b..fc3e0136e 100644 --- a/tests/projects/test_app/pages/fail_fast/with_postprocessors.py +++ b/tests/projects/test_app/pages/fail_fast/with_postprocessors.py @@ -1,4 +1,4 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import HTTPErrorWithPostprocessors, PageHandler, get_current_handler from frontik.routing import router @@ -18,4 +18,4 @@ async def get_page(handler=get_current_handler()): @router.post('/fail_fast/with_postprocessors', cls=Page) async def post_page(): - raise HTTPException(403) + raise HTTPError(403) diff --git a/tests/projects/test_app/pages/finish_with_postprocessors.py b/tests/projects/test_app/pages/finish_with_postprocessors.py index 70e979664..8c3c193d6 100644 --- a/tests/projects/test_app/pages/finish_with_postprocessors.py +++ b/tests/projects/test_app/pages/finish_with_postprocessors.py @@ -1,5 +1,5 @@ -from fastapi import HTTPException from lxml import etree +from tornado.web import HTTPError from frontik.handler import FinishWithPostprocessors, PageHandler, get_current_handler from frontik.routing import router @@ -21,7 +21,7 @@ async def get_page(handler=get_current_handler()): async def fail_request() -> None: await handler.post_url(handler.get_header('host'), handler.path) - raise HTTPException(500) + raise HTTPError(500) handler.run_task(fail_request()) diff --git a/tests/projects/test_app/pages/handler/delete.py b/tests/projects/test_app/pages/handler/delete.py index 69f4c9ae0..c6c940c98 100644 --- a/tests/projects/test_app/pages/handler/delete.py +++ b/tests/projects/test_app/pages/handler/delete.py @@ -1,23 +1,26 @@ from fastapi import Request -from frontik.handler import PageHandler, get_current_handler +from frontik.balancing_client import HttpClientT +from frontik.json_builder import JsonBuilderT from frontik.routing import router -@router.get('/handler/delete', cls=PageHandler) -async def get_page(request: Request, handler: PageHandler = get_current_handler()) -> None: - result = await handler.delete_url('http://' + request.headers.get('host', ''), handler.path, data={'data': 'true'}) +@router.get('/handler/delete') +async def get_page(request: Request, http_client: HttpClientT, json_builder: JsonBuilderT) -> None: + result = await http_client.delete_url( + 'http://' + request.headers.get('host', ''), request.url.path, data={'data': 'true'} + ) if not result.failed: - handler.json.put(result.data) + json_builder.put(result.data) -@router.post('/handler/delete', cls=PageHandler) -async def post_page(handler: PageHandler = get_current_handler()) -> None: - result = await handler.delete_url('http://backend', handler.path, fail_fast=True) +@router.post('/handler/delete') +async def post_page(request: Request, http_client: HttpClientT, json_builder: JsonBuilderT) -> None: + result = await http_client.delete_url('http://backend', request.url.path, fail_fast=True) if not result.failed: - handler.json.put(result.data) + json_builder.put(result.data) -@router.delete('/handler/delete', cls=PageHandler) -async def delete_page(handler: PageHandler = get_current_handler()) -> None: - handler.json.put({'delete': handler.get_query_argument('data')}) +@router.delete('/handler/delete') +async def delete_page(data: str, json_builder: JsonBuilderT) -> None: + json_builder.put({'delete': data}) diff --git a/tests/projects/test_app/pages/handler/head_url.py b/tests/projects/test_app/pages/handler/head_url.py index 839c518ba..a63dc240b 100644 --- a/tests/projects/test_app/pages/handler/head_url.py +++ b/tests/projects/test_app/pages/handler/head_url.py @@ -1,14 +1,12 @@ import http.client -from fastapi import Request - from frontik.handler import PageHandler, get_current_handler from frontik.routing import router @router.get('/handler/head_url', cls=PageHandler) -async def get_page(request: Request, handler: PageHandler = get_current_handler()) -> None: - head_result = await handler.head_url(request.headers.get('host', ''), '/handler/head', name='head') +async def get_page(handler: PageHandler = get_current_handler()) -> None: + head_result = await handler.head_url(handler.request.headers.get('host', ''), '/handler/head', name='head') if head_result.raw_body == b'' and head_result.status_code == http.client.OK: handler.text = 'OK' diff --git a/tests/projects/test_app/pages/http_client/custom_headers.py b/tests/projects/test_app/pages/http_client/custom_headers.py index a4ccbb527..d82d86a57 100644 --- a/tests/projects/test_app/pages/http_client/custom_headers.py +++ b/tests/projects/test_app/pages/http_client/custom_headers.py @@ -16,4 +16,4 @@ async def get_page(handler=get_current_handler()): @router.post('/http_client/custom_headers', cls=Page) async def post_page(handler: Page = get_current_handler()): - handler.json.put(handler.get_request_headers()) + handler.json.put(handler.request.headers) diff --git a/tests/projects/test_app/pages/http_client/post_url.py b/tests/projects/test_app/pages/http_client/post_url.py index 8bee3efc3..422cec1b5 100644 --- a/tests/projects/test_app/pages/http_client/post_url.py +++ b/tests/projects/test_app/pages/http_client/post_url.py @@ -34,7 +34,7 @@ async def get_page(handler=get_current_handler()): @router.post('/http_client/post_url', cls=PageHandler) async def post_page(handler: PageHandler = get_current_handler()): errors_count = 0 - body_parts = handler.body_bytes.split(b'\r\n--') + body_parts = handler.request.body.split(b'\r\n--') for part in body_parts: field_part = re.search(rb'name="(?P.+)"\r\n\r\n(?P.*)', part) diff --git a/tests/projects/test_app/pages/http_error.py b/tests/projects/test_app/pages/http_error.py index aee79b90c..5d37d9cac 100644 --- a/tests/projects/test_app/pages/http_error.py +++ b/tests/projects/test_app/pages/http_error.py @@ -1,4 +1,4 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler, get_current_handler from frontik.routing import router @@ -7,4 +7,4 @@ @router.get('/http_error', cls=PageHandler) async def get_page(handler=get_current_handler()): code = int(handler.get_query_argument('code', '200')) - raise HTTPException(code) + raise HTTPError(code) diff --git a/tests/projects/test_app/pages/json_page.py b/tests/projects/test_app/pages/json_page.py index 65e61a2f2..388e5f0fc 100644 --- a/tests/projects/test_app/pages/json_page.py +++ b/tests/projects/test_app/pages/json_page.py @@ -1,5 +1,3 @@ -from fastapi import Request - from frontik import media_types from frontik.handler import PageHandler, get_current_handler from frontik.routing import router @@ -19,13 +17,13 @@ def jinja_context_provider(handler): @router.get('/json_page', cls=Page) -async def get_page(request: Request, handler: Page = get_current_handler()) -> None: +async def get_page(handler: Page = get_current_handler()) -> None: invalid_json = handler.get_query_argument('invalid', 'false') requests = { - 'req1': handler.post_url(request.headers.get('host', ''), handler.path, data={'param': 1}), + 'req1': handler.post_url(handler.request.headers.get('host', ''), handler.path, data={'param': 1}), 'req2': handler.post_url( - request.headers.get('host', ''), handler.path, data={'param': 2, 'invalid': invalid_json} + handler.request.headers.get('host', ''), handler.path, data={'param': 2, 'invalid': invalid_json} ), } data = await gather_dict(requests) @@ -33,7 +31,7 @@ async def get_page(request: Request, handler: Page = get_current_handler()) -> N if handler.get_query_argument('template_error', 'false') == 'true': del data['req1'] - handler.set_template(handler.get_query_argument('template', 'jinja.html')) + handler.set_template(handler.get_query_argument('template', 'jinja.html')) # type: ignore handler.json.put(data) diff --git a/tests/projects/test_app/pages/log.py b/tests/projects/test_app/pages/log.py index 44499e810..a5d2682fe 100644 --- a/tests/projects/test_app/pages/log.py +++ b/tests/projects/test_app/pages/log.py @@ -12,8 +12,7 @@ async def get_page(handler=get_current_handler()): handler.log.info('info') try: - msg = 'test' - raise Exception(msg) + raise Exception('test') except Exception: handler.log.exception('exception') handler.log.error('error', stack_info=True) diff --git a/tests/projects/test_app/pages/mandatory_headers.py b/tests/projects/test_app/pages/mandatory_headers.py new file mode 100644 index 000000000..aa5547bfd --- /dev/null +++ b/tests/projects/test_app/pages/mandatory_headers.py @@ -0,0 +1,33 @@ +from tornado.web import HTTPError + +from frontik.handler import PageHandler, get_current_handler +from frontik.routing import router + + +@router.get('/mandatory_headers', cls=PageHandler) +async def get_page(handler=get_current_handler()): + if handler.get_argument('test_mandatory_headers', None) is not None: + handler.set_mandatory_header('TEST_HEADER', 'TEST_HEADER_VALUE') + handler.set_mandatory_cookie('TEST_COOKIE', 'TEST_HEADER_COOKIE') + raise HTTPError(500) + + elif handler.get_argument('test_without_mandatory_headers', None) is not None: + handler.add_header('TEST_HEADER', 'TEST_HEADER_VALUE') + handler.set_cookie('TEST_COOKIE', 'TEST_HEADER_COOKIE') + raise HTTPError(500) + + elif handler.get_argument('test_clear_set_mandatory_headers', None) is not None: + handler.set_mandatory_header('TEST_HEADER', 'TEST_HEADER_VALUE') + handler.set_mandatory_cookie('TEST_COOKIE', 'TEST_HEADER_COOKIE') + handler.clear_header('TEST_HEADER') + handler.clear_cookie('TEST_COOKIE') + raise HTTPError(500) + + elif handler.get_argument('test_clear_not_set_headers', None) is not None: + handler.clear_header('TEST_HEADER') + handler.clear_cookie('TEST_COOKIE') + raise HTTPError(500) + + elif handler.get_argument('test_invalid_mandatory_cookie') is not None: + handler.set_mandatory_cookie('TEST_COOKIE', '') + raise HTTPError(500) diff --git a/tests/projects/test_app/pages/postprocess.py b/tests/projects/test_app/pages/postprocess.py index cf1696ce0..2ffe65719 100644 --- a/tests/projects/test_app/pages/postprocess.py +++ b/tests/projects/test_app/pages/postprocess.py @@ -1,4 +1,4 @@ -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.handler import PageHandler, get_current_handler from frontik.routing import router @@ -12,7 +12,7 @@ def postprocessor(self, handler, tpl, meta_info): class Page(PageHandler): @staticmethod def _pp_1(handler): - raise HTTPException(400) + raise HTTPError(400) @staticmethod def _pp_2(handler): diff --git a/tests/projects/test_app/pages/redirect.py b/tests/projects/test_app/pages/redirect.py index a5a967012..20f87649e 100644 --- a/tests/projects/test_app/pages/redirect.py +++ b/tests/projects/test_app/pages/redirect.py @@ -1,7 +1,5 @@ import re -from fastapi import Request - from frontik.handler import PageHandler, get_current_handler from frontik.routing import regex_router @@ -10,15 +8,15 @@ @regex_router.get('^/redirect', cls=PageHandler) -async def get_page(request: Request, handler: PageHandler = get_current_handler()) -> None: - if PERMANENT_REDIRECT_PATTERN.match(request.url.path): +async def get_page(handler: PageHandler = get_current_handler()) -> None: + if PERMANENT_REDIRECT_PATTERN.match(handler.path): permanent = True - elif TEMPORARY_REDIRECT_PATTERN.match(request.url.path): + elif TEMPORARY_REDIRECT_PATTERN.match(handler.path): permanent = False else: raise RuntimeError('123') to_url = '/finish?foo=bar' - if request.url.query: - to_url = to_url + f'&{request.url.query}' + if handler.request.query: + to_url = to_url + f'&{handler.request.query}' handler.redirect(to_url, permanent) diff --git a/tests/projects/test_app/pages/write_after_finish.py b/tests/projects/test_app/pages/write_after_finish.py index de3a59ee9..14dc31975 100644 --- a/tests/projects/test_app/pages/write_after_finish.py +++ b/tests/projects/test_app/pages/write_after_finish.py @@ -15,7 +15,7 @@ def prepare(self): @classmethod async def _pp(cls, handler): - if handler.method != 'POST': + if handler.request.method != 'POST': handler.counter += 1 cls.counter_static = handler.counter diff --git a/tests/projects/test_app/pages/write_error.py b/tests/projects/test_app/pages/write_error.py index 9c8f700ca..6615fb964 100644 --- a/tests/projects/test_app/pages/write_error.py +++ b/tests/projects/test_app/pages/write_error.py @@ -3,14 +3,13 @@ class Page(PageHandler): - async def write_error(self, status_code=500, **kwargs): - self.set_status(status_code) + def write_error(self, status_code=500, **kwargs): self.json.put({'write_error': True}) - if self.get_query_argument('fail_write_error', 'false') == 'true': + if self.get_argument('fail_write_error', 'false') == 'true': raise Exception('exception in write_error') - return await self.finish_with_postprocessors() + self.finish_with_postprocessors() @router.get('/write_error', cls=Page) diff --git a/tests/test_arguments.py b/tests/test_arguments.py index cc365015d..23a9bc8f5 100644 --- a/tests/test_arguments.py +++ b/tests/test_arguments.py @@ -53,7 +53,7 @@ def test_arg_validation_raises_for_empty_value_with_no_default(self): def test_arg_validation_raises_for_default_of_incorrect_type(self) -> None: response = frontik_test_app.get_page('validate_arguments?str_arg=test', method=requests.put, notpl=True) - assert response.status_code == 400 + assert response.status_code == 500 def test_validation_model(self): self.query_args.update(int_arg=0) diff --git a/tests/test_asyncgroup.py b/tests/test_asyncgroup.py index 42773c4f7..b1cd0f107 100644 --- a/tests/test_asyncgroup.py +++ b/tests/test_asyncgroup.py @@ -1,42 +1,125 @@ -import asyncio import logging +import unittest +from functools import partial -import pytest +from tornado.concurrent import Future +from tornado.testing import ExpectLog -from frontik.futures import AsyncGroup +from frontik.futures import AsyncGroup, async_logger logging.root.setLevel(logging.NOTSET) -class TestAsyncGroup: - async def test_exception_in_first(self) -> None: - async def callback1() -> None: - raise Exception('callback1 error') +class TestAsyncGroup(unittest.TestCase): + async def test_callbacks(self): + data = [] - async def callback2() -> None: - await asyncio.sleep(0) + def callback2(): + data.append(2) - ag = AsyncGroup(name='test_group') - ag.add_future(asyncio.create_task(callback1())) - ag.add_future(asyncio.create_task(callback2())) + def finish_callback(): + self.assertEqual(data, [1, 2]) + data.append(3) - with pytest.raises(Exception, match='callback1 error'): - await ag.finish() + ag = AsyncGroup(finish_callback) + cb1 = ag.add(partial(data.append, 1)) + cb2 = ag.add(callback2) - assert ag.done() is True + self.assertEqual(ag._finished, False) - async def test_exception_in_last(self) -> None: - async def callback1() -> None: - await asyncio.sleep(0) + ag.try_finish() - async def callback2() -> None: - raise Exception('callback2 error') + self.assertEqual(ag._finished, False) - ag = AsyncGroup(name='test_group') - ag.add_future(asyncio.create_task(callback1())) - ag.add_future(asyncio.create_task(callback2())) + cb1() - with pytest.raises(Exception, match='callback2 error'): - await ag.finish() + self.assertEqual(ag._finished, False) - assert ag.done() is True + cb2() + + self.assertEqual(ag._finished, True) + self.assertEqual(data, [1, 2, 3]) + + def test_notifications(self) -> None: + f: Future = Future() + ag = AsyncGroup(partial(f.set_result, True)) + not1 = ag.add_notification() + not2 = ag.add_notification() + + self.assertEqual(ag._finished, False) + + not1() + + self.assertEqual(ag._finished, False) + + not2('params', are='ignored') + + self.assertEqual(ag._finished, True) + self.assertEqual(f.result(), True) + + with ExpectLog(async_logger, r'.*trying to finish already finished AsyncGroup\(name=None, finished=True\)'): + ag.finish() + + def test_finish(self) -> None: + f: Future = Future() + ag = AsyncGroup(partial(f.set_result, True)) + + self.assertEqual(ag._finished, False) + + ag.add_notification() + ag.finish() + + self.assertEqual(ag._finished, True) + self.assertEqual(f.result(), True) + + def test_exception_in_first(self) -> None: + def callback1(): + msg = 'callback1 error' + raise Exception(msg) + + def callback2(): + self.fail('callback2 should not be called') + + def finish_callback(): + self.fail('finish_callback should not be called') + + ag = AsyncGroup(finish_callback, name='test_group') + cb1 = ag.add(callback1) + cb2 = ag.add(callback2) + + self.assertRaises(Exception, cb1) + self.assertEqual(ag._finished, True) + + with ExpectLog(async_logger, r'.*ignoring executing callback in AsyncGroup\(name=test_group, finished=True\)'): + cb2() + + self.assertEqual(ag._finished, True) + + def test_exception_in_last(self) -> None: + def callback2(): + msg = 'callback1 error' + raise Exception(msg) + + def finish_callback(): + self.fail('finish_callback should not be called') + + ag = AsyncGroup(finish_callback, name='test_group') + cb1 = ag.add(lambda: None) + cb2 = ag.add(callback2) + + cb1() + + with ExpectLog(async_logger, r'.*aborting AsyncGroup\(name=test_group, finished=False\)'): + self.assertRaises(Exception, cb2) + + self.assertEqual(ag._finished, True) + + def test_exception_in_final(self) -> None: + def finish_callback(): + msg = 'callback1 error' + raise Exception(msg) + + ag = AsyncGroup(finish_callback) + + self.assertRaises(Exception, ag.try_finish) + self.assertEqual(ag._finished, True) diff --git a/tests/test_errors.py b/tests/test_errors.py index 7e45a3161..505beb3fd 100644 --- a/tests/test_errors.py +++ b/tests/test_errors.py @@ -55,7 +55,7 @@ def test_write_error(self) -> None: def test_write_error_exception(self) -> None: response = frontik_test_app.get_page('write_error?fail_write_error=true') assert response.status_code == 500 - assert response.content == b'Internal Server Error' + assert response.content == b'' def test_write_error_405(self): response = frontik_test_app.get_page('write_error', method=requests.put) diff --git a/tests/test_fail_fast.py b/tests/test_fail_fast.py index 5fb0127d9..c543dbf9f 100644 --- a/tests/test_fail_fast.py +++ b/tests/test_fail_fast.py @@ -24,7 +24,7 @@ def test_fail_fast_unknown_method(self): def test_fail_fast_without_done(self): response = frontik_test_app.get_page('fail_fast/fail_fast_without_done') - assert response.status_code == 500 + assert response.status_code == 401 def test_fail_fast_default(self): response = frontik_test_app.get_page('fail_fast?fail_fast_default=true&code=400', method=requests.post) @@ -48,5 +48,5 @@ def test_exception_in_fail_fast(self) -> None: assert response.status_code == 500 def test_fail_fast_with_producer(self): - response = frontik_test_app.get_page('fail_fast/with_postprocessors') - assert response.status_code == 500 + response = frontik_test_app.get_page_json('fail_fast/with_postprocessors') + assert response['error'] == 'some_error' diff --git a/tests/test_handler.py b/tests/test_handler.py index 0d7642c64..8ca6ecc53 100644 --- a/tests/test_handler.py +++ b/tests/test_handler.py @@ -25,7 +25,7 @@ def test_no_method(self): def test_delete_post_arguments(self): response = frontik_test_app.get_page('handler/delete', method=requests.delete) - assert response.status_code == 400 + assert response.status_code == 422 def test_204(self): response = frontik_test_app.get_page('finish_204') diff --git a/tests/test_http_client.py b/tests/test_http_client.py index 3a74034cf..6643e200a 100644 --- a/tests/test_http_client.py +++ b/tests/test_http_client.py @@ -42,7 +42,7 @@ def test_parse_response(self): def test_custom_headers(self): json = frontik_test_app.get_page_json('http_client/custom_headers') - assert json['x-foo'] == 'Bar' + assert json['X-Foo'] == 'Bar' def test_http_client_method_future(self): json = frontik_test_app.get_page_json('http_client/future') diff --git a/tests/test_logging.py b/tests/test_logging.py index 0a587699c..04f019900 100644 --- a/tests/test_logging.py +++ b/tests/test_logging.py @@ -61,7 +61,7 @@ def test_send_to_syslog(self): parsed_logs[tag].append({'priority': priority, 'message': message}) expected_service_logs = [ - {'priority': '14', 'message': {'lvl': 'INFO', 'logger': r'frontik\.routing', 'msg': 'requested url: /log'}}, + {'priority': '14', 'message': {'lvl': 'INFO', 'logger': r'handler', 'msg': 'requested url: /log'}}, {'priority': '15', 'message': {'lvl': 'DEBUG', 'logger': r'handler', 'msg': 'debug'}}, {'priority': '14', 'message': {'lvl': 'INFO', 'logger': r'handler', 'msg': 'info'}}, { diff --git a/tests/test_mandatory_headers.py b/tests/test_mandatory_headers.py new file mode 100644 index 000000000..1bea41316 --- /dev/null +++ b/tests/test_mandatory_headers.py @@ -0,0 +1,32 @@ +from tests.instances import frontik_test_app + + +class TestPostprocessors: + def test_set_mandatory_headers(self): + response = frontik_test_app.get_page('mandatory_headers?test_mandatory_headers') + assert response.status_code == 500 + assert response.headers.get('TEST_HEADER') == 'TEST_HEADER_VALUE' + assert response.cookies.get('TEST_COOKIE') == 'TEST_HEADER_COOKIE' # type: ignore + + def test_mandatory_headers_are_lost(self) -> None: + response = frontik_test_app.get_page('mandatory_headers?test_without_mandatory_headers') + assert response.status_code == 500 + assert response.headers.get('TEST_HEADER') is None + assert response.headers.get('TEST_COOKIE') is None + + def test_mandatory_headers_are_cleared(self) -> None: + response = frontik_test_app.get_page('mandatory_headers?test_clear_set_mandatory_headers') + assert response.status_code == 500 + assert response.headers.get('TEST_HEADER') is None + assert response.headers.get('TEST_COOKIE') is None + + def test_clear_not_set_headers_does_not_faile(self) -> None: + response = frontik_test_app.get_page('mandatory_headers?test_clear_not_set_headers') + assert response.status_code == 500 + assert response.headers.get('TEST_HEADER') is None + assert response.headers.get('TEST_COOKIE') is None + + def test_invalid_mandatory_cookie(self): + response = frontik_test_app.get_page('mandatory_headers?test_invalid_mandatory_cookie') + assert response.status_code == 400 + assert response.headers.get('TEST_COOKIE') is None diff --git a/tests/test_sentry_integration.py b/tests/test_sentry_integration.py index 410d2bda9..8398e858f 100644 --- a/tests/test_sentry_integration.py +++ b/tests/test_sentry_integration.py @@ -4,7 +4,7 @@ import pytest import requests import sentry_sdk -from fastapi import HTTPException +from tornado.web import HTTPError from frontik.app import FrontikApplication from frontik.handler import PageHandler, get_current_handler @@ -33,7 +33,7 @@ async def get_page(handler: Page = get_current_handler()) -> None: @router.post('/sentry_error', cls=Page) async def post_page(): - raise HTTPException(500, 'my_HTTPError') + raise HTTPError(500, 'my_HTTPError') @router.put('/sentry_error', cls=Page) @@ -88,7 +88,7 @@ async def test_sentry_message(self): assert event.get('modules') is not None assert event['request']['url'].endswith('/sentry_error') is True assert event['request']['method'] == 'PUT' - assert event['request']['headers']['maheaderkey'] == 'MaHeaderValue' + assert event['request']['headers']['Maheaderkey'] == 'MaHeaderValue' assert event['extra']['extra_key'] == 'extra_value' assert event['user']['id'] == '123456' diff --git a/tests/test_telemetry.py b/tests/test_telemetry.py index 2150b0e1e..6964de345 100644 --- a/tests/test_telemetry.py +++ b/tests/test_telemetry.py @@ -6,14 +6,14 @@ from opentelemetry import trace from opentelemetry.sdk.resources import Resource from opentelemetry.sdk.trace import TracerProvider -from opentelemetry.sdk.trace.export import ReadableSpan, SpanExporter, SpanExportResult +from opentelemetry.sdk.trace.export import BatchSpanProcessor, ReadableSpan, SpanExporter, SpanExportResult from opentelemetry.sdk.trace.sampling import ParentBased, TraceIdRatioBased from opentelemetry.semconv.resource import ResourceAttributes from frontik import request_context from frontik.app import FrontikApplication from frontik.handler import PageHandler, get_current_handler -from frontik.integrations.telemetry import FrontikIdGenerator, FrontikSpanProcessor, get_netloc +from frontik.integrations.telemetry import FrontikIdGenerator, get_netloc from frontik.options import options from frontik.routing import router from frontik.testing import FrontikTestBase @@ -98,7 +98,7 @@ def make_otel_provider() -> TracerProvider: SPAN_STORAGE: list[ReadableSpan] = [] -BATCH_SPAN_PROCESSOR: list[FrontikSpanProcessor] = [] +BATCH_SPAN_PROCESSOR: list[BatchSpanProcessor] = [] def find_span(attr: str, value: Any) -> Optional[ReadableSpan]: @@ -126,7 +126,7 @@ def frontik_app(self) -> FrontikApplication: test_exporter = TestExporter() provider = make_otel_provider() - batch_span_processor = FrontikSpanProcessor(test_exporter) + batch_span_processor = BatchSpanProcessor(test_exporter) provider.add_span_processor(batch_span_processor) trace.set_tracer_provider(provider)