def load_specs(self, swagger_spec): self.specs = SpecLoader(log=self.log).load_specs(swagger_spec) self.basePath = self.specs['basePath']
class SwaggerServer: def __init__(self, errors=None, cors_origin=None): self.default_content_type = 'application/json' self.specs = {} # Meta() self.custom_error_map = errors self.op_loader = OperatorLoader() self.log = JLog().bind() self.cors_origin = cors_origin def __call__(self, req, resp): # , **kwargs): self.log.debug('remote_addr:{}, uri:{}, method:{}'.format( req.remote_addr, req.uri, req.method)) self.process(req, resp) def load_specs(self, swagger_spec): self.specs = SpecLoader(log=self.log).load_specs(swagger_spec) self.basePath = self.specs['basePath'] def process(self, req, resp): if req.method == 'OPTIONS': if self.cors_origin is not False: self.process_preflight_request(req, resp) response_body = '\n' response_body += 'nothing here\n\n' resp.body = response_body resp.status = falcon.HTTP_200 return try: if self.cors_origin is not False: self.process_preflight_request(req, resp) self.dispatch(req, resp) except Exception as e: self.log.error_trace('process failed') error_type = type(e) error_map = { falcon.errors.HTTPNotFound: http_falcon_handler, falcon.errors.HTTPMissingParam: http_falcon_handler, falcon.errors.HTTPInvalidParam: http_falcon_handler, falcon.errors.HTTPInternalServerError: http_falcon_handler, } if self.custom_error_map: error_map.update(self.custom_error_map) error_func = error_map.get(error_type) if error_func: error_func(req, resp, e) else: default_error_handler(req, resp, e) def process_preflight_request(self, req, resp): self.log.info("option request: ".format(req.relative_uri)) resp.set_header('Vary', 'Origin') resp.set_header('Access-Control-Allow-Origin', self.allowed_origin(req)) resp.set_header('Access-Control-Allow-Credentials', 'true') resp.set_header('Access-Control-Allow-Methods', 'GET, POST, PUT, PATCH, DELETE, OPTIONS') resp.set_header( 'Access-Control-Allow-Headers', 'Authorization, X-Auth-Token, Keep-Alive, Users-Agent, X-Requested-With, If-Modified-Since, Cache-Control, Content-Type' ) # resp.set_header('Access-Control-Max-Age', 1728000) # 20 days def allowed_origin(self, req): if type(self.cors_origin) == str: return self.cors_origin host = req.env['SERVER_NAME'] + ':' + req.env['SERVER_PORT'] return req.env['wsgi.url_scheme'] + '://' + host def dispatch(self, req, resp): base_before, base_after, base_excp, base_final = self.op_loader.load_base( self.specs) for uri_regex, spec in self.specs.items(): # try: route_signature = '/' + req.method.lower() + req.relative_uri if route_signature.find('?') > 0: route_signature = route_signature[:route_signature.find('?')] if type(uri_regex) == str: continue spec['route_signature'] = route_signature req.spec = copy.deepcopy(spec) match = uri_regex.match(route_signature) if match: handler, params, before, after, excp, final, mode = self.op_loader.load( req=req, spec=spec, matched_uri=match) handler_return = None try: if base_before: base_before(req=req, resp=resp, **params) if before: before(req=req, resp=resp, **params) if mode == 'raw': handler_return = handler(req=req, resp=resp) else: if mode == 'more': handler_return = handler(req=req, resp=resp, **params) else: handler_return = handler(**params) content_type = self.produces( spec.get('produces'), self.specs.get('produces')) self.process_response(req, resp, handler_return, content_type) if after: after(req=req, resp=resp, response=handler_return, **params) if base_after: base_after(req=req, resp=resp, **params) except Exception as e: throw_out = True if base_excp is not None: throw_out = base_excp(req=req, resp=resp, error=e) if excp is not None: throw_out = excp(req=req, resp=resp, error=e) if throw_out: raise e finally: if final: final(req=req, resp=resp, response=handler_return, **params) if base_final: base_final(req=req, resp=resp, **params) return # except falcon.HTTPInvalidParam as e: # self.log.error_trace("http invalid param: {}".format(e)) # raise e # except Exception as e: # self.log.error_trace("process error: {}".format(e)) # raise falcon.HTTPInternalServerError(title=str(type(e)), description=str(e)) self.log.info( "url does not match any route signature or match error: {}".format( route_signature)) raise falcon.HTTPNotFound() def process_response(self, req, resp, handler_return, content_type='application/json'): # content_type = 'text/plain' if handler_return is None: return if type(handler_return) == tuple: data = handler_return[0] http_code = handler_return[1] if len(handler_return) > 2: content_type = handler_return[2] else: data = handler_return http_code = falcon.HTTP_200 # if type(data) == dict or type(data) == list: # content_type = 'application/json' if resp.body: try: pre_body = json.loads(resp.body) except Exception as e: pre_body = resp.body if type(pre_body) == dict: if 'json' in content_type: pre_body.update(data) resp.body = json.dumps(pre_body, indent=2, ensure_ascii=False) else: resp.body = json.dumps(pre_body, ensure_ascii=False) + data else: resp.body = pre_body + json.dumps( data, indent=2, ensure_ascii=False ) if 'json' in content_type else json.dumps( pre_body, indent=2, ensure_ascii=False) + str(data) else: if 'json' in content_type or type(data) == dict: resp.body = json.dumps(data, indent=2, ensure_ascii=False) elif type(data) == str or type(data) == bytes: resp.body = data else: resp.body = str(data) resp.content_type = content_type resp.status = http_code def produces(self, mp=None, gp=None): if mp is not None: return mp[0] if gp is not None: return gp[0] return 'application/json'