阅读django源码

从manage.py开始。
manage.py 比较简单就几句话。

  1. #!/usr/bin/env python
  2. #from django.core.management import execute_manager
  3. import os
  4. import sys
  5. if __name__ == "__main__":
  6. #加载配置文件
  7. os.environ.setdefault( "DJANGO_SETTINGS_MODULE", "settings")
  8. from django.core.management import execute_from_command_line
  9. #执行配置
  10. execute_from_command_line(sys.argv)

首先看os.environ.setdefault(),environ为类_Environ的一个实例,它继承自IterableUserDict,而IterableUserDict继承自UserDict,包括setdefault()这个方法也是UserDict的一个方法,看一下setdefault实现的功能:

  1. def setdefault(self, key, failobj=None):
  2. if key not in self:
  3. self[ key] = failobj
  4. return self[ key]

其中的key是 DJANGO_SETTINGS_MODULE,这里的self是一个字典,判断key是不是在里面,不在里面就做一个键值绑定。然后返回。

下面进入execute_from_command_line(sys.argv),sys.argv位一个参数执行的列表,像我们执行python manage.py runserver时,manage.py位sys.argv[0],runserver为sys.argv[1],以此类推,如果有更多参数加的话。

进入django的core模块下,找到management模块。在__init__.py下可以找到该函数

  1. def execute_from_command_line(argv=None):
  2. """
  3. A simple method that runs a ManagementUtility.
  4. """
  5. utility = ManagementUtility(argv)
  6. utility.execute()

我们输入的参数便传到了这个函数里,其中ManagementUtility位一个类穿进去的参数列表会进入一个简单的初始化

  1. def __init__( self, argv=None):
  2. self.argv = argv or sys.argv[:]
  3. self.prog_name = os.path.basename( self.argv[ 0])

将参数列表赋给argv,然后返回sys.argv[0]的文件名字给prog_name。
实例化完了之后,再看utility.execute()即类ManagementUtility的execute()方法。

  1. parser = LaxOptionParser(usage= "%prog subcommand [options] [args]",
  2. version=get_version(),
  3. option_list=BaseCommand.option_list)
  4. self.autocomplete()
  5. try:
  6. options, args = parser.parse_args( self.argv)
  7. handle_default_options(options)
  8. except:
  9. pass # Ignore any option errors at this point.
  10. try:
  11. subcommand = self.argv[ 1]
  12. except IndexError:
  13. subcommand = 'help' # Display help if no arguments were given.
  14. if subcommand == 'help':
  15. if len(args) <= 2:
  16. parser.print_lax_help()
  17. sys.stdout.write( self.main_help_text() + '\n')
  18. elif args[ 2] == '--commands':
  19. sys.stdout.write( self.main_help_text(commands_only= True) + '\n')
  20. else:
  21. self.fetch_command(args[ 2]).print_help( self.prog_name, args[ 2])
  22. elif subcommand == 'version':
  23. sys.stdout.write(parser.get_version() + '\n')
  24. # Special-cases: We want 'django-admin.py --version' and
  25. # 'django-admin.py --help' to work, for backwards compatibility.
  26. elif self.argv[ 1:] == [ '--version']:
  27. # LaxOptionParser already takes care of printing the version.
  28. pass
  29. elif self.argv[ 1:] in ([ '--help'], [ '-h']):
  30. parser.print_lax_help()
  31. sys.stdout.write( self.main_help_text() + '\n')
  32. else:
  33. self.fetch_command(subcommand).run_from_argv( self.argv)

第一个语句为词法分析,可是找遍了LaxOptionParser这个类也没有找到入口,里面并没有init,只有几个函数,可是python也没有正式的构造函数重载的什么的,后来发现此类继承自OptionParser这个类,它来自于optparse模块,在OptionParser模块里有一个init()函数,会接受若干个参数,进行初始化,这里只是传递了3个参数,usage传递了一个字符串,version为版本,BaseCommand.option_list为一个关于命令行的元组,其中调用了make_option这个类init进行一些初始化,主要初始化_short_opts和_long_opts这两个参数,并且进行了一些检查。_short_opts为短命令,_long_opts为长命令,如-h,--help分别为短命令和长命令。
parser为LaxOptionParser的一个实例。 parser.parse_args(self.argv)这个函数会根据上面初始化的内容进行解析命令行参数,之后返回两个值:options,它是一个对象(,保存有命令行参数值。只要知道命令行参数名,如 file,就可以访问其对应的值: options.file 。args,它是一个由 positional arguments 组成的列表。

  1. def handle_default_options(options):
  2. """
  3. Include any default options that all commands should accept here
  4. so that ManagementUtility can handle them before searching for
  5. user commands.
  6. """
  7. if options.settings:
  8. os.environ[ 'DJANGO_SETTINGS_MODULE'] = options.settings
  9. if options.pythonpath:
  10. sys.path.insert( 0, options.pythonpath)

handle_default_options将解析出来的options对象当做参数,判断settings和pythonpath是否存在,然后
设置环境变量和python模块的搜索路径。

接下来是获得命令行的命令参数self.argv[1],并判断这个命令,包括错误处理,是否时help,是否是version,根据不同的情况展示不同的信息。

最重要的是最后一句,即前面的情况都不是,就进入self.fetch_command(subcommand).run_from_argv(self.argv)

  1. def fetch_command(self, subcommand):
  2. """
  3. Tries to fetch the given subcommand, printing a message with the
  4. appropriate command called from the command line (usually
  5. "django-admin.py" or "manage.py") if it can't be found.
  6. """
  7. # Get commands outside of try block to prevent swallowing exceptions
  8. commands = get_commands()
  9. try:
  10. app_name = commands[subcommand]
  11. except KeyError:
  12. sys.stderr.write( "Unknown command: %r\nType '%s help' for usage.\n" %
  13. (subcommand, self.prog_name))
  14. sys.exit( 1)
  15. if isinstance(app_name, BaseCommand):
  16. # If the command is already loaded, use it directly.
  17. klass = app_name
  18. else:
  19. klass = load_command_class(app_name, subcommand)
  20. return klass

获得django/core/management/commands目录下与INSTALLED_APPS/management/commands目录下的子命令对应的模块前缀

  1. def load_command_class(app_name, name):
  2. """
  3. Given a command name and an application name, returns the Command
  4. class instance. All errors raised by the import process
  5. (ImportError, AttributeError) are allowed to propagate.
  6. """
  7. module = import_module( '%s.management.commands.%s' % (app_name, name))
  8. return module.Command()

返回Command类的实例。进入management/commands下进入每个文件会发现,每个都有Command类对应相应的命令。

self.fetch_command(subcommand).run_from_argv(self.argv)找到我们输入的命令参数,并且使用run_from_argv执行。

进入runserver这个命令下看一下,Command这个类继承了BaseCommand这个类,BaseCommand在Base.py中,里面有run_from_arv这个方法,在其他的命令里
有的重写了这个方法,不过runserver没有。

  1. def run_from_argv(self, argv):
  2. """
  3. Set up any environment changes requested (e.g., Python path
  4. and Django settings), then run this command. If the
  5. command raises a ``CommandError``, intercept it and print it sensibly
  6. to stderr. If the ``--traceback`` option is present or the raised
  7. ``Exception`` is not ``CommandError``, raise it.
  8. """
  9. #设置环境变量,然后运行这个命令
  10. #如python manage.py runserver, manage.py就是argv[0],runserver是argv[1]
  11. #create_parser直接调用OptionParser(prog=prog_name,usage=self.usage(subcommand),version=self.get_version(),option_list=self.option_list)
  12. #将manage.py 和runserver加入参数列表
  13. #接下来用parser.parse_args进行解析,argv[2]之后为默认参数,ip地址还有端口号,我们也可以显示的传进去,默认是127.0.0.1,端口号8000.
  14. #下面又到了handle_default_options函数。
  15. #接下来是执行execute函数,还有异常捕获。
  16. parser = self.create_parser(argv[ 0], argv[ 1])
  17. options, args = parser.parse_args(argv[ 2:])
  18. handle_default_options(options)
  19. try:
  20. self.execute(*args, **options.__dict__)
  21. except Exception as e:
  22. if options.traceback or not isinstance(e, CommandError):
  23. raise
  24. # self.stderr is not guaranteed to be set here
  25. stderr = getattr(self, 'stderr', OutputWrapper(sys.stderr, self.style.ERROR))
  26. stderr.write( '%s: %s' % (e.__class__.__name__, e))
  27. sys.exit( 1)

下面进入execute函数,前边是一些设置和错误检查,最主要的是 output = self.handle(*args, **options),可以发现handle在BaseCommand里是空的,每个命令对其进行了重写。
runserver也是。看看runserver的handle:

  1. def handle( self, addrport= '', *args, **options):
  2. #导入设置模块
  3. from django.conf import settings
  4. #DEBUG和ALLOWED_HOSTS的设置
  5. if not settings.DEBUG and not settings.ALLOWED_HOSTS:
  6. raise CommandError( 'You must set settings.ALLOWED_HOSTS if DEBUG is False.')
  7. #ipv6的设置
  8. self.use_ipv6 = options.get( 'use_ipv6')
  9. if self.use_ipv6 and not socket.has_ipv6:
  10. raise CommandError( 'Your Python does not support IPv6.')
  11. if args:
  12. raise CommandError( 'Usage is runserver %s' % self.args)
  13. self._raw_ipv6 = False
  14. if not addrport:
  15. self.addr = ''
  16. self.port = DEFAULT_PORT
  17. else:
  18. #如果设置了ip地址和端口号,用正则匹配出来
  19. m = re.match(naiveip_re, addrport)
  20. if m is None:
  21. raise CommandError( '"%s" is not a valid port number '
  22. 'or address:port pair.' % addrport)
  23. self.addr, _ipv4, _ipv6, _fqdn, self.port = m.groups()
  24. if not self.port.isdigit():
  25. raise CommandError( "%r is not a valid port number." % self.port)
  26. if self.addr:
  27. if _ipv6:
  28. self.addr = self.addr[ 1: -1]
  29. self.use_ipv6 = True
  30. self._raw_ipv6 = True
  31. elif self.use_ipv6 and not _fqdn:
  32. raise CommandError( '"%s" is not a valid IPv6 address.' % self.addr)
  33. if not self.addr:
  34. #如果没有设置ip地址使用127.0.0.1代替
  35. self.addr = '::1' if self.use_ipv6 else '127.0.0.1'
  36. self._raw_ipv6 = bool( self.use_ipv6)
  37. #运行命令
  38. self.run(*args, **options)

runserver里的run方法主要时调用了inner_run(*args, **options)这个方法。

  1. def inner_run(self, *args, **options):
  2. #省略了部分代码
  3. #很熟悉吧,这就是我们开始运行时,终端上输出的信息啊。
  4. self.stdout.write((
  5. "%(started_at)s\n"
  6. "Django version %(version)s, using settings %(settings)r\n"
  7. "Starting development server at http://%(addr)s:%(port)s/\n"
  8. "Quit the server with %(quit_command)s.\n"
  9. ) % {
  10. "started_at": now,
  11. "version": self.get_version(),
  12. "settings": settings.SETTINGS_MODULE,
  13. "addr": '[%s]' % self.addr if self._raw_ipv6 else self.addr,
  14. "port": self.port,
  15. "quit_command": quit_command,
  16. })
  17. #加载编码设置
  18. translation.activate(settings.LANGUAGE_CODE)
  19. try:
  20. handler = self.get_handler(*args, **options)
  21. run(self.addr, int(self.port), handler,
  22. ipv6=self.use_ipv6, threading=threading)
  23. #省略部分代码
  24. def get_handler(self, *args, **options):
  25. """
  26. Returns the default WSGI handler for the runner.
  27. """
  28. return get_internal_wsgi_application()

get_internal__wsgi_application()和run(self.addr, int(self.port), handler,ipv6=self.use_ipv6, threading=threading)都在django.core.servers.basehttp中:

  1. def get_internal_wsgi_application():
  2. from django.conf import settings
  3. #在settings模块中并没有找到WSGI_APPLICATION这个环境变量,因此app_path位空
  4. app_path = getattr(settings, 'WSGI_APPLICATION')
  5. if app_path is None:
  6. return get_wsgi_application()

在django/core下wsig.py中的文件如下:就几句话

  1. import django
  2. from django.core.handlers.wsgi import WSGIHandler
  3. def get_wsgi_application():
  4. #setup是加载log和settings.INSTALLED_APPS
  5. django.setup()
  6. #返回WSGIHhadler类的一个实例
  7. return WSGIHandler()
  8. class WSGIHandler(base.BaseHandler):
  9. #初始化一个线程锁
  10. initLock = Lock()
  11. #WSGIRequest为http.HttpRequest的一个子类,
  12. #WSGIRequest实现了wsgi规范
  13. request_class = WSGIRequest

下面进入run方法:run(self.addr, int(self.port), handler,ipv6=self.use_ipv6, threading=threading),标准的wsgi实现:

  1. def run(addr, port, wsgi_handler, ipv6=False, threading=False):
  2. server_address = (addr, port)
  3. if threading:
  4. httpd_cls = type(str( 'WSGIServer'), (socketserver.ThreadingMixIn, WSGIServer), {})
  5. else:
  6. httpd_cls = WSGIServer
  7. httpd = httpd_cls(server_address, WSGIRequestHandler, ipv6=ipv6)
  8. httpd.set_app(wsgi_handler)
  9. httpd.serve_forever()

httpd_cls(server_address, WSGIRequestHandler, ipv6=ipv6)为实例化一个WSGIServer类,最终的实例化方法在父类SocketServer中的TCPServer和BaseServer中。包括初始化线程,初始化网络句柄,像下面的__is_shut_down和__shutdown_request都是在其中初始化的

  1. #处理一个http请求直到关闭
  2. def serve_forever(self, poll_interval=0.5):
  3. #__is_shut_down为一个初始化的threading.Event()的句柄,用于线程间通信
  4. #.clear()将标识设置为false
  5. self.__is_shut_down.clear()
  6. try:
  7. while not self.__shutdown_request:
  8. #下面的函数就是一个封装好了的select函数,后面的四个为传递给select函数的参数,分别表示输入对象列表,输出对象列表,错误对象列表以及超时时间。
  9. #返回值为三个列表,代表可写,可读,错误的事件的对象列表。
  10. r, w, e = _eintr_retry(select.select, [self], [], [],
  11. poll_interval)
  12. #判断self事件是否在可读对象列表中,在就进入进行处理
  13. if self in r:
  14. #处理连接请求
  15. self._handle_request_noblock()
  16. finally:
  17. self.__shutdown_request = False
  18. #将标识设置为true
  19. self.__is_shut_down.set()
  20. #对select函数的封装
  21. def _eintr_retry(func, *args):
  22. """restart a system call interrupted by EINTR"""
  23. while True:
  24. try:
  25. return func(*args)
  26. except (OSError, select.error) as e:
  27. if e.args[ 0] != errno.EINTR:
  28. raise
  29. def _handle_request_noblock(self):
  30. try:
  31. #返回请求句柄,客户端地址,get_request()中调用了self.socket.accept()来实现客户端的连接
  32. request, client_address = self.get_request()
  33. except socket.error:
  34. return
  35. if self.verify_request(request, client_address):
  36. try:
  37. #真正的处理连接请求的地方,调用了self.finish_request(request, client_address)
  38. self.process_request(request, client_address)
  39. except:
  40. self.handle_error(request, client_address)
  41. self.shutdown_request(request)
  42. def finish_request(self, request, client_address):
  43. """Finish one request by instantiating RequestHandlerClass."""
  44. 此处的RequestHandlerClass为初始化的时候传进来的WSGIRequestHandler,实现了回调。
  45. self.RequestHandlerClass(request, client_address, self)

在WSGIRequestHandler中实现下面的初始化函数:

  1. def __init__(self, *args, **kwargs):
  2. from django.conf import settings
  3. self.admin_static_prefix = urljoin(settings.STATIC_URL, 'admin/')
  4. # We set self.path to avoid crashes in log_message() on unsupported
  5. # requests (like "OPTIONS").
  6. self.path = ''
  7. self.style = color_style()
  8. #调用父类的WSGIRequestHandler进行初始化,它的父类是simple_server.py里的WSGIRequestHandler,它里面没有__init__,继续找它的父类
  9. #如此重复,在最终的基类中可以找到__init__方法,位于SocketServer.py中的 BaseRequestHandler类。
  10. super(WSGIRequestHandler, self).__init__(*args, **kwargs)
  11. def __init__(self, request, client_address, server):
  12. self.request = request
  13. self.client_address = client_address
  14. self.server = server
  15. #setup函数
  16. self.setup()
  17. try:
  18. self.handle()
  19. finally:
  20. self.finish()
  21. def setup(self):
  22. pass
  23. def handle(self):
  24. pass
  25. def finish(self):
  26. pass

可以看到里面的方法并没有实现,只是给出了定义,实现都在子类中。

再回到WSGIRequestHandler类中,在simple_server.py中实现了handle函数,实现对请求的处理

  1. def handle(self):
  2. """Handle a single HTTP request"""
  3. self.raw_requestline = self.rfile.readline()
  4. if not self.parse_request(): # An error code has been sent, just exit
  5. return
  6. #传入的参数,读,写,错误,环境变量。在其父类SimpleHandler中进行了初始化,并且打开了多线程和多进程选项
  7. handler = ServerHandler(
  8. self.rfile, self.wfile, self.get_stderr(), self.get_environ()
  9. )
  10. handler.request_handler = self # backpointer for logging
  11. 在SimpleHandler的父类BaseHandler中含实现了run方法。此处get_app()是上面的run方法中我们传进去的wsgi_handler,httpd.set_app(wsgi_handler)
  12. handler.run(self.server.get_app())
  13. def run(self, application):
  14. try:
  15. #设置环境变量
  16. self.setup_environ()
  17. #执行WSGIHandler(self.environ,self.start_response)
  18. #因为类中实现了__call__方法,调用类的实例就相当于调用了__call__方法
  19. self.result = application(self.environ, self.start_response)
  20. self.finish_response()
  21. except:
  22. try:
  23. self.handle_error()
  24. except:
  25. # If we get an error handling an error, just give up already!
  26. self.close()
  27. raise # ...and let the actual server figure it out.
  28. def __call__(self, environ, start_response):
  29. # Set up middleware if needed. We couldn't do this earlier, because
  30. # settings weren't available.
  31. if self._request_middleware is None:
  32. with self.initLock:
  33. try:
  34. # Check that middleware is still uninitialised.
  35. if self._request_middleware is None:
  36. #尝试加载中间件
  37. self.load_middleware()
  38. except:
  39. # Unload whatever middleware we got
  40. self._request_middleware = None
  41. raise
  42. #设置脚本路径
  43. set_script_prefix(base.get_script_name(environ))
  44. signals.request_started.send(sender=self.__class__)
  45. try:
  46. request = self.request_class(environ)
  47. except UnicodeDecodeError:
  48. logger.warning( 'Bad Request (UnicodeDecodeError)',
  49. exc_info=sys.exc_info(),
  50. extra={
  51. 'status_code': 400,
  52. }
  53. )
  54. response = http.HttpResponseBadRequest()
  55. else:
  56. response = self.get_response(request)
  57. response._handler_class = self.__class__
  58. status = '%s %s' % (response.status_code, response.reason_phrase)
  59. response_headers = [(str(k), str(v)) for k, v in response.items()]
  60. for c in response.cookies.values():
  61. response_headers.append((str( 'Set-Cookie'), str(c.output(header= ''))))
  62. start_response(force_str(status), response_headers)
  63. return response

以上为我阅读django源码的一些心得,大体了解了django的一点原理

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值