- 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
-
-
- 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):
- now = timezone.now()
- month_before = now - timedelta(days=30)
- month_after = now + timedelta(days=30)
-
- report_due = Q(next_report_date__lt=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)
- q = q.filter(user__date_joined__lt=month_before)
- profile = q.first()
-
- if profile is None:
- return False
-
- # A sort of optimistic lock. 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
- )
-
- num_updated = qq.update(next_report_date=month_after)
- 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)
|