CSC/ECE 517 Spring 2025 - E2516. Reimplement teams users controller.rb
E2516: Reimplement teams_users_controller.rb
Introduction
Expertiza is an open-source project implementation for an educational platform. It hosts a lot of functionalities: the professor can create assignments for the students; the students can submit their assignments supported across multiple file formats and also review assignments uploaded by other students; the students can form team and work together for an assignment or a project. This platform provides synchronicity to the professor as well as students to effectively manage their work.
Project Overview
Through this project, we have reimplement a controller component, namely, teams_users_controller. The entire component is shifted to another controller, namely teams_participants_controller. It now achieves below mentioned functionalities:
- The code change now follows the SOLID and DRY principles
- All associations and references between participants and assignments and courses have been modified
- Clear distinction between "general user" and "assignment participant"
- Improved code readability by integrating helper methods
Files Modified
The following files have been modified to achieve the mentioned functionalities:
- TeamsParticipantsController
- AssignmentTeam
- MentoredTeam
- TeamsParticipant
TeamsParticipantsController
The changes made in this controller included refactoring the TeamsUsersController to TeamsParticipantsController. This name change reflects a very important feature that needs to be implemented: A participant is compulsorily to be assigned a course or an assignment whereas a user is general purpose member who uses the expertiza platform. Helper methods and various other methods have been modified in this controller file. One other association that is also implemented simultaneously is that a participant is to be assigned only and only one team unless the participant is a mentor.
def valid_participant?(user, assignment_or_course) if assignment_or_course.user_on_team?(user) flash[:error] = "This user is already assigned to a team for this #{assignment_or_course.class.name.downcase}." return false end participant = assignment_or_course.participants.find_by(user_id: user.id) unless participant flash[:error] = participant_not_found_message(user.name, assignment_or_course) return false end true end
Above method in the controller file implements this functionality where it checks whether a participant is assigned a team or not. If the participant is already assigned a team, it throws an error that the participant is already assigned a team. If not, it finds the participant id and accordingly assigns a team.
AssignmentTeam
The AssignmentTeam class is inherited from the Team class to include all functionalities present in the Team class. The primary purpose of this model is to depict the team members present for a given assignment or course. A lot of things could be achieved by having such a model and necessary methods have been devised to achieve them. Firstly, association rules have been established where each team can be assigned to multiple reviewers and can have multiple review responses. It defines rules for returning participants of a team, whether a participant is part of a team, assigning a reviewer to a team, drafting the review map when a reviewer reviews the work of the team, detailing participants of a team and if necessary, adding participants to a team, creating a new team and assigning a topic to the team, and lastly adding or modifying or deleting a hyperlink which represents the link to the work done by a particular team.
belongs_to :assignment, class_name: 'Assignment', foreign_key: 'parent_id' has_many :review_mappings, class_name: 'ReviewResponseMap', foreign_key: 'reviewee_id' has_many :review_response_maps, foreign_key: 'reviewee_id' has_many :responses, through: :review_response_maps, foreign_key: 'map_id'
MentoredTeam
The MentoredTeam class is inherited from the AssignmentTeam class. The purpose of this class is to validating a team in regards to participants already present and adding a mentor to the team compulsorily and that too only once. One other feature achieved through this model is to automate the task of creating teams by importing the team member data from a csv file and validating at the same time whether a user can be added to the team based on the capacity of the team and whether the user exists or not. One other feature achieved through this is that of logging every action performed from above description.
TeamsParticipant
The TeamsParticipant class is inherited from the ApplicationRecord class meaning that it is an active record model. This model manages the association between the participant and the teams including adding, removing and fetching participants from the team. It establishes relationships which state that each team can have many participants and one participant can have only one team.
Test Cases for AssignmentTeam
Below are the test cases implemented for the `AssignmentTeam` model:
require 'rails_helper' RSpec.describe AssignmentTeam, type: :model do let(:assignment) { create(:assignment) } let(:team) { create(:assignment_team, assignment: assignment) } let(:user) { create(:user) } let(:participant) { create(:participant, user: user, assignment: assignment) } let(:reviewer) { create(:participant, assignment: assignment) } let(:review_response_map) { create(:review_response_map, reviewee: team, reviewer: reviewer) } describe '#user_id' do it 'returns the user_id of the first team member' do team.users << user expect(team.user_id).to eq(user.id) end it 'returns current_user.id if they are in the team' do team.users << user expect(team.user_id(user)).to eq(user.id) end end describe '#includes?' do it 'returns true if a participant is in the team' do allow(team).to receive(:participants).and_return([participant]) expect(team.includes?(participant)).to be true end it 'returns false if a participant is not in the team' do allow(team).to receive(:participants).and_return([]) expect(team.includes?(participant)).to be false end end describe '#parent_model' do it 'returns "Assignment"' do expect(team.parent_model).to eq('Assignment') end end describe '#assign_reviewer' do it 'raises an error if the assignment is not found' do allow(Assignment).to receive(:find_by).and_return(nil) expect { team.assign_reviewer(reviewer) }.to raise_error('The assignment cannot be found.') end it 'creates a review map for the reviewer' do expect { team.assign_reviewer(reviewer) }.to change { ReviewResponseMap.count }.by(1) end end describe '#create_review_map' do it 'creates a new ReviewResponseMap' do expect { team.create_review_map(reviewer, assignment) }.to change { ReviewResponseMap.count }.by(1) end end describe '#reviewed_by?' do it 'returns true if the team has been reviewed by the given reviewer' do review_response_map expect(team.reviewed_by?(reviewer)).to be true end it 'returns false if the team has not been reviewed by the given reviewer' do expect(team.reviewed_by?(reviewer)).to be false end end describe '#participants' do it 'returns the participants of the team' do allow(TeamsParticipant).to receive(:team_members).with(team.id).and_return([participant]) expect(team.participants).to include(participant) end end describe '#add_participant' do it 'adds a participant to the team' do expect { team.add_participant(assignment.id, user) }.to change { TeamsParticipant.count }.by(1) end it 'does not add a participant if they are already in the team' do team.add_participant(assignment.id, user) expect { team.add_participant(assignment.id, user) }.not_to change { TeamsParticipant.count } end end describe '#create_new_team' do let(:signuptopic) { create(:sign_up_topic, assignment: assignment) } it 'creates a new team user and associates topic' do expect { team.create_new_team(user.id, signuptopic) }.to change { TeamsUser.count }.by(1) .and change { SignedUpTeam.count }.by(1) .and change { TeamNode.count }.by(1) .and change { TeamUserNode.count }.by(1) end end describe '#submit_hyperlink' do it 'calls TeamFileService.submit_hyperlink' do expect(TeamFileService).to receive(:submit_hyperlink).with(team, 'http://example.com') team.submit_hyperlink('http://example.com') end end describe '#remove_hyperlink' do it 'calls TeamFileService.remove_hyperlink' do expect(TeamFileService).to receive(:remove_hyperlink).with(team, 'http://example.com') team.remove_hyperlink('http://example.com') end end describe '#has_submissions?' do it 'returns true if the team has submissions' do allow(team).to receive(:submitted_files).and_return(['file1']) allow(team).to receive(:submitted_hyperlinks).and_return(nil) expect(team.has_submissions?).to be true end it 'returns false if the team has no submissions' do allow(team).to receive(:submitted_files).and_return([]) allow(team).to receive(:submitted_hyperlinks).and_return(nil) expect(team.has_submissions?).to be false end end describe '#most_recent_submission' do it 'returns the latest submission' do submission1 = create(:submission_record, team: team, assignment: assignment, updated_at: 1.day.ago) submission2 = create(:submission_record, team: team, assignment: assignment, updated_at: Time.current) expect(team.most_recent_submission).to eq(submission2) end end end
Test Cases for MentoredTeam
Below are the test cases implemented for the `MentoredTeam` model:
require 'rails_helper' RSpec.describe MentoredTeam, type: :model do let(:team) { create(:mentored_team) } let(:user) { create(:user) } let(:mentor) { create(:user, role: :mentor) } describe '#import_team_members' do it 'imports members successfully from a given list' do members = [create(:user), create(:user)] expect { team.import_team_members(members) }.to change { team.users.count }.by(2) end end describe '#find_or_raise_user' do it 'returns the user if found' do expect(team.find_or_raise_user(user.id)).to eq(user) end it 'raises an error if the user is not found' do expect { team.find_or_raise_user(999) }.to raise_error(ActiveRecord::RecordNotFound) end end describe '#user_not_in_team?' do it 'returns true if user is not in the team' do expect(team.user_not_in_team?(user)).to be true end it 'returns false if user is in the team' do team.users << user expect(team.user_not_in_team?(user)).to be false end end describe '#mentor_assignment_valid?' do it 'returns true if a mentor can be assigned' do expect(team.mentor_assignment_valid?(mentor)).to be true end it 'returns false if an invalid mentor is assigned' do expect(team.mentor_assignment_valid?(user)).to be false end end describe '#add_member' do it 'adds a user to the team' do expect { team.add_member(user) }.to change { team.users.count }.by(1) end it 'does not add a user who is already in the team' do team.users << user expect { team.add_member(user) }.not_to change { team.users.count } end end describe '#can_add_member?' do it 'returns true if the team can add a member' do expect(team.can_add_member?).to be true end it 'returns false if the team has reached its limit' do allow(team).to receive(:users).and_return(Array.new(10) { create(:user) }) expect(team.can_add_member?).to be false end end describe '#add_team_user' do it 'adds a user to the team successfully' do expect { team.add_team_user(user) }.to change { team.users.count }.by(1) end end describe '#add_participant_to_team' do it 'adds a participant to the team' do participant = create(:participant) expect { team.add_participant_to_team(participant) }.to change { team.users.count }.by(1) end end describe '#assign_mentor_if_needed' do it 'assigns a mentor if no mentor is present' do team.assign_mentor_if_needed(mentor) expect(team.mentor).to eq(mentor) end it 'does not assign a mentor if one is already assigned' do existing_mentor = create(:user, role: :mentor) team.mentor = existing_mentor team.assign_mentor_if_needed(mentor) expect(team.mentor).to eq(existing_mentor) end end end
Mentor
- Vihar Manojkumar Shah(vshah23@ncsu.edu)
Team Members
- Manav Kamdar (mkamdar@ncsu.edu)
- Aditya Singh (asingh78@ncsu.edu)
- Shivang Patel (spatel74@ncsu.edu)