Skip to content

tasks

main.tasks ¤

Task definitions for project notifications using Huey.

Classes¤

Functions¤

daily_project_status_check() ¤

Daily task to check project statuses and notify leads.

Source code in main/tasks.py
77
78
79
80
81
82
83
84
@db_periodic_task(crontab(hour=10, minute=0))
def daily_project_status_check() -> None:
    """Daily task to check project statuses and notify leads."""
    from .models import Project

    projects = Project.objects.filter(status="Active")
    for project in projects:
        project.check_and_notify_status()

email_monthly_charges_report() ¤

Email the HoRSE the charges report for the last month.

Source code in main/tasks.py
273
274
275
276
277
278
279
@db_periodic_task(crontab(day=10, hour=10))
def email_monthly_charges_report() -> None:
    """Email the HoRSE the charges report for the last month."""
    last_month_start, last_month_name, _, _ = get_current_and_last_month()
    email_monthly_charges_report_logic(
        last_month_start.month, last_month_start.year, last_month_name
    )

email_monthly_charges_report_logic(month, year, month_name) ¤

Logic to email the HoRSE the charges report for the last month.

Source code in main/tasks.py
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
def email_monthly_charges_report_logic(month: int, year: int, month_name: str) -> None:
    """Logic to email the HoRSE the charges report for the last month."""
    subject = f"Charges report for {month_name}/{year}"
    head_email = get_head_email()
    message = _template_charges_report.format(month=month_name, year=year)
    csv_attachment = create_charges_report_for_attachment(month, year)

    email_attachment(
        subject,
        head_email,
        message,
        f"charges_report_{month}-{year}.csv",
        csv_attachment,
        "text/csv",
    )

notify_funding_status() ¤

Daily task to notify about funding status.

Source code in main/tasks.py
239
240
241
242
@db_periodic_task(crontab(hour=11, minute=0))
def notify_funding_status() -> None:
    """Daily task to notify about funding status."""
    notify_funding_status_logic()

notify_funding_status_logic(date=None) ¤

Logic for notifying the lead about funding status.

Source code in main/tasks.py
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
def notify_funding_status_logic(
    date: datetime.date | None = None,
) -> None:
    """Logic for notifying the lead about funding status."""
    funds_ran_out_not_expired, funding_expired_budget_left = get_budget_status(
        date=date
    )

    if funds_ran_out_not_expired.exists():
        for funding in funds_ran_out_not_expired:
            subject = f"[Funding Update] {funding.project.name}"
            head_email = get_head_email()
            lead = funding.project.lead
            lead_name = lead.get_full_name() if lead is not None else "Project Leader"
            lead_email = lead.email if lead is not None else ""
            activity = funding.activity if funding.activity else "Funding Activity"
            message = _template_funds_ran_out_but_not_expired.format(
                lead=lead_name,
                project_name=funding.project.name,
                activity=activity,
            )
            email_user_and_cc_head(
                subject=subject,
                message=message,
                email=lead_email,
                head_email=head_email,
            )

    if funding_expired_budget_left.exists():
        for funding in funding_expired_budget_left:
            subject = f"[Funding Expired] {funding.project.name}"
            head_email = get_head_email()
            lead = funding.project.lead
            lead_name = lead.get_full_name() if lead is not None else "Project Leader"
            lead_email = lead.email if lead is not None else ""
            message = _template_funding_expired_but_has_budget.format(
                lead=lead_name,
                project_name=funding.project.name,
                budget=funding.budget,
            )
            email_user_and_cc_head(
                subject=subject,
                message=message,
                email=lead_email,
                head_email=head_email,
            )

notify_left_threshold(email, lead, project_name, threshold_type, threshold, value) ¤

Huey task wrapper that calls the core notify logic.

Source code in main/tasks.py
61
62
63
64
65
66
67
68
69
70
71
72
73
@task()
def notify_left_threshold(
    email: str,
    lead: str,
    project_name: str,
    threshold_type: str,
    threshold: int,
    value: int,
) -> None:
    """Huey task wrapper that calls the core notify logic."""
    notify_left_threshold_logic(
        email, lead, project_name, threshold_type, threshold, value
    )

notify_left_threshold_logic(email, lead, project_name, threshold_type, threshold, value) ¤

Logic for notifying the lead about project status.

Source code in main/tasks.py
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
def notify_left_threshold_logic(
    email: str,
    lead: str,
    project_name: str,
    threshold_type: str,
    threshold: int,
    value: int,
) -> None:
    """Logic for notifying the lead about project status."""
    if threshold_type not in ("effort", "weeks"):
        raise ValueError("Invalid threshold type provided.")
    unit = "days" if threshold_type == "effort" else "weeks"
    subject = f"[Project Status Update] {project_name}"
    message = _template.format(
        project_leader=lead,
        project_name=project_name,
        threshold=threshold,
        threshold_type=threshold_type.rsplit("_")[0],
        value=value,
        unit=unit,
    )

    email_user(subject, email, message)

notify_monthly_days_used_exceeding_days_left() ¤

Monthly task to notify project leads and HoRSE if days used exceed days left.

Source code in main/tasks.py
409
410
411
412
@db_periodic_task(crontab(day=7, hour=9, minute=30))
def notify_monthly_days_used_exceeding_days_left() -> None:
    """Monthly task to notify project leads and HoRSE if days used exceed days left."""
    notify_monthly_days_used_exceeding_days_left_logic()

notify_monthly_days_used_exceeding_days_left_logic(date=None) ¤

Logic to notify project lead and HoRSE if total days used exceed days left.

This function checks each project to see if the days used for the project exceed the days left. If they do, it sends an email notification to the project lead and HoRSE.

Source code in main/tasks.py
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
def notify_monthly_days_used_exceeding_days_left_logic(
    date: datetime.datetime | None = None,
) -> None:
    """Logic to notify project lead and HoRSE if total days used exceed days left.

    This function checks each project to see if the days used for the
    project exceed the days left. If they do,
    it sends an email notification to the project lead and HoRSE.
    """
    if date is None:
        date = datetime.datetime.today()

    projects = get_projects_with_days_used_exceeding_days_left(date=date)

    for project, days_used, days_left in projects:
        lead = project.lead
        lead_name = lead.get_full_name() if lead else "Project Leader"
        lead_email = lead.email if lead else ""

        subject = f"[Monthly Days Used Exceed Days Left] {project.name}"
        message = _template_days_used_exceeded_days_left.format(
            lead=lead_name,
            project_name=project.name,
            days_used=days_used,
            days_left=days_left,
        )

        head_email = get_head_email()

        email_user_and_cc_head(
            subject=subject,
            message=message,
            email=lead_email,
            head_email=head_email,
        )

notify_monthly_time_logged_logic(last_month_start, last_month_name, current_month_start, current_month_name) ¤

Logic to notify users about their monthly time logged.

Source code in main/tasks.py
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
def notify_monthly_time_logged_logic(
    last_month_start: datetime.date,
    last_month_name: str,
    current_month_start: datetime.date,
    current_month_name: str,
) -> None:
    """Logic to notify users about their monthly time logged."""
    from .models import TimeEntry, User

    avg_work_days_per_month = 220 / 12  # Approximately 18.33 days per month

    time_entries = TimeEntry.objects.filter(
        start_time__gte=last_month_start, end_time__lt=current_month_start
    )

    if not time_entries.exists():
        return  # No entries to process

    users = User.objects.filter(timeentry__in=time_entries).distinct()

    for user in users:
        total_hours, project_work_summary = get_logged_hours(
            time_entries.filter(user=user)
        )

        total_days = total_hours / 7  # Assuming 7 hours/workday
        percentage = round((total_days * 100) / avg_work_days_per_month, 1)

        message = _template_time_logged.format(
            user=user.get_full_name(),
            last_month_name=last_month_name,
            project_work_summary=project_work_summary,
            percentage=percentage,
            current_month_name=current_month_name,
        )

        subject = f"Your Project Time Logged Summary for {last_month_name}"

        email_user(
            subject=subject,
            message=message,
            email=user.email,
        )

notify_monthly_time_logged_summary() ¤

Monthly task to notify users about their time logged.

Source code in main/tasks.py
150
151
152
153
154
155
156
157
158
159
160
161
162
@db_periodic_task(crontab(day=3, hour=10))
def notify_monthly_time_logged_summary() -> None:
    """Monthly task to notify users about their time logged."""
    last_month_start, last_month_name, current_month_start, current_month_name = (
        get_current_and_last_month()
    )

    notify_monthly_time_logged_logic(
        last_month_start,
        last_month_name,
        current_month_start,
        current_month_name,
    )

sync_clockify_time_entries() ¤

Task to sync time entries from Clockify API to TimeEntry model.

Source code in main/tasks.py
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
def sync_clockify_time_entries() -> None:
    """Task to sync time entries from Clockify API to TimeEntry model."""
    if not settings.CLOCKIFY_API_KEY or not settings.CLOCKIFY_WORKSPACE_ID:
        logger.warning("Clockify API key not found in environment variables")
        return
    days_back = 30
    api = ClockifyAPI(settings.CLOCKIFY_API_KEY, settings.CLOCKIFY_WORKSPACE_ID)
    end_date = timezone.now()
    start_date = end_date - datetime.timedelta(days=days_back)

    projects = Project.objects.filter(status="Active").exclude(clockify_id="")
    for project in projects:
        logger.info(f"Processing project ID: {project.clockify_id}")
        payload = {
            "dateRangeStart": start_date.strftime("%Y-%m-%dT00:00:00.000Z"),
            "dateRangeEnd": end_date.strftime("%Y-%m-%dT23:59:59.000Z"),
            "detailedFilter": {"page": 1, "pageSize": 200},
            "projects": {"contains": "CONTAINS", "ids": [project.clockify_id]},
        }

        try:
            response = api.get_time_entries(payload)
        except Exception as e:
            logger.error(
                f"Error fetching time entries for project {project.clockify_id}: {e}"
            )
            continue

        entries = response.get("timeentries", [])
        if not isinstance(entries, list):
            entries = []

        for entry in entries:
            entry_id = entry.get("id") or entry.get("_id")
            project_id = entry.get("projectId")
            user_email = entry.get("userEmail")
            time_interval = entry.get("timeInterval", {})
            start = time_interval.get("start")
            end = time_interval.get("end")

            if not (project_id and user_email and start and end):
                logger.warning(f"Skipping incomplete entry: {entry_id}")
                continue

            try:
                user = User.objects.get(email=user_email)
            except User.DoesNotExist:
                logger.warning(
                    f"User {user_email} not found. Skipping entry {entry_id}."
                )
                continue

            start_time = datetime.datetime.fromisoformat(start)
            end_time = datetime.datetime.fromisoformat(end)

            TimeEntry.objects.get_or_create(
                clockify_id=entry_id,
                defaults={
                    "user": user,
                    "project": project,
                    "start_time": start_time,
                    "end_time": end_time,
                },
            )

sync_clockify_time_entries_task() ¤

Scheduled task to sync time entries from Clockify API.

Source code in main/tasks.py
348
349
350
351
352
@db_periodic_task(crontab(day_of_week="mon", hour=2, minute=0))
def sync_clockify_time_entries_task() -> None:
    """Scheduled task to sync time entries from Clockify API."""
    sync_clockify_time_entries()
    logger.info("Clockify time entries sync completed.")