django的web server的源代碼流程

 1 class WSGIServer(simple_server.WSGIServer):
 2     """BaseHTTPServer that implements the Python WSGI protocol"""
 3 
 4     request_queue_size = 10
 5 
 6     def __init__(self, *args, ipv6=False, allow_reuse_address=True, **kwargs):
 7         if ipv6:
 8             self.address_family = socket.AF_INET6
 9         self.allow_reuse_address = allow_reuse_address
10         super().__init__(*args, **kwargs)
11 
12     def handle_error(self, request, client_address):
13         if is_broken_pipe_error():
14             logger.info("- Broken pipe from %s\n", client_address)
15         else:
16             super().handle_error(request, client_address)
17 
18 class WSGIRequestHandler(simple_server.WSGIRequestHandler):
19     protocol_version = 'HTTP/1.1'
20 
21     def address_string(self):
22         # Short-circuit parent method to not call socket.getfqdn
23         return self.client_address[0]
24 
25     def log_message(self, format, *args):
26         extra = {
27             'request': self.request,
28             'server_time': self.log_date_time_string(),
29         }
30         if args[1][0] == '4':
31             # 0x16 = Handshake, 0x03 = SSL 3.0 or TLS 1.x
32             if args[0].startswith('\x16\x03'):
33                 extra['status_code'] = 500
34                 logger.error(
35                     "You're accessing the development server over HTTPS, but "
36                     "it only supports HTTP.\n", extra=extra,
37                 )
38                 return
39 
40         if args[1].isdigit() and len(args[1]) == 3:
41             status_code = int(args[1])
42             extra['status_code'] = status_code
43 
44             if status_code >= 500:
45                 level = logger.error
46             elif status_code >= 400:
47                 level = logger.warning
48             else:
49                 level = logger.info
50         else:
51             level = logger.info
52 
53         level(format, *args, extra=extra)
54 
55     def get_environ(self):
56         # Strip all headers with underscores in the name before constructing
57         # the WSGI environ. This prevents header-spoofing based on ambiguity
58         # between underscores and dashes both normalized to underscores in WSGI
59         # env vars. Nginx and Apache 2.4+ both do this as well.
60         for k in self.headers:
61             if '_' in k:
62                 del self.headers[k]
63 
64         return super().get_environ()
65 
66     def handle(self):
67         self.close_connection = True
68         self.handle_one_request()
69         while not self.close_connection:
70             self.handle_one_request()
71         try:
72             self.connection.shutdown(socket.SHUT_WR)
73         except (socket.error, AttributeError):
74             pass
75 
76     def handle_one_request(self):
77         """Copy of WSGIRequestHandler.handle() but with different ServerHandler"""
78         self.raw_requestline = self.rfile.readline(65537)
79         if len(self.raw_requestline) > 65536:
80             self.requestline = ''
81             self.request_version = ''
82             self.command = ''
83             self.send_error(414)
84             return
85 
86         if not self.parse_request():  # An error code has been sent, just exit
87             return
88 
89         handler = ServerHandler(
90             self.rfile, self.wfile, self.get_stderr(), self.get_environ()
91         )
92         handler.request_handler = self      # backpointer for logging & connection closing
93         handler.run(self.server.get_app())
WSGIServer的父類是wsgiref.simple_server.WSGIServer, wsgiref.simple_server.WSGIServer的父類是http.server.HTTPServer,http.server.HTTPServer的父類是socketserver.TCPServer;
WSGIRequestHandler的父類是wsgiref.simple_server.WSGIRequestHandler,wsgiref.simple_server.WSGIRequestHandler的父類是http.server.BaseHTTPRequestHandler,
http.server.BaseHTTPRequestHandler的父類是socketserver.StreamRequestHandler,socketserver.StreamRequestHandler的父類是socketserver.BaseRequestHandler
def run(addr, port, wsgi_handler, ipv6=False, threading=False, server_cls=WSGIServer):
    server_address = (addr, port)
    if threading:
        httpd_cls = type('WSGIServer', (socketserver.ThreadingMixIn, server_cls), {})
    else:
        httpd_cls = server_cls
    httpd = httpd_cls(server_address, WSGIRequestHandler, ipv6=ipv6)  實例化WSGIserver
    if threading:
        # ThreadingMixIn.daemon_threads indicates how threads will behave on an
        # abrupt shutdown; like quitting the server by the user or restarting
        # by the auto-reloader. True means the server will not wait for thread
        # termination before it quits. This will make auto-reloader faster
        # and will prevent the need to kill the server manually if a thread
        # isn't terminating correctly.
        httpd.daemon_threads = True
    httpd.set_app(wsgi_handler)
    httpd.serve_forever()  #這裏調用TCPserver的父類BaseServer的serve_forever方法

在socketserver.BaseServer.serve_forever中git

 1     def serve_forever(self, poll_interval=0.5):
 2         """Handle one request at a time until shutdown.
 3 
 4         Polls for shutdown every poll_interval seconds. Ignores
 5         self.timeout. If you need to do periodic tasks, do them in
 6         another thread.
 7         """
 8         self.__is_shut_down.clear()
 9         try:
10             # XXX: Consider using another file descriptor or connecting to the
11             # socket to wake this up instead of polling. Polling reduces our
12             # responsiveness to a shutdown request and wastes cpu at all other
13             # times.
14             with _ServerSelector() as selector:
15                 selector.register(self, selectors.EVENT_READ)
16 
17                 while not self.__shutdown_request:
18                     ready = selector.select(poll_interval)
19                     # bpo-35017: shutdown() called during select(), exit immediately.
20                     if self.__shutdown_request:
21                         break
22                     if ready:
23                         self._handle_request_noblock()  在這裏調用process_request方法
24 
25                     self.service_actions()
26         finally:
27             self.__shutdown_request = False
28             self.__is_shut_down.set()

在process_request方法中調用finish_request方法去實例化WSGIRequestHandlercookie

1     def process_request(self, request, client_address):
2         """Call finish_request.
3 
4         Overridden by ForkingMixIn and ThreadingMixIn.
5 
6         """
7         self.finish_request(request, client_address)
8         self.shutdown_request(request)

在finish_request中實例化WSGIRequestHandlerapp

    def finish_request(self, request, client_address):
        """Finish one request by instantiating RequestHandlerClass."""
        self.RequestHandlerClass(request, client_address, self)  #這個self是WSGIserver實例化的對象
class BaseRequestHandler:  #實例化WSGIRequestHandler必須先初始化BaseRequestHandler類

    """Base class for request handler classes.

    This class is instantiated for each request to be handled.  The
    constructor sets the instance variables request, client_address
    and server, and then calls the handle() method.  To implement a
    specific service, all you need to do is to derive a class which
    defines a handle() method.

    The handle() method can find the request as self.request, the
    client address as self.client_address, and the server (in case it
    needs access to per-server information) as self.server.  Since a
    separate instance is created for each request, the handle() method
    can define other arbitrary instance variables.

    """

    def __init__(self, request, client_address, server):
        self.request = request
        self.client_address = client_address
        self.server = server
        self.setup()
        try:
            self.handle()  #這個類沒有實現handle方法,因此須要從WSGIRequestHandler裏面去找
        finally:
            self.finish()

    def setup(self):
        pass

    def handle(self):
        pass

    def finish(self):
        pass

class StreamRequestHandler(BaseRequestHandler):

    """Define self.rfile and self.wfile for stream sockets."""

    # Default buffer sizes for rfile, wfile.
    # We default rfile to buffered because otherwise it could be
    # really slow for large data (a getc() call per byte); we make
    # wfile unbuffered because (a) often after a write() we want to
    # read and we need to flush the line; (b) big writes to unbuffered
    # files are typically optimized by stdio even when big reads
    # aren't.
    rbufsize = -1
    wbufsize = 0

    # A timeout to apply to the request socket, if not None.
    timeout = None

    # Disable nagle algorithm for this socket, if True.
    # Use only when wbufsize != 0, to avoid small packets.
    disable_nagle_algorithm = False

    def setup(self):
        self.connection = self.request
        if self.timeout is not None:
            self.connection.settimeout(self.timeout)
        if self.disable_nagle_algorithm:
            self.connection.setsockopt(socket.IPPROTO_TCP,
                                       socket.TCP_NODELAY, True)
        self.rfile = self.connection.makefile('rb', self.rbufsize)
        if self.wbufsize == 0:
            self.wfile = _SocketWriter(self.connection)
        else:
            self.wfile = self.connection.makefile('wb', self.wbufsize)

    def finish(self):
        if not self.wfile.closed:
            try:
                self.wfile.flush()
            except socket.error:
                # A final socket error may have occurred here, such as
                # the local error ECONNABORTED.
                pass
        self.wfile.close()
        self.rfile.close()

在handle方法中socket

 1 class WSGIRequestHandler(BaseHTTPRequestHandler):
 2 
 3     server_version = "WSGIServer/" + __version__
 4 
 5     def get_environ(self):
 6         env = self.server.base_environ.copy()
 7         env['SERVER_PROTOCOL'] = self.request_version
 8         env['SERVER_SOFTWARE'] = self.server_version
 9         env['REQUEST_METHOD'] = self.command
10         if '?' in self.path:
11             path,query = self.path.split('?',1)
12         else:
13             path,query = self.path,''
14 
15         env['PATH_INFO'] = urllib.parse.unquote(path, 'iso-8859-1')
16         env['QUERY_STRING'] = query
17 
18         host = self.address_string()
19         if host != self.client_address[0]:
20             env['REMOTE_HOST'] = host
21         env['REMOTE_ADDR'] = self.client_address[0]
22 
23         if self.headers.get('content-type') is None:
24             env['CONTENT_TYPE'] = self.headers.get_content_type()
25         else:
26             env['CONTENT_TYPE'] = self.headers['content-type']
27 
28         length = self.headers.get('content-length')
29         if length:
30             env['CONTENT_LENGTH'] = length
31 
32         for k, v in self.headers.items():
33             k=k.replace('-','_').upper(); v=v.strip()
34             if k in env:
35                 continue                    # skip content length, type,etc.
36             if 'HTTP_'+k in env:
37                 env['HTTP_'+k] += ','+v     # comma-separate multiple headers
38             else:
39                 env['HTTP_'+k] = v
40         return env
41 
42     def get_stderr(self):
43         return sys.stderr
44 
45     def handle(self):
46         """Handle a single HTTP request"""
47 
48         self.raw_requestline = self.rfile.readline(65537)  #讀取遊覽器發來的請求包文起始
49         if len(self.raw_requestline) > 65536:  #判斷數據長度是否大於65536
50             self.requestline = ''
51             self.request_version = ''
52             self.command = ''
53             self.send_error(414)   #發送414錯誤
54             return
55 
56         if not self.parse_request(): # An error code has been sent, just exit
57             return
58 
59         handler = ServerHandler(
60             self.rfile, self.wfile, self.get_stderr(), self.get_environ()
61         )  #rfile至關於socket中的recv,wfile至關於socket中的send,   get_atderr是系統的標準錯誤輸出句柄, get_environ是先將下劃線開頭的頭部字段刪除,再調用父類的get_envrion方法將請求的頭部字段加上HTTP_,再添加一些公共字段
62         handler.request_handler = self      # backpointer for logging
63         handler.run(self.server.get_app())

在run中async

 1     def run(self, application):
 2         """Invoke the application"""
 3         # Note to self: don't move the close()!  Asynchronous servers shouldn't
 4         # call close() from finish_response(), so if you close() anywhere but
 5         # the double-error branch here, you'll break asynchronous servers by
 6         # prematurely closing.  Async servers must return from 'run()' without
 7         # closing if there might still be output to iterate over.
 8         try:
 9             self.setup_environ()  #設置環境,將系統環境拷貝,而後將傳入的環境變量擴展進該環境內同時增長一些WSGI通用環境
10             self.result = application(self.environ, self.start_response)#設置完環境後開始執行咱們傳入的handler對象
11             self.finish_response()
12         except:
13             try:
14                 self.handle_error()
15             except:
16                 # If we get an error handling an error, just give up already!
17                 self.close()
18                 raise   # ...and let the actual server figure it out.

也就是執行如下代碼ide

 1 class WSGIHandler(base.BaseHandler):
 2     request_class = WSGIRequest
 3 
 4     def __init__(self, *args, **kwargs):
 5         super(WSGIHandler, self).__init__(*args, **kwargs)
 6         self.load_middleware()
 7 
 8     def __call__(self, environ, start_response):
 9         set_script_prefix(get_script_name(environ))
10         signals.request_started.send(sender=self.__class__, environ=environ)
11         request = self.request_class(environ)
12         response = self.get_response(request)
13         response._handler_class = self.__class__
14 
15         status = '%d %s' % (response.status_code, response.reason_phrase)
16         response_headers = [(str(k), str(v)) for k, v in response.items()]
17         for c in response.cookies.values():
18             response_headers.append((str('Set-Cookie'), str(c.output(header=''))))
19         start_response(force_str(status), response_headers)
20         if getattr(response, 'file_to_stream', None) is not None and environ.get('wsgi.file_wrapper'):
21             response = environ['wsgi.file_wrapper'](response.file_to_stream)
22         return response
相關文章
相關標籤/搜索