使用 CherryPy 作为Django的Development Server

Django自带的Development Server作为调试开发时的临时服务器是一个非常方便的工具,因为避免了在还没有代码之前就需要配置一大堆环境的麻烦。

但是此服务器也有一个不足的地方就是它是单线程模式的,也就是同时只能处理一个请求,当需要调试ajax应用时,这就显得非常麻烦了,因为这经常需要同时处理多个请求。

经过搜索,查找,简单的解决方法有以下两种:

1、修改Django自带的Development Server为多线程模式,这其实只需要改几行代码就可以了。(详见http://code.djangoproject.com/ticket/3357 )

 

打开文件 django/core/servers/basehttp.py

第10行:

from BaseHTTPServer import BaseHTTPRequestHandler, HTTPServer

修改为

from BaseHTTPServer import BaseHTTPRequestHandler

 

在所有的 import 语句之后加上以下代码:

Python代码  
  1. import BaseHTTPServer, SocketServer  
  2. class HTTPServer(SocketServer.ThreadingMixIn, BaseHTTPServer.HTTPServer):    
  3.     def __init__(self, server_address, RequestHandlerClass=None):    
  4.         BaseHTTPServer.HTTPServer.__init__(self, server_address, RequestHandlerClass)  

 

是的,这样就可以了。 但是他还有一个小问题就是它不是线程安全的。可能在请求很多时会出现数据不同步,当然,这一般不是问题,因为我们通常只在自己机器上调试时才用Development Server。

 

2、使用CherryPy 作为 Development Server

在安装好CherryPy 之后,将以下代码保存为文件runservercp.py

Python代码  
  1. from django.core.management.base import BaseCommand, CommandError  
  2. from optparse import make_option  
  3. import os, sys  
  4. import cherrypy  
  5. from cherrypy import wsgiserver  
  6. class Command(BaseCommand):  
  7.     option_list = BaseCommand.option_list + (  
  8.         make_option('--noreload', action='store_false', dest='use_reloader', default=True,  
  9.             help='Tells Django to NOT use the auto-reloader.'),  
  10.         make_option('--adminmedia', dest='admin_media_path', default='',  
  11.             help='Specifies the directory from which to serve admin media.'),  
  12.     )  
  13.     help = "Starts a lightweight Web server for development."  
  14.     args = '[optional port number, or ipaddr:port]'  
  15.   
  16.     # Validation is called explicitly each time the server is reloaded.  
  17.     requires_model_validation = False  
  18.   
  19.     def handle(self, addrport='', *args, **options):  
  20.         import django  
  21.         from django.core.servers.basehttp import AdminMediaHandler  
  22.         from django.core.handlers.wsgi import WSGIHandler  
  23.         if args:  
  24.             raise CommandError('Usage is runservercp %s' % self.args)  
  25.         if not addrport:  
  26.             addr = ''  
  27.             port = '8000'  
  28.         else:  
  29.             try:  
  30.                 addr, port = addrport.split(':')  
  31.             except ValueError:  
  32.                 addr, port = '', addrport  
  33.         if not addr:  
  34.             addr = '127.0.0.1'  
  35.   
  36.         if not port.isdigit():  
  37.             raise CommandError("%r is not a valid port number." % port)  
  38.   
  39.         use_reloader = options.get('use_reloader'True)  
  40.         admin_media_path = options.get('admin_media_path''')  
  41.         shutdown_message = options.get('shutdown_message''')  
  42.         quit_command = (sys.platform == 'win32'and 'CTRL-BREAK' or 'CONTROL-C'  
  43.           
  44.         def inner_run():  
  45.             from django.conf import settings  
  46.             from django.utils import translation  
  47.             print "/nValidating models..."  
  48.             self.validate(display_num_errors=True)  
  49.             print "/nDjango version %s, using settings %r" % (django.get_version(), settings.SETTINGS_MODULE)  
  50.             print "Development server is running at http://%s:%s/" % (addr, port)  
  51.             print "Quit the server with %s." % quit_command  
  52.               
  53.             # django.core.management.base forces the locale to en-us. We should  
  54.             # set it up correctly for the first request (particularly important  
  55.             # in the "--noreload" case).  
  56.             translation.activate(settings.LANGUAGE_CODE)  
  57.               
  58.             try:  
  59.                 path = admin_media_path or django.__path__[0] + '/contrib/admin/media'  
  60.                 handler = AdminMediaHandler(WSGIHandler(), path)  
  61.                 cherrypy.config.update({  
  62.                         'server.socket_host': addr,  
  63.                         'server.socket_port': int(port),  
  64.                         'engine.autoreload_on':  use_reloader  
  65.                     })  
  66.       
  67.                 cherrypy.tree.graft(handler, '/')  
  68.       
  69.                 cherrypy.engine.start()  
  70.                 cherrypy.engine.block()  
  71.             finally:  
  72.                 if shutdown_message:  
  73.                     print shutdown_message  
  74.                 cherrypy.engine.exit()  
  75.         inner_run()  

 

将该文件放到djang源代码目录 django/core/management/commands 下,这样运行命令

python manage.py runservercp

时就使用了CherryPy 作为服务器了,其中命令名是根据文件的名字而定的,因为该文件保存为runservercp.py,所以命令名为 runservercp。

另外放在django的源代码目录的好处是所有的django project都可以使用该命令使用到CherryPy 。

 

另外一种放置方式是新建一个app,在该app下新建package management,再在该package下建另一package commands,将该文件放在commands 目录下(可参考django/contrib/auth 应用),同时在settings.py 中的 INSTALLED_APPS 加上该app,那么该project 就可以使用该命令了。其他project不受影响。

你可能感兴趣的:(Django)