from datetime import timedelta import time from django.core.management.base import BaseCommand from django.db.models import Q from django.utils import timezone from hc.accounts.models import NO_NAG, Profile from hc.api.models import Check from hc.lib.date import choose_next_report_date def num_pinged_checks(profile): q = Check.objects.filter(user_id=profile.user.id) q = q.filter(last_ping__isnull=False) return q.count() class Command(BaseCommand): help = "Send due monthly reports and nags" tmpl = "Sent monthly report to %s" def pause(self): time.sleep(1) def add_arguments(self, parser): parser.add_argument( "--loop", action="store_true", dest="loop", default=False, help="Keep running indefinitely in a 300 second wait loop", ) def handle_one_monthly_report(self): report_due = Q(next_report_date__lt=timezone.now()) report_not_scheduled = Q(next_report_date__isnull=True) q = Profile.objects.filter(report_due | report_not_scheduled) q = q.filter(reports_allowed=True) profile = q.first() if profile is None: # No matching profiles found – nothing to do right now. return False # A sort of optimistic lock. Will try to update next_report_date, # and if does get modified, we're in drivers seat: qq = Profile.objects.filter( id=profile.id, next_report_date=profile.next_report_date ) # Next report date is currently not scheduled: schedule it and move on. if profile.next_report_date is None: qq.update(next_report_date=choose_next_report_date()) return True num_updated = qq.update(next_report_date=choose_next_report_date()) if num_updated != 1: # next_report_date was already updated elsewhere, skipping return True if profile.send_report(): self.stdout.write(self.tmpl % profile.user.email) # Pause before next report to avoid hitting sending quota self.pause() return True def handle_one_nag(self): now = timezone.now() q = Profile.objects.filter(next_nag_date__lt=now) q = q.exclude(nag_period=NO_NAG) profile = q.first() if profile is None: return False qq = Profile.objects.filter(id=profile.id, next_nag_date=profile.next_nag_date) num_updated = qq.update(next_nag_date=now + profile.nag_period) if num_updated != 1: # next_rag_date was already updated elsewhere, skipping return True if profile.send_report(nag=True): self.stdout.write("Sent nag to %s" % profile.user.email) # Pause before next report to avoid hitting sending quota self.pause() else: profile.next_nag_date = None profile.save() return True def handle(self, *args, **options): self.stdout.write("sendreports is now running") while True: # Monthly reports while self.handle_one_monthly_report(): pass # Daily and hourly nags while self.handle_one_nag(): pass if not options["loop"]: break formatted = timezone.now().isoformat() self.stdout.write("-- MARK %s --" % formatted) # Sleep for 1 minute before looking for more work time.sleep(60)