Skip to content

Commit

Permalink
Avoid re-fetching rosters that have failed recently
Browse files Browse the repository at this point in the history
We already have a mechanism, checking the existence of the roster, to
avoid constantly fetching the same rosters.

We don't have however a way to avoid re-fetching rosters that have
failed recently.

We'll insert a row in task_done before the roster fetch to signal that
we have attempted to fetch that roster.

When looking for candidates to fetch new rosters for we'll exclude ones
that have a task_done row.
  • Loading branch information
marcospri committed Sep 13, 2024
1 parent 2e82ba5 commit ecddeeb
Show file tree
Hide file tree
Showing 2 changed files with 72 additions and 6 deletions.
45 changes: 39 additions & 6 deletions lms/tasks/roster.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@

from datetime import datetime, timedelta

from sqlalchemy import exists, select
from sqlalchemy import exists, func, select

from lms.models import (
Assignment,
Expand All @@ -11,6 +11,7 @@
CourseRoster,
Event,
LMSCourse,
TaskDone,
)
from lms.services.roster import RosterService
from lms.tasks.celery import app
Expand All @@ -37,6 +38,13 @@ def schedule_fetching_course_rosters() -> None:
# We use the python version (and not func.now()) for easier mocking during tests
now = datetime.now()

# Only fetch roster for courses for which we haven't schedule a fetch recently
no_recent_scheduled_roster_fetch_clause = ~exists(
select(TaskDone).where(
TaskDone.key == func.concat("roster::course::", LMSCourse.id),
)
)

# Only fetch roster for courses that don't have recent roster information
no_recent_roster_clause = ~exists(
select(CourseRoster).where(
Expand All @@ -46,7 +54,7 @@ def schedule_fetching_course_rosters() -> None:
)

# Only fetch rosters for courses that have been recently launched
recent_launches_cluase = exists(
recent_launches_clause = exists(
select(Event)
.join(Course, Event.course_id == Course.id)
.where(
Expand All @@ -63,7 +71,8 @@ def schedule_fetching_course_rosters() -> None:
# Courses for which we have a LTIA membership service URL
LMSCourse.lti_context_memberships_url.is_not(None),
no_recent_roster_clause,
recent_launches_cluase,
no_recent_scheduled_roster_fetch_clause,
recent_launches_clause,
)
# Prefer newer courses
.order_by(LMSCourse.created.desc())
Expand All @@ -72,6 +81,15 @@ def schedule_fetching_course_rosters() -> None:
)
for lms_course_id in request.db.scalars(query).all():
fetch_course_roster.delay(lms_course_id=lms_course_id)
# Record that the roster fetching has been scheduled
# We set the expiration date to ROSTER_REFRESH_WINDOW so we'll try again after that period
request.db.add(
TaskDone(
key=f"roster::course::{lms_course_id}",
data=None,
expires_at=datetime.now() + ROSTER_REFRESH_WINDOW,
)
)


def schedule_fetching_assignment_rosters() -> None:
Expand All @@ -80,16 +98,21 @@ def schedule_fetching_assignment_rosters() -> None:
# We use the python version (and not func.now()) for easier mocking during tests
now = datetime.now()

# Only fetch roster for assignments that don't have recent roster information
no_recent_roster_clause = ~exists(
select(AssignmentRoster).where(
AssignmentRoster.assignment_id == Assignment.id,
AssignmentRoster.updated >= now - ROSTER_REFRESH_WINDOW,
)
)

no_recent_scheduled_roster_fetch_clause = ~exists(
select(TaskDone).where(
TaskDone.key == func.concat("roster::assignment::", Assignment.id),
)
)

# Only fetch rosters for assignments that have been recently launched
recent_launches_cluase = exists(
recent_launches_clause = exists(
select(Event)
.join(Assignment, Event.assignment_id == Assignment.id)
.where(
Expand All @@ -111,8 +134,9 @@ def schedule_fetching_assignment_rosters() -> None:
LMSCourse.lti_context_memberships_url.is_not(None),
# Assignments for which we have stored the LTI1.3 ID
Assignment.lti_v13_resource_link_id.is_not(None),
no_recent_scheduled_roster_fetch_clause,
no_recent_roster_clause,
recent_launches_cluase,
recent_launches_clause,
)
# Prefer newer assignments
.order_by(Assignment.created.desc())
Expand All @@ -121,6 +145,15 @@ def schedule_fetching_assignment_rosters() -> None:
)
for assignment_id in request.db.scalars(query).all():
fetch_assignment_roster.delay(assignment_id=assignment_id)
# Record that the roster fetching has been scheduled
# We set the expiration date to ROSTER_REFRESH_WINDOW so we'll try again after that period
request.db.add(
TaskDone(
key=f"roster::assignment::{assignment_id}",
data=None,
expires_at=datetime.now() + ROSTER_REFRESH_WINDOW,
)
)


@app.task(
Expand Down
33 changes: 33 additions & 0 deletions tests/unit/lms/tasks/roster_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@ def test_schedule_fetching_rosters(
"lms_course_with_no_launch",
"lms_course_with_no_service_url",
"lms_course_with_launch_and_recent_roster",
"lms_course_with_recent_launch_and_task_done_row",
)
def test_schedule_fetching_course_rosters(
self, lms_course_with_recent_launch, db_session, fetch_course_roster
Expand All @@ -60,6 +61,7 @@ def test_schedule_fetching_course_rosters(
@pytest.mark.usefixtures(
"assignment_with_no_launch",
"assignment_with_no_lti_v13_id",
"assignment_with_recent_launch_and_task_done_row",
"assignment_with_launch_and_recent_roster",
)
def test_schedule_fetching_assignment_rosters(
Expand Down Expand Up @@ -105,6 +107,22 @@ def lms_course_with_recent_launch(self):
course=course,
)

@pytest.fixture
def lms_course_with_recent_launch_and_task_done_row(self, db_session):
course = factories.Course()
factories.Event(
course=course,
timestamp=datetime(2024, 8, 28),
)
lms_course = factories.LMSCourse(
lti_context_memberships_url="URL",
h_authority_provided_id=course.authority_provided_id,
course=course,
)
db_session.flush() # Make sure we have an ID for the course
factories.TaskDone(key=f"roster::course::{lms_course.id}")
return lms_course

@pytest.fixture
def assignment_with_recent_launch(self, lms_course_with_recent_launch):
assignment = factories.Assignment(
Expand All @@ -116,6 +134,21 @@ def assignment_with_recent_launch(self, lms_course_with_recent_launch):
)
return assignment

@pytest.fixture
def assignment_with_recent_launch_and_task_done_row(
self, lms_course_with_recent_launch, db_session
):
assignment = factories.Assignment(
lti_v13_resource_link_id="ID", course=lms_course_with_recent_launch.course
)
factories.Event(
assignment=assignment,
timestamp=datetime(2024, 8, 28),
)
db_session.flush() # Make sure we have an ID for the assignment
factories.TaskDone(key=f"roster::assignment::{assignment.id}")
return assignment

@pytest.fixture
def lms_course_with_launch_and_recent_roster(self):
course = factories.Course()
Expand Down

0 comments on commit ecddeeb

Please sign in to comment.