cron_task.py 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  1. #!/usr/bin/env python2
  2. import signal
  3. import traceback
  4. from datetime import datetime, timedelta
  5. from flask.ext.mail import Message
  6. from flask import url_for
  7. import itsdangerous
  8. from ffdnispdb.crawler import TextValidator
  9. from ffdnispdb.models import ISP
  10. from ffdnispdb import create_app, db, mail, utils
  11. app = create_app({
  12. 'SERVER_NAME': 'db.ffdn.org',
  13. })
  14. MAX_RUNTIME = 15 * 60
  15. class Timeout(Exception):
  16. pass
  17. class ScriptTimeout(Exception):
  18. """
  19. Script exceeded its allowed run time
  20. """
  21. strike = 1
  22. last_isp = -1
  23. script_begin = datetime.now()
  24. def timeout_handler(signum, frame):
  25. global last_isp, strike
  26. if script_begin < datetime.now() - timedelta(seconds=MAX_RUNTIME):
  27. raise ScriptTimeout
  28. if last_isp == isp.id:
  29. strike += 1
  30. if strike > 2:
  31. signal.alarm(6)
  32. raise Timeout
  33. else:
  34. last_isp = isp.id
  35. strike = 1
  36. signal.alarm(6)
  37. signal.signal(signal.SIGALRM, timeout_handler)
  38. signal.alarm(6)
  39. def gen_reactivate_key(isp):
  40. s = itsdangerous.URLSafeSerializer(app.secret_key,
  41. salt='reactivate')
  42. return s.dumps([
  43. isp.id,
  44. str(isp.last_update_attempt)
  45. ])
  46. def send_warning_email(isp, debug_msg):
  47. msg = Message(u"Problem while updating your ISP's data", sender=app.config['EMAIL_SENDER'])
  48. msg.body = """
  49. Hello,
  50. You are receiving this message because your are listed as technical contact for "%s" on the FFDN ISP database.
  51. Our automatic update script could not access or process your ISP's data located at %s.
  52. Automatic updates of your ISP were disabled until you fix the problem.
  53. Here is some debug output to help you locate the issue:
  54. %s
  55. ---
  56. When the issue is resolved, please click on the link below to reactivate automatic updates on your ISP:
  57. %s?key=%s
  58. Thanks,
  59. The FFDN ISP Database team
  60. https://db.ffdn.org
  61. """.strip() % (isp.complete_name, isp.json_url, debug_msg.strip(),
  62. url_for('ispdb.reactivate_isp', projectid=isp.id), gen_reactivate_key(isp))
  63. msg.add_recipient(isp.tech_email)
  64. print u' Sending notification email to %s' % (isp.tech_email)
  65. mail.send(msg)
  66. app.app_context().push()
  67. try:
  68. for isp in ISP.query.filter(ISP.is_disabled == False,
  69. ISP.json_url != None,
  70. ISP.next_update < utils.utcnow(),
  71. ISP.update_error_strike < 3)\
  72. .order_by(ISP.last_update_success):
  73. try:
  74. print u'%s: Attempting to update %s' % (datetime.now(), isp)
  75. print u' last successful update=%s' % (utils.tosystemtz(isp.last_update_success))
  76. print u' last update attempt=%s' % (utils.tosystemtz(isp.last_update_attempt))
  77. print u' next update was scheduled %s ago' % (utils.utcnow() - isp.next_update)
  78. print u' strike=%d' % (isp.update_error_strike)
  79. isp.last_update_attempt = utils.utcnow()
  80. db.session.add(isp)
  81. db.session.commit()
  82. validator = TextValidator()
  83. log = ''
  84. exc, exc_trace = None, None
  85. try:
  86. for l in validator(isp.json_url, isp.cache_info or {}):
  87. log += l
  88. except Exception as e:
  89. exc = e
  90. exc_trace = traceback.format_exc()
  91. if not validator.success: # handle error
  92. isp.update_error_strike += 1
  93. # reset cache info (to force refetch next time)
  94. isp.cache_info = {}
  95. isp.next_update = utils.utcnow() + timedelta(seconds=validator.jdict_max_age)
  96. db.session.add(isp)
  97. db.session.commit()
  98. print u'%s: Error while updating:' % (datetime.now())
  99. if isp.update_error_strike >= 3:
  100. print u' three strikes, you\'re out'
  101. send_warning_email(isp, log)
  102. print log.rstrip().encode('utf-8') + '\n'
  103. if exc:
  104. print u'Unexpected exception in the validator: %r' % exc
  105. print exc_trace
  106. continue
  107. if validator.modified:
  108. isp.json = validator.jdict
  109. isp.cache_info = validator.cache_info
  110. isp.last_update_success = isp.last_update_attempt
  111. isp.update_error_strike = 0
  112. isp.next_update = utils.utcnow() + timedelta(seconds=validator.jdict_max_age)
  113. db.session.add(isp)
  114. db.session.commit()
  115. print u'%s: Update successful !' % (datetime.now())
  116. print u' next update is scheduled for %s\n' % (isp.next_update)
  117. except Timeout:
  118. print u'%s: Timeout while updating:' % (datetime.now())
  119. isp = ISP.query.get(isp.id)
  120. isp.update_error_strike += 1
  121. db.session.add(isp)
  122. db.session.commit()
  123. if isp.update_error_strike >= 3:
  124. send_warning_email(isp, 'Your ISP took more then 18 seconds to process. '
  125. 'Having problems with your webserver ?')
  126. print u' three strikes, you\'re out'
  127. print traceback.format_exc()
  128. except:
  129. print u'Unknown error, see call stack below.'
  130. traceback.print_exc() # To help debugging the cause of the failure
  131. db.session.rollback()
  132. except ScriptTimeout:
  133. pass
  134. except Timeout:
  135. pass