|
| 1 | +# coding=utf-8 |
| 2 | +import logging |
| 3 | +import sys |
| 4 | + |
| 5 | +import json_logging |
| 6 | +import json_logging.framework |
| 7 | +from json_logging import JSONLogWebFormatter |
| 8 | +from json_logging.framework_base import AppRequestInstrumentationConfigurator, RequestAdapter, ResponseAdapter |
| 9 | + |
| 10 | + |
| 11 | +def is_connexion_present(): |
| 12 | + # noinspection PyPep8,PyBroadException |
| 13 | + try: |
| 14 | + import connexion |
| 15 | + return True |
| 16 | + except: |
| 17 | + return False |
| 18 | + |
| 19 | + |
| 20 | +if is_connexion_present(): |
| 21 | + from connexion import request as request_obj |
| 22 | + import connexion as connexion |
| 23 | + from flask import g |
| 24 | + |
| 25 | + _current_request = request_obj |
| 26 | + _connexion = connexion |
| 27 | + _connexion.g = g |
| 28 | + |
| 29 | + |
| 30 | +class ConnexionAppRequestInstrumentationConfigurator(AppRequestInstrumentationConfigurator): |
| 31 | + def config(self, app): |
| 32 | + if not is_connexion_present(): |
| 33 | + raise RuntimeError("connexion is not available in system runtime") |
| 34 | + from flask.app import Flask |
| 35 | + if not isinstance(app.app, Flask): |
| 36 | + raise RuntimeError("app is not a valid connexion.app.Connexion app instance") |
| 37 | + |
| 38 | + # Disable standard logging |
| 39 | + logging.getLogger('werkzeug').disabled = True |
| 40 | + |
| 41 | + json_logging.util.update_formatter_for_loggers([logging.getLogger('werkzeug')], JSONLogWebFormatter) |
| 42 | + |
| 43 | + # noinspection PyAttributeOutsideInit |
| 44 | + self.request_logger = logging.getLogger('connexion-request-logger') |
| 45 | + self.request_logger.setLevel(logging.DEBUG) |
| 46 | + self.request_logger.addHandler(logging.StreamHandler(sys.stdout)) |
| 47 | + |
| 48 | + from flask import g |
| 49 | + |
| 50 | + @app.app.before_request |
| 51 | + def before_request(): |
| 52 | + g.request_info = json_logging.RequestInfo(_current_request) |
| 53 | + |
| 54 | + @app.app.after_request |
| 55 | + def after_request(response): |
| 56 | + request_info = g.request_info |
| 57 | + request_info.update_response_status(response) |
| 58 | + # TODO:handle to print out request instrumentation in non-JSON mode |
| 59 | + self.request_logger.info("", extra={'request_info': request_info}) |
| 60 | + return response |
| 61 | + |
| 62 | + |
| 63 | +class ConnexionRequestAdapter(RequestAdapter): |
| 64 | + @staticmethod |
| 65 | + def get_request_class_type(): |
| 66 | + raise NotImplementedError |
| 67 | + |
| 68 | + @staticmethod |
| 69 | + def support_global_request_object(): |
| 70 | + return True |
| 71 | + |
| 72 | + @staticmethod |
| 73 | + def get_current_request(): |
| 74 | + return _current_request |
| 75 | + |
| 76 | + def get_remote_user(self, request): |
| 77 | + if request.authorization is not None: |
| 78 | + return request.authorization.username |
| 79 | + else: |
| 80 | + return json_logging.EMPTY_VALUE |
| 81 | + |
| 82 | + def is_in_request_context(self, request_): |
| 83 | + return _connexion.has_request_context() |
| 84 | + |
| 85 | + def get_http_header(self, request, header_name, default=None): |
| 86 | + if header_name in request.headers: |
| 87 | + return request.headers.get(header_name) |
| 88 | + return default |
| 89 | + |
| 90 | + def set_correlation_id(self, request_, value): |
| 91 | + _connexion.g.correlation_id = value |
| 92 | + |
| 93 | + def get_correlation_id_in_request_context(self, request): |
| 94 | + return _connexion.g.get('correlation_id', None) |
| 95 | + |
| 96 | + def get_protocol(self, request): |
| 97 | + return request.environ.get('SERVER_PROTOCOL') |
| 98 | + |
| 99 | + def get_path(self, request): |
| 100 | + return request.path |
| 101 | + |
| 102 | + def get_content_length(self, request): |
| 103 | + return request.content_length |
| 104 | + |
| 105 | + def get_method(self, request): |
| 106 | + return request.method |
| 107 | + |
| 108 | + def get_remote_ip(self, request): |
| 109 | + return request.remote_addr |
| 110 | + |
| 111 | + def get_remote_port(self, request): |
| 112 | + return request.environ.get('REMOTE_PORT') |
| 113 | + |
| 114 | + |
| 115 | +class ConnexionResponseAdapter(ResponseAdapter): |
| 116 | + def get_status_code(self, response): |
| 117 | + return response.status_code |
| 118 | + |
| 119 | + def get_response_size(self, response): |
| 120 | + return response.calculate_content_length() |
| 121 | + |
| 122 | + def get_content_type(self, response): |
| 123 | + return response.content_type |
0 commit comments