Schedule Creator
2025-03-22
Python implementation of creating schedules for the FSE LAS programmes.
Defining the Problem
A schedule
contains modules. A module
can be a course or skill and has one or more teaching activities
. For a module to be available for a selection, the requirement for its teaching activities must be met. Some teaching activities require availability for all of its timeslots, others only for some. For instance, a course
module might have:
- Lecture: Monday, 8:30-10:00
- Tutorial 1: Monday 10:15-11:15
- Tutorial 2: Wednesday: 8:30-10:00
A skill
module in contrast might have:
- Lab 1: Tuesday, 9:00-16:00
- Lab 2: Wednesday, 9:00-16:00
- Lab 3: Friday, 9:00-16:00
The course
is available if the schedule
holds empty slots for all time-slots (Lecture, Tutorial 1, Tutorial 2). The skill
is available if at least one of the time-slots (Lab 1, Lab 2, Lab 3) can be fitted into the schedule
.
This example was carefully chosen to illustrate a point: It becomes clear that the above yields 2 possible schedules. The time-slots for the course
are fixed, but we have flexibility when to schedule the skill
- either on Tuesday or on Friday. Now, we do not want to make any assumptions and keep track of all possible variations of the schedule. Therefore, we need to branch out. If the consider a tree-like structure, after the selecting the course
, we have one parent node. After selecting the skill
, we branch out from that node into two child nodes. Preserving all possible combinations is important for subsequent selections. Let us now add another course 2
:
- Lecture: Monday, 12:30-14:00
- Tutorial 1: Monday 14:15-15:45
- Tutorial 2: Friday: 8:30-10:00
Now we can see that there remains only one possible combination to fit all modules (course
, skill
, course2
) into our schedule - the skill
Friday time-slot drops out due to course2
's Tutorial 2. This example showcase the importance of maintaining a record of all possible schedule variations at any given point in the selection process.
Above, I defined two different types of modules - course
and skill
. I do not intend to hard-encode these into the implementation. Therefore, identifying whether any given time-slot is singly mandatory or as part of a group should be handled via a flag/identifier as part of the time-slots data representation. As an example, let us imagine another course3
with the following time-slots:
- Lecture 1: Monday, 8:30-10:00
- Tutorial 1: Monday 10:15-11:15
- Tutorial 2: Wednesday: 8:30-10:00
AND
- Lecture 2: Monday, 12:30-14:00
- Tutorial 3: Monday 14:15-15:45
- Tutorial 4: Friday: 8:30-10:00
Course 3
has a high demand which is why it is given twice per week. A student must be available for all time-slots of one of the above time-slot combinations, i.e. a student cannot take Lecture 1, Tutorial 1 and then Tutorial 4.
Similarly, skill2
is a Biology module that requires bacteria to cultivate for 24 hours. Therefore it is structured like this:
- Lab 1: Monday, 9:00-12:30 AND Tuesday, 13:00-16:00
- Lab 2: Thursday, 9:00-12:30 AND Friday, 13:00-16:00
These examples illustrate that we cannot rely on the mere type of a module to define the relationships between its constituent teaching activities. We need a flexible approach, the differentiation I am making between a course
and a skill
is merely to illustrate the example.
Implementation Ideas:
We are going to approach the implementation by designing a Python library. We will have one entry-point called ScheduleCreator
utilizing the Facade Pattern. Since this will be used as an API, we will need to be able to create multiple instance of the ScheduleCreator
simultaneously. The ScheduleCreator
exposes methods such as:
select_module()
deselect_module()
view_current_schedules()
- a nice
__repr__
for development
Using a contextmanager
, we make sure that Schedules
can only be created through the ScheduleCreator
.
We should also create our own generic data structure to capture the above illustrated tree-like structure for selecting modules and thereby creating schedules. The generic data structure should allow to:
- hold schedule variations
- add a module to all available schedule schedule variations -> this should be validated for overlaps first
- remove a module from all schedule variations
When it comes to the different classes to implement I was thinking about the following:
ScheduleCreator
Module
(maybe with a ModuleBase Abstract Base Class and then variations forskill
andcourse
- this would allow for future extension adhering to the Open/Closed Principle)Schedule
- a collection of ModulesValidatingTree
- our generic data structure
These are the guidelines. Please give me feedback, correct me or extend upon my ideas.