Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Development: Remove quiz view and switch to DTOs #10571

Merged
merged 30 commits into from
Mar 25, 2025

Conversation

KonstiAnon
Copy link
Contributor

@KonstiAnon KonstiAnon commented Mar 23, 2025

Checklist

General

Server

  • Important: I implemented the changes with a very good performance and prevented too many (unnecessary) and too complex database calls.
  • I strictly followed the principle of data economy for all database calls.
  • I strictly followed the server coding and design guidelines.
  • I adapted multiple integration tests (Spring) related to the features (with a high test coverage).
  • I documented the Java code using JavaDoc style.

Motivation and Context

Up until now, a JsonView was used in parts of the codebase to filter which information is sent to users based on the current context. This approach has been replaced by the usage of DTOs over time and as such the QuizView has become deprecated. This PR removes the QuizView from the code and replaces its usage in endpoints with new DTOs.

Description

I have replaced the usage of the QuizView in the following functions:

  • sendQuizExerciseToSubscribedClients(), QuizMessagingService.java
  • startParticipation(), QuizParticipationResource.java
  • participationForQuizExercise(), ParticipationResource.java

with the usage of DTOs. For this I have implemented a number of DTOs, which currently replicate the exact same data structure as before. There are three situations based on which an appropriate DTO is selected:

  • A Quiz has not yet started (Students can not yet see the quiz questions)
  • A Quiz is currently running (Students can see the questions, but not the answers)
  • A Quiz has ended (Students can see the questions and the answers)

The StudentQuizParticipationWithoutQuestionsDTO is used as a replacement for QuizView.Before, the StudentQuizParticipationWithQuestionsDTO replaces QuizView.During and the StudentQuizParticipationWithSolutions replaces the QuizView.After.

Steps for Testing

Prerequisites:

  • 1 Instructor
  • 2 Students
  • 1 Quiz exercise
  1. Create the course quiz exercise
  2. Participate in the quiz
  3. Review the solution
  4. In all cases make sure that students cannot see the question before the quiz starts (also look into the network tab and inspect the json) and they cannot see the solution before the quiz ends

Exam Mode Testing

Prerequisites:

  • 1 Instructor
  • 2 Students
  • 1 Exam with Quiz Exercises
  1. Create the exam quiz exercise
  2. Participate in the quiz
  3. Review the solution after the exam ends
  4. In all cases make sure that students cannot see the question before the quiz starts (also look into the network tab and inspect the json) and they cannot see the solution before the quiz ends

Testserver States

You can manage test servers using Helios. Check environment statuses in the environment list. To deploy to a test server, go to the CI/CD page, find your PR or branch, and trigger the deployment.

Review Progress

Performance Review

  • I (as a reviewer) confirm that the server changes (in particular related to database calls) are implemented with a very good performance even for very large courses with more than 2000 students.

Code Review

  • Code Review 1
  • Code Review 2

Manual Tests

  • Test 1
  • Test 2

Exam Mode Test

  • Test 1
  • Test 2

Test Coverage

Summary by CodeRabbit

  • New Features

    • Introduced tailored data responses for quiz exercises and participation that dynamically adjust the content based on quiz state (e.g., active, ended).
    • Added new data representations for questions, answer options, and submissions, ensuring clearer and more secure API responses.
  • Refactor

    • Streamlined JSON serialization by replacing conditional view annotations with dedicated data transfer objects.
    • Optimized endpoint logic in the quiz messaging and participation services for precise data delivery.
  • Tests

    • Enhanced integration tests to verify correct handling of diverse question types and participation behavior under active quiz conditions.

@github-actions github-actions bot added server Pull requests that update Java code. (Added Automatically!) assessment Pull requests that affect the corresponding module core Pull requests that affect the corresponding module exercise Pull requests that affect the corresponding module programming Pull requests that affect the corresponding module quiz Pull requests that affect the corresponding module labels Mar 23, 2025
@KonstiAnon KonstiAnon changed the title Quiz exercises: Remove QuizView and switch to DTOs ´Quiz exercises´: Remove QuizView and switch to DTOs Mar 23, 2025
@KonstiAnon KonstiAnon changed the title ´Quiz exercises´: Remove QuizView and switch to DTOs 'Quiz exercises': Remove QuizView and switch to DTOs Mar 23, 2025
@helios-aet helios-aet bot temporarily deployed to artemis-test3.artemis.cit.tum.de March 24, 2025 17:56 Inactive
@helios-aet helios-aet bot temporarily deployed to artemis-test3.artemis.cit.tum.de March 24, 2025 18:20 Inactive
@helios-aet helios-aet bot temporarily deployed to artemis-test3.artemis.cit.tum.de March 25, 2025 10:49 Inactive
AjayvirS
AjayvirS previously approved these changes Mar 25, 2025
Copy link
Contributor

@AjayvirS AjayvirS left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Tested on TS3, works as intended

@helios-aet helios-aet bot temporarily deployed to artemis-test4.artemis.cit.tum.de March 25, 2025 13:19 Inactive
Copy link
Member

@Hialus Hialus left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Code looks good. Only relevant thing is to get the tests to pass by renaming the interface.
Everything else will be done in follow ups

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This file is specific to quiz exercises and should therefore be within the quiz module

Comment on lines 342 to 336
protected Course getCourse() {
public Course getCourse() {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please revert this change

@@ -798,6 +802,7 @@ public ResponseEntity<MappingJacksonValue> getParticipationForCurrentUser(@PathV
}

@Nullable
// TODO: use a proper DTO (or interface here for the return type and avoid MappingJacksonValue)
private MappingJacksonValue participationForQuizExercise(QuizExercise quizExercise, User user) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In this method you still return a raw StudentParticipation in Line 822.

Comment on lines +836 to +849
Object responseDTO = null;
if (participation != null) {
if (quizExercise.isQuizEnded()) {
responseDTO = StudentQuizParticipationWithSolutionsDTO.of(participation);
}
else if (quizBatch.get().isStarted()) {
responseDTO = StudentQuizParticipationWithQuestionsDTO.of(participation);
}
else {
responseDTO = StudentQuizParticipationWithoutQuestionsDTO.of(participation);
}
}

return responseDTO != null ? new MappingJacksonValue(responseDTO) : null;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These null checks is either redundant or you should probably throw a proper exception

@@ -126,7 +119,8 @@ public void setSimilarityValue(Integer similarityValue) {
this.similarityValue = similarityValue;
}

public Boolean matchLetterCase() {
@JsonInclude
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this needed? Might make more sense to put it at the attribute

@@ -0,0 +1,5 @@
package de.tum.cit.aet.artemis.quiz.dto.participation;

public interface StudentQuizParticipation {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Rename this to StudentQuizParticipationDTO for now. We'll think of a possible better naming later.

public static StudentQuizParticipationWithQuestionsDTO of(final StudentParticipation studentParticipation) {
Exercise participationExercise = studentParticipation.getExercise();
if (!(participationExercise instanceof QuizExercise quizExercise)) {
// TODO: Figure out error handling here
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Comment just to track this for a follow up

import de.tum.cit.aet.artemis.quiz.domain.QuizQuestion;
import de.tum.cit.aet.artemis.quiz.domain.ShortAnswerQuestion;

// Note: Only one of the three questions will be non-null depending on the question type
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please add a more detailed JavaDoc describing this "hack"

Comment on lines 108 to 117
StudentQuizParticipation responseDTO;
if (exercise.isQuizEnded()) {
responseDTO = StudentQuizParticipationWithSolutionsDTO.of(participation);
}
else if (quizBatch.isPresent() && quizBatch.get().isStarted()) {
responseDTO = StudentQuizParticipationWithQuestionsDTO.of(participation);
}
else {
responseDTO = StudentQuizParticipationWithoutQuestionsDTO.of(participation);
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

See if this can be moved into a method in the QuizExercise

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

♻️ Duplicate comments (3)
src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationWithSolutionsDTO.java (1)

25-30: ⚠️ Potential issue

Implement error handling instead of returning null

The factory method has a TODO comment about error handling but currently returns null in case of type mismatch. This could lead to NullPointerExceptions. Consider implementing proper error handling by throwing an appropriate exception.

    public static StudentQuizParticipationWithSolutionsDTO of(final StudentParticipation studentParticipation) {
        Exercise participationExercise = studentParticipation.getExercise();
        if (!(participationExercise instanceof QuizExercise quizExercise)) {
-            // TODO: Figure out error handling here
-            return null;
+            throw new IllegalArgumentException("Cannot create StudentQuizParticipationWithSolutionsDTO: " +
+                    "The provided participation's exercise is not a QuizExercise");
        }
src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationWithQuestionsDTO.java (2)

27-30: ⚠️ Potential issue

Fix the error handling instead of returning null.

The current implementation returns null when the participation exercise is not a QuizExercise. This can lead to NullPointerExceptions downstream. Throw an appropriate exception with a descriptive message instead.

-        if (!(participationExercise instanceof QuizExercise quizExercise)) {
-            // TODO: Figure out error handling here
-            return null;
-        }
+        if (!(participationExercise instanceof QuizExercise quizExercise)) {
+            throw new IllegalArgumentException("StudentParticipation is not associated with a QuizExercise");
+        }

31-33: 🛠️ Refactor suggestion

Remove deprecated "Results" mapping now that QuizView is removed.

Since this PR is removing QuizView and the TODO comment indicates that Results mapping should be removed after QuizView is removed, now is the appropriate time to address this TODO.

-        // ToDo: Results is deprecated, will be removed after QuizView is removed
-        return new StudentQuizParticipationWithQuestionsDTO(StudentQuizParticipationBaseDTO.of(studentParticipation), QuizExerciseWithQuestionsDTO.of(quizExercise),
-                studentParticipation.getResults().stream().map(ResultBeforeEvaluationDTO::of).collect(Collectors.toSet()));
+        return new StudentQuizParticipationWithQuestionsDTO(StudentQuizParticipationBaseDTO.of(studentParticipation), 
+                QuizExerciseWithQuestionsDTO.of(quizExercise), null);
🧹 Nitpick comments (2)
src/test/java/de/tum/cit/aet/artemis/quiz/QuizSubmissionIntegrationTest.java (2)

314-317: Consider extracting batch creation logic to a helper method.

This block creates a quiz batch and joins it for non-synchronized quiz modes. Since this pattern appears in other test methods (e.g., lines 831-836), consider extracting it to a reusable helper method to avoid duplication.

-        if (quizMode != QuizMode.SYNCHRONIZED) {
-            var batch = quizBatchService.save(QuizExerciseFactory.generateQuizBatch(quizExercise, ZonedDateTime.now().minusSeconds(10)));
-            request.postWithResponseBody("/api/quiz/quiz-exercises/" + quizExercise.getId() + "/join", new QuizBatchJoinDTO(batch.getPassword()), QuizBatch.class, HttpStatus.OK);
-        }
+        if (quizMode != QuizMode.SYNCHRONIZED) {
+            joinQuizBatch(quizExercise);
+        }

With a helper method like:

private void joinQuizBatch(QuizExercise quizExercise) throws Exception {
    var batch = quizBatchService.save(QuizExerciseFactory.generateQuizBatch(quizExercise, ZonedDateTime.now().minusSeconds(10)));
    request.postWithResponseBody("/api/quiz/quiz-exercises/" + quizExercise.getId() + "/join", 
        new QuizBatchJoinDTO(batch.getPassword()), QuizBatch.class, HttpStatus.OK);
}

321-344: Enhance test assertions with descriptive messages.

The assertions would be more maintainable with descriptive failure messages that explain the expected behavior.

-        assertThat(participation).isNotNull();
-        Exercise exercise = participation.getExercise();
-        assertThat(exercise).isNotNull();
-        assertThat(exercise).isInstanceOf(QuizExercise.class);
+        assertThat(participation).as("Participation should not be null").isNotNull();
+        Exercise exercise = participation.getExercise();
+        assertThat(exercise).as("Exercise in participation should not be null").isNotNull();
+        assertThat(exercise).as("Exercise should be a QuizExercise").isInstanceOf(QuizExercise.class);
📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between b6b4c56 and b2b6b1f.

📒 Files selected for processing (6)
  • src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationDTO.java (1 hunks)
  • src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationWithQuestionsDTO.java (1 hunks)
  • src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationWithSolutionsDTO.java (1 hunks)
  • src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationWithoutQuestionsDTO.java (1 hunks)
  • src/main/java/de/tum/cit/aet/artemis/quiz/web/QuizParticipationResource.java (3 hunks)
  • src/test/java/de/tum/cit/aet/artemis/quiz/QuizSubmissionIntegrationTest.java (2 hunks)
✅ Files skipped from review due to trivial changes (1)
  • src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationDTO.java
🚧 Files skipped from review as they are similar to previous changes (2)
  • src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationWithoutQuestionsDTO.java
  • src/main/java/de/tum/cit/aet/artemis/quiz/web/QuizParticipationResource.java
🧰 Additional context used
📓 Path-based instructions (2)
`src/test/java/**/*.java`: test_naming: descriptive; test_size: small_specific; fixed_data: true; junit5_features: true; assert_use: assertThat; assert_specificity: true; archunit_...

src/test/java/**/*.java: test_naming: descriptive; test_size: small_specific; fixed_data: true; junit5_features: true; assert_use: assertThat; assert_specificity: true; archunit_use: enforce_package_rules; db_query_count_tests: track_performance; util_service_factory_pattern: true; avoid_db_access: true; mock_strategy: static_mocks; context_restart_minimize: true

  • src/test/java/de/tum/cit/aet/artemis/quiz/QuizSubmissionIntegrationTest.java
`src/main/java/**/*.java`: naming:CamelCase; principles:{single_responsibility,small_methods,no_duplication}; db:{perf_queries,datetime_not_timestamp}; rest:{stateless,singleton,de...

src/main/java/**/*.java: naming:CamelCase; principles:{single_responsibility,small_methods,no_duplication}; db:{perf_queries,datetime_not_timestamp}; rest:{stateless,singleton,delegate_logic,http_only,minimal_dtos}; dtos:{java_records,no_entities,min_data,single_resp}; di:constructor_injection; kiss:simple_code; file_handling:os_indep_paths; practices:{least_access,avoid_transactions,code_reuse,static_member_ref,prefer_primitives}; sql:{param_annotation,uppercase,avoid_subqueries};java:avoid_star_imports

  • src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationWithSolutionsDTO.java
  • src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationWithQuestionsDTO.java
🧠 Learnings (1)
src/test/java/de/tum/cit/aet/artemis/quiz/QuizSubmissionIntegrationTest.java (2)
Learnt from: julian-christl
PR: ls1intum/Artemis#9322
File: src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseStudentParticipationRepository.java:0-0
Timestamp: 2025-03-22T12:51:55.492Z
Learning: In Artemis, an `ExerciseGroup` always has an associated `Exam`, so `exerciseGroup.exam` is never null.
Learnt from: julian-christl
PR: ls1intum/Artemis#9322
File: src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseStudentParticipationRepository.java:170-172
Timestamp: 2025-03-22T12:51:51.144Z
Learning: In Artemis, `exercise.exerciseGroup` may be null, as not all exercises belong to an `ExerciseGroup`.
🧬 Code Definitions (1)
src/test/java/de/tum/cit/aet/artemis/quiz/QuizSubmissionIntegrationTest.java (1)
src/test/java/de/tum/cit/aet/artemis/quiz/util/QuizExerciseFactory.java (1)
  • QuizExerciseFactory (49-784)
⏰ Context from checks skipped due to timeout of 90000ms (7)
  • GitHub Check: Codacy Static Code Analysis
  • GitHub Check: Build and Push Docker Image / Build Docker Image for ls1intum/artemis
  • GitHub Check: Build and Push Docker Image / Build Docker Image for ls1intum/artemis
  • GitHub Check: Build .war artifact
  • GitHub Check: client-tests
  • GitHub Check: server-tests
  • GitHub Check: Analyse
🔇 Additional comments (3)
src/main/java/de/tum/cit/aet/artemis/quiz/dto/participation/StudentQuizParticipationWithSolutionsDTO.java (2)

1-17: Good use of Java records for DTOs and data minimization.

The implementation follows the coding guidelines by using Java records for DTOs, which provides immutability and reduces boilerplate code. The @JsonInclude(JsonInclude.Include.NON_EMPTY) annotation ensures data economy by excluding empty fields from JSON serialization. The record implementation of the StudentQuizParticipationDTO interface adheres to the single responsibility principle.


31-34: Review the ToDo comment about deprecated Results

Since this PR aims to remove QuizView, you should revisit this ToDo comment. If QuizView is being removed as part of this PR, then the comment may need to be updated or the deprecated functionality properly addressed.

Additionally, the code follows best practices by using streams for collection transformation and maintains clean, explicit imports as per the coding guidelines.

src/test/java/de/tum/cit/aet/artemis/quiz/QuizSubmissionIntegrationTest.java (1)

304-345: LGTM! Comprehensive test for quiz participation.

The new test method thoroughly verifies that the appropriate data is sent to students when starting a quiz participation. It correctly ensures that solution data (explanations, correct mappings) isn't exposed during the active phase.

coderabbitai[bot]
coderabbitai bot previously approved these changes Mar 25, 2025
Copy link
Contributor

@florian-glombik florian-glombik left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I did not notice information that should not be passed in the conduct network call during exams for quiz exercises and also checked it for start-participation before, during, and after the quiz conduction.

Tested on ts4

@Hialus Hialus added the maintainer-approved The feature maintainer has approved the PR label Mar 25, 2025
@Hialus Hialus merged commit b2dbd89 into develop Mar 25, 2025
16 of 22 checks passed
@Hialus Hialus deleted the chore/quiz-exercises/dto-migration branch March 25, 2025 15:10
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
assessment Pull requests that affect the corresponding module core Pull requests that affect the corresponding module exercise Pull requests that affect the corresponding module maintainer-approved The feature maintainer has approved the PR programming Pull requests that affect the corresponding module quiz Pull requests that affect the corresponding module ready for review server Pull requests that update Java code. (Added Automatically!) tests
Projects
Status: Merged
Development

Successfully merging this pull request may close these issues.

6 participants