關於flask線程安全的簡單研究

  flask是python web開發比較主流的框架之一,也是我在工做中使用的主要開發框架。一直對其是如何保證線程安全的問題比較好奇,因此簡單的探究了一番,因爲只是簡單查看了源碼,並未深刻細緻研究,所以如下內容僅爲我的理解,不保證正確性。python

  首先是不少文章都說flask會爲每個request啓動一個線程,每一個request都在單獨線程中處理,所以保證了線程安全。因而就作了一個簡單的測試。首先是寫一個簡單的flask程序(只須要有最簡單的功能用於測試便可),而後咱們知道一個flask應用啓動以後其實是做爲一個 WSGI application的,以後全部接收到的請求都會經由flask的wsgi_app(self, environ, start_response)方法去處理,因此就來看一下這個方法(註釋已去掉)。web

def wsgi_app(self, environ, start_response):
        ctx = self.request_context(environ)
        ctx.push()
        error = None
        try:
            try:
                response = self.full_dispatch_request()
            except Exception as e:
                error = e
                response = self.handle_exception(e)
            return response(environ, start_response)
        finally:
            if self.should_ignore_error(error):
                error = None
            ctx.auto_pop(error)

那麼這個request_context又是什麼東西呢?它是一個RequestContext對象,文檔是這麼說的:shell

The request context contains all request relevant information.  It is
created at the beginning of the request and pushed to the
`_request_ctx_stack` and removed at the end of it.  It will create the
URL adapter and request object for the WSGI environment provided.

說的很清楚,這個對象的上下文包含着request相關的信息。也就是說每個請求到來以後,flask都會爲它新建一個RequestContext對象,而且將這個對象push進全局變量_request_ctx_stack中,在push前還要檢查_app_ctx_stack,若是_app_ctx_stack的棧頂元素不存在或是與當前的應用不一致,則首先push appcontext 到_app_ctx_stack中,再push requestcontext。源碼以下:flask

def push(self):

        top = _request_ctx_stack.top
        if top is not None and top.preserved:
            top.pop(top._preserved_exc)

        # Before we push the request context we have to ensure that there
        # is an application context.
        app_ctx = _app_ctx_stack.top
        if app_ctx is None or app_ctx.app != self.app:
            app_ctx = self.app.app_context()
            app_ctx.push()
            self._implicit_app_ctx_stack.append(app_ctx)
        else:
            self._implicit_app_ctx_stack.append(None)
        if hasattr(sys, 'exc_clear'):
            sys.exc_clear()

        _request_ctx_stack.push(self)
        # Open the session at the moment that the request context is
        # available. This allows a custom open_session method to use the
        # request context (e.g. code that access database information
        # stored on `g` instead of the appcontext).
        self.session = self.app.open_session(self.request)
        if self.session is None:
            self.session = self.app.make_null_session()

經過上面的兩步,每個請求的應用上下文和請求上下文就被push到了全局變量_request_ctx_stack和_app_ctx_stack中。
  如今咱們知道了_request_ctx_stack和_app_ctx_stack是什麼時候被push的,每個請求到來都會致使新的RequestContext和AppContext被創建並push,一旦請求處理完畢就被pop出去。而不管是_app_ctx_stack仍是_request_ctx_stack都是一個LocalStack對象,這是werkzeug中的一個對象,看看它裏邊有什麼:安全

class LocalStack(object):

    def __init__(self):
        self._local = Local()

    def __release_local__(self):
        self._local.__release_local__()

    def _get__ident_func__(self):
        return self._local.__ident_func__

    def _set__ident_func__(self, value):
        object.__setattr__(self._local, '__ident_func__', value)
    __ident_func__ = property(_get__ident_func__, _set__ident_func__)
    del _get__ident_func__, _set__ident_func__

    def __call__(self):
        def _lookup():
            rv = self.top
            if rv is None:
                raise RuntimeError('object unbound')
            return rv
        return LocalProxy(_lookup)

    def push(self, obj):
        """Pushes a new item to the stack"""
        rv = getattr(self._local, 'stack', None)
        if rv is None:
            self._local.stack = rv = []
        rv.append(obj)
        return rv

    def pop(self):
        """Removes the topmost item from the stack, will return the
        old value or `None` if the stack was already empty.
        """
        stack = getattr(self._local, 'stack', None)
        if stack is None:
            return None
        elif len(stack) == 1:
            release_local(self._local)
            return stack[-1]
        else:
            return stack.pop()

    @property
    def top(self):
        """The topmost item on the stack.  If the stack is empty,
        `None` is returned.
        """
        try:
            return self._local.stack[-1]
        except (AttributeError, IndexError):
            return None

  能夠看到,這個對象的幾乎全部重要屬性在_local這一屬性中,它是一個Local對象,頗有意思,若是看一下Local的構造器,會發現其中包含有重要屬性__ident_func__,session

def __init__(self):
object.__setattr__(self, '__storage__', {})
object.__setattr__(self, '__ident_func__', get_ident)

 

   這一屬性由get_ident方法提供,這個方法的做用是提供當前線程的id,用於區別同時存在的多個線程Return a non-zero integer that uniquely identifiamongst other threads that exist simultaneously.多線程


  
到此爲止,可見做爲一個全局變量_request_ctx_stack和_app_ctx_stack應該都是隻有一個線程去處理,沒有發現哪裏有能夠爲每一個請求都開啓一個線程的代碼,實際測試一下,能夠發現確實全部的請求都只運行在一個線程上(使用pycharm的debug模式能夠看到當前程序啓動
的全部線程,在當前這種情型下除了主線程外只有一個Thread-6,不管多少請求都同樣)

 
這下就有趣了,傳說中的每一個請求一個線程果真沒有出現,那麼flask的線程安全是如何保證的呢?若是把每次請求到來時附帶的environ(wsgi_app方法參數中的environ)打印看看的話就會發現,每一個environ都攜帶了請求相關的所有上下文信息,在請求到來的時候經過附帶的
environ重建context,並push到棧中,而後馬上處理該請求,處理完畢後將其pop出去

  
那麼不少文章說的每一個請求一個線程究竟是在哪裏創建的呢?這就要去仔細看一下flask.app的run方法了:
    def run(self, host=None, port=None, debug=None, **options):
        
        from werkzeug.serving import run_simple
        if host is None:
            host = '127.0.0.1'
        if port is None:
            server_name = self.config['SERVER_NAME']
            if server_name and ':' in server_name:
                port = int(server_name.rsplit(':', 1)[1])
            else:
                port = 5000
        if debug is not None:
            self.debug = bool(debug)
        options.setdefault('use_reloader', self.debug)
        options.setdefault('use_debugger', self.debug)
        try:
            run_simple(host, port, self, **options)
        finally:
            # reset the first request information if the development server
            # reset normally.  This makes it possible to restart the server
            # without reloader and that stuff from an interactive shell.
            self._got_first_request = False
 

 

這個方法其實是對werkzeug的run_simple方法的簡單包裝。而run_simple方法則有趣的多(這一段把註釋也貼上)
def run_simple(hostname, port, application, use_reloader=False,
               use_debugger=False, use_evalex=True,
               extra_files=None, reloader_interval=1,
               reloader_type='auto', threaded=False,
               processes=1, request_handler=None, static_files=None,
               passthrough_errors=False, ssl_context=None):
    """Start a WSGI application. Optional features include a reloader,
    multithreading and fork support.

    This function has a command-line interface too::

        python -m werkzeug.serving --help

    .. versionadded:: 0.5
       `static_files` was added to simplify serving of static files as well
       as `passthrough_errors`.

    .. versionadded:: 0.6
       support for SSL was added.

    .. versionadded:: 0.8
       Added support for automatically loading a SSL context from certificate
       file and private key.

    .. versionadded:: 0.9
       Added command-line interface.

    .. versionadded:: 0.10
       Improved the reloader and added support for changing the backend
       through the `reloader_type` parameter.  See :ref:`reloader`
       for more information.

    :param hostname: The host for the application.  eg: ``'localhost'``
    :param port: The port for the server.  eg: ``8080``
    :param application: the WSGI application to execute
    :param use_reloader: should the server automatically restart the python
                         process if modules were changed?
    :param use_debugger: should the werkzeug debugging system be used?
    :param use_evalex: should the exception evaluation feature be enabled?
    :param extra_files: a list of files the reloader should watch
                        additionally to the modules.  For example configuration
                        files.
    :param reloader_interval: the interval for the reloader in seconds.
    :param reloader_type: the type of reloader to use.  The default is
                          auto detection.  Valid values are ``'stat'`` and
                          ``'watchdog'``. See :ref:`reloader` for more
                          information.
    :param threaded: should the process handle each request in a separate
                     thread?
    :param processes: if greater than 1 then handle each request in a new process
                      up to this maximum number of concurrent processes.
    :param request_handler: optional parameter that can be used to replace
                            the default one.  You can use this to replace it
                            with a different
                            :class:`~BaseHTTPServer.BaseHTTPRequestHandler`
                            subclass.
    :param static_files: a dict of paths for static files.  This works exactly
                         like :class:`SharedDataMiddleware`, it's actually
                         just wrapping the application in that middleware before
                         serving.
    :param passthrough_errors: set this to `True` to disable the error catching.
                               This means that the server will die on errors but
                               it can be useful to hook debuggers in (pdb etc.)
    :param ssl_context: an SSL context for the connection. Either an
                        :class:`ssl.SSLContext`, a tuple in the form
                        ``(cert_file, pkey_file)``, the string ``'adhoc'`` if
                        the server should automatically create one, or ``None``
                        to disable SSL (which is the default).
    """
    if use_debugger:
        from werkzeug.debug import DebuggedApplication
        application = DebuggedApplication(application, use_evalex)
    if static_files:
        from werkzeug.wsgi import SharedDataMiddleware
        application = SharedDataMiddleware(application, static_files)

    def log_startup(sock):
        display_hostname = hostname not in ('', '*') and hostname or 'localhost'
        if ':' in display_hostname:
            display_hostname = '[%s]' % display_hostname
        quit_msg = '(Press CTRL+C to quit)'
        port = sock.getsockname()[1]
        _log('info', ' * Running on %s://%s:%d/ %s',
             ssl_context is None and 'http' or 'https',
             display_hostname, port, quit_msg)

    def inner():
        try:
            fd = int(os.environ['WERKZEUG_SERVER_FD'])
        except (LookupError, ValueError):
            fd = None
        srv = make_server(hostname, port, application, threaded,
                          processes, request_handler,
                          passthrough_errors, ssl_context,
                          fd=fd)
        if fd is None:
            log_startup(srv.socket)
        srv.serve_forever()

    if use_reloader:
        # If we're not running already in the subprocess that is the
        # reloader we want to open up a socket early to make sure the
        # port is actually available.
        if os.environ.get('WERKZEUG_RUN_MAIN') != 'true':
            if port == 0 and not can_open_by_fd:
                raise ValueError('Cannot bind to a random port with enabled '
                                 'reloader if the Python interpreter does '
                                 'not support socket opening by fd.')

            # Create and destroy a socket so that any exceptions are
            # raised before we spawn a separate Python interpreter and
            # lose this ability.
            address_family = select_ip_version(hostname, port)
            s = socket.socket(address_family, socket.SOCK_STREAM)
            s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
            s.bind((hostname, port))
            if hasattr(s, 'set_inheritable'):
                s.set_inheritable(True)

            # If we can open the socket by file descriptor, then we can just
            # reuse this one and our socket will survive the restarts.
            if can_open_by_fd:
                os.environ['WERKZEUG_SERVER_FD'] = str(s.fileno())
                s.listen(LISTEN_QUEUE)
                log_startup(s)
            else:
                s.close()

        from ._reloader import run_with_reloader
        run_with_reloader(inner, extra_files, reloader_interval,
                          reloader_type)
    else:
        inner()
 

 

默認狀況下會執行inner方法,inner方法建立了一個server並啓動,這樣一個flask應用算是真正的啓動了。那麼祕密就在make_server裏了
def make_server(host=None, port=None, app=None, threaded=False, processes=1,
                request_handler=None, passthrough_errors=False,
                ssl_context=None, fd=None):
    """Create a new server instance that is either threaded, or forks
    or just processes one request after another.
    """
    if threaded and processes > 1:
        raise ValueError("cannot have a multithreaded and "
                         "multi process server.")
    elif threaded:
        return ThreadedWSGIServer(host, port, app, request_handler,
                                  passthrough_errors, ssl_context, fd=fd)
    elif processes > 1:
        return ForkingWSGIServer(host, port, app, processes, request_handler,
                                 passthrough_errors, ssl_context, fd=fd)
    else:
        return BaseWSGIServer(host, port, app, request_handler,
                              passthrough_errors, ssl_context, fd=fd)
 

 

好了,這一下咱們一直以來的疑問就找到答案了,原來一個flask應用的server並不是只有一種類型,它是能夠設定的,默認狀況下建立的是一個 BaseWSGIServer,若是指定了threaded參數就啓動一個ThreadedWSGIServer,若是設定的processes>1則啓動一個ForkingWSGIServer。
事已至此,後面的事情就是追本溯源了:
class ThreadedWSGIServer(ThreadingMixIn, BaseWSGIServer):

    """A WSGI server that does threading."""
    multithread = True

 

 
ThreadedWSGIServer是ThreadingMixIn和BaseWSGIServer的子類,
class ThreadingMixIn:
    """Mix-in class to handle each request in a new thread."""

    # Decides how threads will act upon termination of the
    # main process
    daemon_threads = False

    def process_request_thread(self, request, client_address):
        """Same as in BaseServer but as a thread.

        In addition, exception handling is done here.

        """
        try:
            self.finish_request(request, client_address)
            self.shutdown_request(request)
        except:
            self.handle_error(request, client_address)
            self.shutdown_request(request)

    def process_request(self, request, client_address):
        """Start a new thread to process the request."""
        t = threading.Thread(target = self.process_request_thread,
                             args = (request, client_address))
        t.daemon = self.daemon_threads
        t.start()

 

 
源碼寫的太明白了,原來是ThreadingMixIn的實例以多線程的方式去處理每個請求,這樣對開發者來講,只有在啓動app時將threaded參數設定爲True,flask纔會真正以多線程的方式去處理每個請求。
  實際去測試一下,發現將threaded設置沒True後,
果真每個請求都會開啓一個單獨的線程去處理。
相關文章
相關標籤/搜索