mirror of
				https://github.com/langgenius/dify.git
				synced 2025-11-04 04:43:09 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			116 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			116 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import re
 | 
						|
import sys
 | 
						|
 | 
						|
from flask import got_request_exception, current_app
 | 
						|
from flask_restful import Api, http_status_message
 | 
						|
from werkzeug.datastructures import Headers
 | 
						|
from werkzeug.exceptions import HTTPException
 | 
						|
 | 
						|
 | 
						|
class ExternalApi(Api):
 | 
						|
 | 
						|
    def handle_error(self, e):
 | 
						|
        """Error handler for the API transforms a raised exception into a Flask
 | 
						|
        response, with the appropriate HTTP status code and body.
 | 
						|
 | 
						|
        :param e: the raised Exception object
 | 
						|
        :type e: Exception
 | 
						|
 | 
						|
        """
 | 
						|
        got_request_exception.send(current_app, exception=e)
 | 
						|
 | 
						|
        headers = Headers()
 | 
						|
        if isinstance(e, HTTPException):
 | 
						|
            if e.response is not None:
 | 
						|
                resp = e.get_response()
 | 
						|
                return resp
 | 
						|
 | 
						|
            status_code = e.code
 | 
						|
            default_data = {
 | 
						|
                'code': re.sub(r'(?<!^)(?=[A-Z])', '_', type(e).__name__).lower(),
 | 
						|
                'message': getattr(e, 'description', http_status_message(status_code)),
 | 
						|
                'status': status_code
 | 
						|
            }
 | 
						|
            headers = e.get_response().headers
 | 
						|
        elif isinstance(e, ValueError):
 | 
						|
            status_code = 400
 | 
						|
            default_data = {
 | 
						|
                'code': 'invalid_param',
 | 
						|
                'message': str(e),
 | 
						|
                'status': status_code
 | 
						|
            }
 | 
						|
        else:
 | 
						|
            status_code = 500
 | 
						|
            default_data = {
 | 
						|
                'message': http_status_message(status_code),
 | 
						|
            }
 | 
						|
 | 
						|
        # Werkzeug exceptions generate a content-length header which is added
 | 
						|
        # to the response in addition to the actual content-length header
 | 
						|
        # https://github.com/flask-restful/flask-restful/issues/534
 | 
						|
        remove_headers = ('Content-Length',)
 | 
						|
 | 
						|
        for header in remove_headers:
 | 
						|
            headers.pop(header, None)
 | 
						|
 | 
						|
        data = getattr(e, 'data', default_data)
 | 
						|
 | 
						|
        error_cls_name = type(e).__name__
 | 
						|
        if error_cls_name in self.errors:
 | 
						|
            custom_data = self.errors.get(error_cls_name, {})
 | 
						|
            custom_data = custom_data.copy()
 | 
						|
            status_code = custom_data.get('status', 500)
 | 
						|
 | 
						|
            if 'message' in custom_data:
 | 
						|
                custom_data['message'] = custom_data['message'].format(
 | 
						|
                    message=str(e.description if hasattr(e, 'description') else e)
 | 
						|
                )
 | 
						|
            data.update(custom_data)
 | 
						|
 | 
						|
        # record the exception in the logs when we have a server error of status code: 500
 | 
						|
        if status_code and status_code >= 500:
 | 
						|
            exc_info = sys.exc_info()
 | 
						|
            if exc_info[1] is None:
 | 
						|
                exc_info = None
 | 
						|
            current_app.log_exception(exc_info)
 | 
						|
 | 
						|
        if status_code == 406 and self.default_mediatype is None:
 | 
						|
            # if we are handling NotAcceptable (406), make sure that
 | 
						|
            # make_response uses a representation we support as the
 | 
						|
            # default mediatype (so that make_response doesn't throw
 | 
						|
            # another NotAcceptable error).
 | 
						|
            supported_mediatypes = list(self.representations.keys())  # only supported application/json
 | 
						|
            fallback_mediatype = supported_mediatypes[0] if supported_mediatypes else "text/plain"
 | 
						|
            data = {
 | 
						|
                'code': 'not_acceptable',
 | 
						|
                'message': data.get('message')
 | 
						|
            }
 | 
						|
            resp = self.make_response(
 | 
						|
                data,
 | 
						|
                status_code,
 | 
						|
                headers,
 | 
						|
                fallback_mediatype = fallback_mediatype
 | 
						|
            )
 | 
						|
        elif status_code == 400:
 | 
						|
            if isinstance(data.get('message'), dict):
 | 
						|
                param_key, param_value = list(data.get('message').items())[0]
 | 
						|
                data = {
 | 
						|
                    'code': 'invalid_param',
 | 
						|
                    'message': param_value,
 | 
						|
                    'params': param_key
 | 
						|
                }
 | 
						|
            else:
 | 
						|
                if 'code' not in data:
 | 
						|
                    data['code'] = 'unknown'
 | 
						|
 | 
						|
            resp = self.make_response(data, status_code, headers)
 | 
						|
        else:
 | 
						|
            if 'code' not in data:
 | 
						|
                data['code'] = 'unknown'
 | 
						|
 | 
						|
            resp = self.make_response(data, status_code, headers)
 | 
						|
 | 
						|
        if status_code == 401:
 | 
						|
            resp = self.unauthorized(resp)
 | 
						|
        return resp
 |