Oh no! The registrar recently contracted Programmer Bob's CodingDoneCheapAndDirty, inc. to build a new registration system for classes at Macalester. Unfortunately, Bob did a terrible job and the code is a mess! Registration time is in two weeks, and it's up to you to fix the system.
This assignment is an introduction to:
- Refactoring: changing the internal structure of software to make it easier to understand and cheaper to modify without changing its observable behavior.
- Object modeling: representing concepts as object-oriented code.
- Programming by contract: writing code that satisfies logical constraints.
- Defensive programming: writing code that actively prevents other code from using it incorrectly.
- This is not a group assignment. Please review the rules about collaboration vs. copying in the syllabus. (Bret’s syllabus | Paul’s syllabus).
- Use strict defensive programming: make it impossible to get any model object into an illegal state using public methods.
Programmer Bob created two model classes to represent students and courses. The code in these classes smells pretty bad, but thankfully Bob's partner Alice wrote a collection of tests that you can use to help fix the code.
The original specification for the code has the following invariant conditions:
- Students should know their registered courses, and courses should know the list of students enrolled in them.
For all students and courses,
student.getCourses().contains(course)
if and only ifcourse.getStudents().contains(student)
.
- Courses can have a max enrollment limit on the number of students. The enrollment limit cannot be changed once a student registers for the course.
For all courses,
course.getStudents().size()
≤course.getEnrollmentLimit()
.
- Courses can have a waitlist when they go over their enrollment limit. When a student attempts to register for a course that is full, they automatically go on the wait list. This is not an error condition; however, the
enroll()
method should let the caller know whether the enrollment was successful or the student was waitlisted.
A student is never both enrolled in and wait listed for the same course. If a course is not full, then its wait list is empty.
- Students should be able to drop a course. If an enrolled student drops, then the first wait-listed student is automatically enrolled. (That’s not realistic, of course, but it makes for a better programming exercise!)
Your task is to refactor the Student and Course classes to make the code easier to read and modify in the future. The goal isn't to make the code as small as possible, but rather to make it as clear and simple as possible. Think carefully about the separation of concerns between the classes. You should also pay attention to variable names and rename them if it makes the code clearer. You do not need to modify the test classes.
In addition to refactoring, try to engage in defensive programming. You should identify and fix any gaps where it is possible to use the code incorrectly.
…three things:
- Make sure that all of the invariants listed above still hold, even after all of your changes.
- Make sure that you test with the
RegistrarTest
test scenario to satisfactorily demonstrate that it works. - Practice good hygiene. Make sure your code is tidy, legible, and free of waste.