senkai 2017-11-03
要在django项目中定期执行任务,比如每天一定的时间点抓取数据,刷新数据库等,可以参考stackoverflow的方法,先编写一个manage.py命令,然后使用crontab来定时执行这个命令。
定制manage.py命令
app可以使用manage.py注册自己的命令,比如要在polls这个app中定制一个closepoll命令,要先向polls文件夹中添加一个management/commands的目录:
polls/
  __init__.py 
  models.py 
  management/
    __init__.py 
    commands/
      __init__.py 
      _private.py 
      closepoll.py 
  tests.py 
  views.py这样,commands目录下每一个不以"_"开头的py文件都被注册为一个manage.py命令。
python 2版本中注意management和commands目录下都要包含一个__init__.py文件。
closepoll.py文件中必须定义一个继承自BaseCommand的类Command。
from django.core.management.base import BaseCommand, CommandError 
from polls.models import Poll 
 
class Command(BaseCommand): 
  help = 'Closes the specified poll for voting'
 
  def add_arguments(self, parser): 
    parser.add_argument('poll_id', nargs='+', type=int) 
 
  def handle(self, *args, **options): 
    for poll_id in options['poll_id']: 
      try: 
        poll = Poll.objects.get(pk=poll_id) 
      except Poll.DoesNotExist: 
        raise CommandError('Poll "%s" does not exist' % poll_id) 
 
      poll.opened = False
      poll.save() 
 
      self.stdout.write(self.style.SUCCESS('Successfully closed poll "%s"' % poll_id))在django1.8之前,manage.py命令的命令行解析是基于optparse库的,其中位置参数会被传给*args,而可选参数传给**options。1.8之后,命令行解析基于argparse库,参数都是传给**options的。
closepoll.py文件中添加了名为poll_id的位置参数,数目为一个或者更多,类型为整数:
def add_arguments(self, parser): 
parser.add_argument('poll_id', nargs='+', type=int)closepoll命令就可以这样使用了:
python manage.py closepoll <poll_id>
self.stdout和self.stderr可以在控制台中显示你想要显示的信息。
可以添加可选参数:
class Command(BaseCommand): 
  def add_arguments(self, parser): 
    # Positional arguments 
    parser.add_argument('poll_id', nargs='+', type=int) 
 
    # Named (optional) arguments 
    parser.add_argument('--delete', 
      action='store_true', 
      dest='delete', 
      default=False, 
      help='Delete poll instead of closing it') 
 
  def handle(self, *args, **options): 
    # ... 
    if options['delete']: 
      poll.delete()添加一个名为--delete值为True的可选参数。
执行python manage.py closepoll 1 2 3 --delete时,将id为1,2,3的poll删除。
执行python manage.py closepoll 1 2 3 时,--delete取值为默认值False。
使用crontab定时执行manage.py命令
使用crontab -e命令编辑cron,窗口会有提示:
# m h dom mon dow command
在某月(mon)的某天(dom)或者星期几(dow)的几点(h,24小时制)几分(m)执行某个命令(command),*表示任意时间。
* */2 * * * python manage.py closepoll <poll_id>
这样表示每两个小时执行一次closepoll命令,为了防止manage.py文件找不到,可以写上manage.py文件的真实地址。
保存后重启cron即可:
sudo service cron restart