E1837 refactor review mapping controller: Difference between revisions
| No edit summary | |||
| (11 intermediate revisions by the same user not shown) | |||
| Line 1: | Line 1: | ||
| == Background == | |||
| At the heart of Expertiza is the ability to review the work of other teams and review your own teammates.The Review Mapping Controller is at the heart of coordinating all of the necessary steps to assign reviews in various ways to specific students or teams. | |||
| As such, this controller has become rather large and unmaintainable over time as new types of reviews and review assignment strategies are created. The purpose of this work is to improve the maintainability of this controller based on the goals listed below. | |||
| == Goals == | == Goals == | ||
| :- remove unused  | :- separation of concerns | ||
| :- remove unused methods | |||
| :- properly group like methods | |||
| :- remove SQL statements from the controller into models/helpers where applicable | :- remove SQL statements from the controller into models/helpers where applicable | ||
| == Separation of Concerns == | |||
| ==  | |||
| As part of refactoring, we have introduced several new Helper classes and methods. | As part of refactoring, we have introduced several new Helper classes and methods. | ||
| Line 13: | Line 16: | ||
| This helper now contains isolated methods for each report type that needs to be built from the controller. This helper is implemented as a module, so any/all instance variables are added to the class including it. This allows us to keep the controller logic very small but still populate all needed values for the report views. | This helper now contains isolated methods for each report type that needs to be built from the controller. This helper is implemented as a module, so any/all instance variables are added to the class including it. This allows us to keep the controller logic very small but still populate all needed values for the report views. | ||
| Each report type is represented by a separate method in the Helper and the Helper exposes a single public method to render the report. This allows us to hide implementation details to be changed later as needed from within the module and the controller can generally stay the same. The only caveat is that all of the correct instance variables will need to be defined and populated per report. We considered refactoring the reports section, but after discussion with our  | |||
| mentor, we decided it would be a large enough effort in of itself to change how reports are generated and rendered. | |||
| === Strategy Pattern for Automatic Review Mapping === | === Strategy Pattern for Automatic Review Mapping === | ||
| The previous implementation for the Automatic Review mapping method (automatic_review_mapping) was lengthy and overrode specific arguments within the Automatic Review Mapping Strategy method (automatic_review_mapping_strategy) before actually assigning reviews. We decided to implement an actual Strategy pattern here. | The previous implementation for the Automatic Review mapping method (<code>automatic_review_mapping</code>) was lengthy and overrode specific arguments within the Automatic Review Mapping Strategy method (<code>automatic_review_mapping_strategy</code>) before actually assigning reviews. We decided to implement an actual Strategy pattern here. | ||
| We created a simple ReviewStrategy class with the following subclasses: | We created a simple ReviewStrategy class with the following subclasses: | ||
| Line 23: | Line 27: | ||
| Each of these strategies determines the number of reviews per team, the number of reviews per student, and the total number of reviews needed. After implementing this pattern, we were able to simply use the methods provided on the ReviewStrategy class in the algorithm that assigns reviews with minimal refactoring. | Each of these strategies determines the number of reviews per team, the number of reviews per student, and the total number of reviews needed. After implementing this pattern, we were able to simply use the methods provided on the ReviewStrategy class in the algorithm that assigns reviews with minimal refactoring. | ||
| Below is a simple UML diagram of these new classes: | Below is a simple UML diagram of these new classes: | ||
| Line 31: | Line 34: | ||
| == Removed Methods == | == Removed Methods == | ||
| We were able to remove the following methods that were either obsolete or had functionality that could be refactored into other methods. | We were able to remove the following methods that were either obsolete or had functionality that could be refactored into other methods. | ||
| :- execute_peer_review_strategy: This functionality simply performed a check that is now in automatic_review_mapping | :- <code>execute_peer_review_strategy</code>: This functionality simply performed a check that is now in <code>automatic_review_mapping</code> | ||
| == Removed SQL == | |||
| Several methods had raw SQL, or at the very least SQL-like statements. Where possible, these calls were pushed down into their respective models. | |||
| === Start Self Review === | |||
| This method had the following SQL | |||
| <code> | |||
| SELECT t.id as t_id FROM teams_users u, teams t WHERE u.team_id = t.id and t.parent_id = ? and user_id = ? | |||
| </code> | |||
| It was migrated to be a scope inside of the Team model (<code>find_team_for_assignment_and_user</code>). | |||
| == Test Plan == | |||
| The majority of our changes were refactors, and thusly required a few new tests, but we did need to adjust existing tests to expect proper mocked/seamed methods. | |||
| === Tests Added === | |||
| {| class="wikitable" | |||
| |- | |||
| ! Test Name | |||
| ! Class | |||
| ! Description | |||
| |- | |||
| | when student review num is greater than or equal to team size throws an error stating that student review number cannot be greater than or equal to team size | |||
| | /spec/review_mapping_controller_spec.rb | |||
| | Validates that the instructor cannot request more reviews than the team size for an assignment | |||
| |- | |||
| |} | |||
| == Suggestions for Additional Improvements == | |||
| The report generation should be refactored to allow each report to specify how it should be rendered. One idea was setting each report to supply a collection of "report lines" that are rendered one at a time in a table. This would allow for each report to specify its data and its rendering functions separatly. | |||
Latest revision as of 18:54, 9 November 2018
Background
At the heart of Expertiza is the ability to review the work of other teams and review your own teammates.The Review Mapping Controller is at the heart of coordinating all of the necessary steps to assign reviews in various ways to specific students or teams.
As such, this controller has become rather large and unmaintainable over time as new types of reviews and review assignment strategies are created. The purpose of this work is to improve the maintainability of this controller based on the goals listed below.
Goals
- - separation of concerns
- - remove unused methods
- - properly group like methods
- - remove SQL statements from the controller into models/helpers where applicable
Separation of Concerns
As part of refactoring, we have introduced several new Helper classes and methods.
Report Formatter Helper
This helper now contains isolated methods for each report type that needs to be built from the controller. This helper is implemented as a module, so any/all instance variables are added to the class including it. This allows us to keep the controller logic very small but still populate all needed values for the report views.
Each report type is represented by a separate method in the Helper and the Helper exposes a single public method to render the report. This allows us to hide implementation details to be changed later as needed from within the module and the controller can generally stay the same. The only caveat is that all of the correct instance variables will need to be defined and populated per report. We considered refactoring the reports section, but after discussion with our mentor, we decided it would be a large enough effort in of itself to change how reports are generated and rendered.
Strategy Pattern for Automatic Review Mapping
The previous implementation for the Automatic Review mapping method (automatic_review_mapping) was lengthy and overrode specific arguments within the Automatic Review Mapping Strategy method (automatic_review_mapping_strategy) before actually assigning reviews. We decided to implement an actual Strategy pattern here.
We created a simple ReviewStrategy class with the following subclasses:
- - StudentReviewStrategy
- - TeamReviewStrategy
Each of these strategies determines the number of reviews per team, the number of reviews per student, and the total number of reviews needed. After implementing this pattern, we were able to simply use the methods provided on the ReviewStrategy class in the algorithm that assigns reviews with minimal refactoring.
Below is a simple UML diagram of these new classes:
Removed Methods
We were able to remove the following methods that were either obsolete or had functionality that could be refactored into other methods.
- - execute_peer_review_strategy: This functionality simply performed a check that is now inautomatic_review_mapping
Removed SQL
Several methods had raw SQL, or at the very least SQL-like statements. Where possible, these calls were pushed down into their respective models.
Start Self Review
This method had the following SQL
SELECT t.id as t_id FROM teams_users u, teams t WHERE u.team_id = t.id and t.parent_id = ? and user_id = ?
It was migrated to be a scope inside of the Team model (find_team_for_assignment_and_user).
Test Plan
The majority of our changes were refactors, and thusly required a few new tests, but we did need to adjust existing tests to expect proper mocked/seamed methods.
Tests Added
| Test Name | Class | Description | 
|---|---|---|
| when student review num is greater than or equal to team size throws an error stating that student review number cannot be greater than or equal to team size | /spec/review_mapping_controller_spec.rb | Validates that the instructor cannot request more reviews than the team size for an assignment | 
Suggestions for Additional Improvements
The report generation should be refactored to allow each report to specify how it should be rendered. One idea was setting each report to supply a collection of "report lines" that are rendered one at a time in a table. This would allow for each report to specify its data and its rendering functions separatly.
