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

General: Migrate exercise group module to signals, inject and standalone #9891

Open
wants to merge 29 commits into
base: develop
Choose a base branch
from

Conversation

coolchock
Copy link
Contributor

@coolchock coolchock commented Nov 27, 2024

Checklist

General

Client

Description

This pull request migrates exercise group module to signals, inject and standalone

Steps for Testing

Prerequisites:

  • 1 Instructor
  1. Log in to Artemis
  2. Navigate to Course Administration -> Exams -> Selected Exam -> Exercise Groups
  3. Create an exercise group which contains programming, modeling, quiz and file upload exercise.
  4. Verify that all rows are displayed properly

Testserver States

Note

These badges show the state of the test servers.
Green = Currently available, Red = Currently locked
Click on the badges to get to the test servers.







Review Progress

Summary by CodeRabbit

Based on the comprehensive summary, here are the updated release notes:

  • Refactoring

    • Updated dependency injection mechanism across multiple components.
    • Transitioned to standalone components with modular imports.
    • Simplified input property handling using Angular's input() function.
  • Testing

    • Updated test suites to utilize modern Angular testing utilities.
    • Streamlined component testing approaches.
    • Removed unnecessary mock components and directives.
  • Module Changes

    • Removed several components related to exercise groups from exam-management.module.ts.
    • Enhanced modularity of exercise group and exam-related components.
  • Performance

    • Improved code structure and reduced boilerplate in component definitions.
    • More direct and type-safe input property management.

The changes primarily focus on modernizing the Angular application's architecture, making components more self-contained and improving overall code quality.

@github-actions github-actions bot added tests client Pull requests that update TypeScript code. (Added Automatically!) labels Nov 27, 2024
BBesrour
BBesrour previously approved these changes Nov 28, 2024
Copy link
Member

@BBesrour BBesrour left a comment

Choose a reason for hiding this comment

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

code lgtm

@coolchock coolchock temporarily deployed to artemis-test3.artemis.cit.tum.de November 28, 2024 19:46 — with GitHub Actions Inactive
@coolchock coolchock marked this pull request as ready for review November 28, 2024 21:52
@coolchock coolchock requested a review from a team as a code owner November 28, 2024 21:52
Copy link

coderabbitai bot commented Nov 28, 2024

Walkthrough

This pull request focuses on refactoring Angular components and services related to exercise groups in the Artemis application. The changes primarily involve modernizing dependency injection by using Angular's inject() function, marking components as standalone, and updating input property handling. The modifications span multiple files across exercise group management, including components for programming, file upload, modeling, and quiz exercises. The refactoring maintains the existing functionality while improving code structure and adhering to newer Angular best practices.

Changes

File Path Change Summary
src/main/webapp/app/exam/manage/exercise-groups/exercise-group-update.component.ts Refactored dependency injection, marked as standalone, simplified input handling
src/main/webapp/app/exam/manage/exercise-groups/exercise-group.service.ts Updated HTTP client injection, removed unused Router dependency
src/main/webapp/app/exam/manage/exercise-groups/exercise-groups.component.ts Modernized dependency injection, marked as standalone
src/main/webapp/app/exam/manage/exercise-groups/*-exercise-cell/*.component.ts Updated input handling, marked as standalone, simplified component structure
src/main/webapp/app/exam/manage/exam-management.module.ts Removed exercise group components from declarations
src/test/javascript/spec/component/exam/manage/exercise-groups/*.spec.ts Updated test setups, replaced HttpClientModule with new providers

Possibly related issues

Possibly related PRs

Suggested labels

ready to merge, refactoring, small

Suggested reviewers

  • JohannesStoehr
  • florian-glombik
  • HawKhiem
  • rabeatwork
  • SimonEntholzer
  • az108

📜 Recent review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 9d5f0a8 and 7e121d9.

📒 Files selected for processing (1)
  • src/main/webapp/app/exam/manage/exam-management.module.ts (0 hunks)
💤 Files with no reviewable changes (1)
  • src/main/webapp/app/exam/manage/exam-management.module.ts

Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR. (Beta)
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai or @coderabbitai title anywhere in the PR title to generate the title automatically.

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link

@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: 8

🧹 Outside diff range and nitpick comments (29)
src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.html (1)

1-3: Consider optimizing multiple exercise() calls.

The exercise() function is called twice in quick succession. Consider storing the result in a variable to avoid potential redundant computations.

Here's a suggested optimization:

-@if (exercise().type === exerciseType.FILE_UPLOAD) {
-    {{ exercise().filePattern || '' }}
+@if ((exerciseData = exercise()).type === exerciseType.FILE_UPLOAD) {
+    {{ exerciseData.filePattern || '' }}
}

Note: This assumes you have declared exerciseData as a property in your component. If not, you might need to add it:

exerciseData?: FileUploadExercise;
src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.html (1)

2-2: Consider enhancing the display text for better UX.

The current implementation only shows a number without context. Consider making it more descriptive for better user experience.

-    {{ exercise().quizQuestions?.length || 0 }}
+    <span>Questions: {{ exercise().quizQuestions?.length || 0 }}</span>
src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.html (1)

1-3: Consider adding null safety checks

The current implementation might be vulnerable to null/undefined values. Consider adding safety checks:

-@if (exercise().type === exerciseType.MODELING) {
+@if (exercise() && exercise().type === exerciseType.MODELING) {
     {{ 'artemisApp.DiagramType.' + exercise().diagramType | artemisTranslate }}
}

Additionally, you might want to consider using the nullish coalescing operator for the diagram type:

-    {{ 'artemisApp.DiagramType.' + exercise().diagramType | artemisTranslate }}
+    {{ 'artemisApp.DiagramType.' + (exercise().diagramType ?? 'UNKNOWN') | artemisTranslate }}
src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.ts (1)

Line range hint 6-10: Consider migrating to a standalone component.

Since this PR aims to modernize the architecture, consider converting this component to a standalone component. This would:

  • Reduce bundle size through better tree-shaking
  • Make dependencies explicit
  • Align with Angular's modern architectural practices

Example migration:

 @Component({
     selector: 'jhi-quiz-exercise-group-cell',
     templateUrl: './quiz-exercise-group-cell.component.html',
     styles: [':host{display: contents}'],
+    standalone: true,
+    imports: []  // Add required imports here
 })
src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.ts (1)

Line range hint 6-10: Consider adding component documentation.

While the component is well-structured, adding a brief JSDoc comment describing its purpose and usage would improve maintainability.

Add documentation above the @component decorator:

+/**
+ * Renders a file upload exercise within an exercise group cell.
+ * Used in the exam management interface to display and manage file upload exercises.
+ */
 @Component({
     selector: 'jhi-file-upload-exercise-group-cell',
     templateUrl: './file-upload-exercise-group-cell.component.html',
     styles: [':host{display: contents}'],
 })
src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group.service.spec.ts (1)

19-36: Enhance test coverage and readability.

While the happy path is well tested, consider these improvements:

  1. Add error case testing
  2. Make the test description more specific
  3. Use constants for test data
  4. Add explicit response handling verification

Here's an improved version:

-    it('should set additional parameters correctly in delete', () => {
+    const TEST_DATA = {
+        courseId: 1,
+        examId: 2,
+        exerciseGroupId: 3,
+    };
+
+    it('should send DELETE request with correct parameters when deleting exercise group', () => {
         const courseId = 1;
         const examId = 2;
         const exerciseGroupId = 3;
         const deleteStudentReposBuildPlans = true;
         const deleteBaseReposBuildPlans = false;

-        service.delete(courseId, examId, exerciseGroupId, deleteStudentReposBuildPlans, deleteBaseReposBuildPlans).subscribe();
+        service
+            .delete(courseId, examId, exerciseGroupId, deleteStudentReposBuildPlans, deleteBaseReposBuildPlans)
+            .subscribe({
+                complete: () => expect(true).toBe(true),
+            });

         const req = httpMock.expectOne(
             `api/courses/${courseId}/exams/${examId}/exercise-groups/${exerciseGroupId}?deleteStudentReposBuildPlans=true&deleteBaseReposBuildPlans=false`,
         );

         expect(req.request.method).toBe('DELETE');
         expect(req.request.params.get('deleteStudentReposBuildPlans')).toBe('true');
         expect(req.request.params.get('deleteBaseReposBuildPlans')).toBe('false');

         req.flush(null);
     });
+
+    it('should handle error when delete request fails', () => {
+        const errorMessage = 'Delete failed';
+        
+        service
+            .delete(TEST_DATA.courseId, TEST_DATA.examId, TEST_DATA.exerciseGroupId, true, false)
+            .subscribe({
+                error: (error) => expect(error.message).toBe(errorMessage),
+            });
+
+        const req = httpMock.expectOne(
+            `api/courses/${TEST_DATA.courseId}/exams/${TEST_DATA.examId}/exercise-groups/${TEST_DATA.exerciseGroupId}?deleteStudentReposBuildPlans=true&deleteBaseReposBuildPlans=false`,
+        );
+
+        req.flush(
+            { message: errorMessage },
+            { status: 404, statusText: 'Not Found' }
+        );
+    });
src/test/javascript/spec/component/exam/manage/exercise-groups/quiz-exercise-group-cell.component.spec.ts (1)

Line range hint 8-46: Enhance test coverage with additional scenarios

The current test suite covers basic functionality but would benefit from additional test cases:

  1. Edge cases (0 questions, undefined exercise)
  2. Error handling scenarios
  3. Component lifecycle tests

Consider adding these test cases:

it('should handle undefined exercise gracefully', () => {
    fixture.componentRef.setInput('exercise', undefined);
    fixture.detectChanges();
    expect(fixture.nativeElement.querySelector('.question-count')).toBeNull();
});

it('should handle quiz with no questions', () => {
    const exercise = new QuizExercise({
        id: 1,
        title: 'Empty Quiz',
        type: ExerciseType.QUIZ,
        quizQuestions: [],
        maxPoints: 0,
        duration: 300
    });
    fixture.componentRef.setInput('exercise', exercise);
    fixture.detectChanges();
    expect(fixture.nativeElement.querySelector('.question-count').textContent).toBe('0');
});

it('should update display when exercise input changes', () => {
    const exercise = new QuizExercise({
        id: 1,
        type: ExerciseType.QUIZ,
        quizQuestions: [{ id: 1 }]
    });
    fixture.componentRef.setInput('exercise', exercise);
    fixture.detectChanges();
    expect(fixture.nativeElement.querySelector('.question-count').textContent).toBe('1');

    exercise.quizQuestions.push({ id: 2 });
    fixture.componentRef.setInput('exercise', exercise);
    fixture.detectChanges();
    expect(fixture.nativeElement.querySelector('.question-count').textContent).toBe('2');
});
src/test/javascript/spec/component/exam/manage/exercise-groups/file-upload-exercise-group-cell.component.spec.ts (2)

Line range hint 25-33: Improve test robustness and type safety

Several improvements can be made to enhance the test quality:

  1. Avoid using as any type casting
  2. Create a complete exercise object
  3. Use more specific expectations

Consider this refactor:

-        const exercise: FileUploadExercise = {
-            id: 1,
-            type: ExerciseType.FILE_UPLOAD,
-            filePattern: '*.pdf',
-        } as any as FileUploadExercise;
+        const exercise = new FileUploadExercise();
+        exercise.id = 1;
+        exercise.type = ExerciseType.FILE_UPLOAD;
+        exercise.filePattern = '*.pdf';
+        exercise.title = 'Test Exercise';
 
         fixture.componentRef.setInput('exercise', exercise);
         fixture.detectChanges();
-        expect(fixture.nativeElement.textContent).toContain(exercise.filePattern);
+        expect(fixture.nativeElement.textContent.trim()).toBe(exercise.filePattern);

Line range hint 8-45: Consider adding more test cases for better coverage

The test suite would benefit from additional test cases to cover:

  1. Undefined/null exercise input
  2. Empty file pattern
  3. Invalid file pattern
  4. Component initialization state

Would you like me to provide example implementations for these additional test cases?

src/test/javascript/spec/component/exam/manage/exercise-groups/modeling-exercise-group-cell.component.spec.ts (3)

Line range hint 1-23: Refactor test setup to use NgMocks and avoid full module imports

The current setup imports the full ArtemisTestModule which violates our coding guidelines. Consider refactoring to use NgMocks and mock only the required dependencies.

Here's the suggested implementation:

-import { ArtemisTestModule } from '../../../../test.module';
+import { MockBuilder, MockRender } from '@ng-mocks';
+import { ModelingExerciseGroupCellComponent } from 'app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component';

 describe('Modeling Exercise Group Cell Component', () => {
     let fixture: ComponentFixture<ModelingExerciseGroupCellComponent>;

     beforeEach(() => {
-        TestBed.configureTestingModule({
-            imports: [ArtemisTestModule],
-            declarations: [ModelingExerciseGroupCellComponent, TranslatePipeMock],
-            providers: [],
-        })
-            .compileComponents()
-            .then(() => {
-                fixture = TestBed.createComponent(ModelingExerciseGroupCellComponent);
-            });
+        return MockBuilder(ModelingExerciseGroupCellComponent)
+            .mock(TranslatePipeMock)
+            .then(() => {
+                fixture = MockRender(ModelingExerciseGroupCellComponent);
+            });
     });

Line range hint 26-34: Improve test case robustness and type safety

The test case has several areas for improvement:

  1. Avoid as any type casting
  2. Use more specific expectations
  3. Avoid string concatenation in expectations

Here's the suggested implementation:

     it('should display diagram type', () => {
-        const exercise: ModelingExercise = {
+        const exercise = new ModelingExercise({
             id: 1,
             type: ExerciseType.MODELING,
             diagramType: UMLDiagramType.ClassDiagram,
-        } as any as ModelingExercise;
+        });

         fixture.componentRef.setInput('exercise', exercise);
         fixture.detectChanges();
-        expect(fixture.nativeElement.textContent).toContain('artemisApp.DiagramType.' + exercise.diagramType);
+        const expectedText = `artemisApp.DiagramType.${UMLDiagramType.ClassDiagram}`;
+        expect(fixture.nativeElement.textContent).toContainEqual(expectedText);
     });

Line range hint 1-46: Good migration to setInput, but test suite needs modernization

The changes to use fixture.componentRef.setInput are a step in the right direction. However, the test suite would benefit from a broader modernization effort to align with current best practices and coding guidelines.

Consider:

  1. Using NgMocks for better isolation
  2. Implementing stronger typing
  3. Using more specific expectations
  4. Adding test cases for edge cases and error scenarios
src/main/webapp/app/exam/manage/exercise-groups/exercise-group-update.component.ts (3)

Line range hint 1-16: Consider enhancing type safety for route parameters

The component looks well-structured, but we could improve type safety for route parameters using ParamMap type.

- import { ActivatedRoute, Router } from '@angular/router';
+ import { ActivatedRoute, ParamMap, Router } from '@angular/router';

Line range hint 34-57: Enhance type safety and error handling

Several improvements could make the code more robust:

  1. Replace type assertions with proper null checks
  2. Add parameter validation
  3. Implement more specific error handling
     ngOnInit(): void {
-        this.courseId = Number(this.route.snapshot.paramMap.get('courseId'));
+        const courseId = this.route.snapshot.paramMap.get('courseId');
+        if (!courseId) {
+            this.alertService.error('error.courseIdMissing');
+            this.previousState();
+            return;
+        }
+        this.courseId = Number(courseId);
     }

     save() {
         this.isSaving = true;
+        if (!this.exam?.id) {
+            this.onSaveError(new HttpErrorResponse({ 
+                error: 'Exam ID is required',
+                status: 400 
+            }));
+            return;
+        }
         if (this.exerciseGroup.id !== undefined) {
-            this.subscribeToSaveResponse(this.exerciseGroupService.update(this.courseId, this.exam.id!, this.exerciseGroup));
+            this.subscribeToSaveResponse(this.exerciseGroupService.update(this.courseId, this.exam.id, this.exerciseGroup));
         } else {
             this.exerciseGroup.exam = this.exam;
-            this.subscribeToSaveResponse(this.exerciseGroupService.create(this.courseId, this.exam.id!, this.exerciseGroup));
+            this.subscribeToSaveResponse(this.exerciseGroupService.create(this.courseId, this.exam.id, this.exerciseGroup));
         }
     }

Line range hint 71-74: Improve error handling specificity

The current error handling could be more informative for users and easier to debug.

     private onSaveError(error: HttpErrorResponse) {
-        onError(this.alertService, error);
+        const errorMessage = error.status === 400 
+            ? 'exerciseGroup.error.validation'
+            : error.status === 403
+            ? 'exerciseGroup.error.forbidden'
+            : 'exerciseGroup.error.unknown';
+        this.alertService.error(errorMessage, error.message, { param: this.exerciseGroup.id });
         this.isSaving = false;
     }
src/main/webapp/app/exam/manage/exercise-groups/exercise-group.service.ts (3)

13-13: Consider making the resource URL configurable

The hardcoded API URL could make it challenging to switch between different environments (development, staging, production). Consider moving this to an environment configuration file.

-    public resourceUrl = 'api/courses';
+    public resourceUrl = environment.apiUrl + '/courses';

Line range hint 20-24: Consider enhancing error handling and memory management

While the HTTP methods are well-implemented, consider these improvements:

  1. Add error handling using catchError operator
  2. Add request timeout using timeoutWith operator
  3. Implement proper cleanup using take/takeUntil operators when subscribing

Example implementation for the create method:

create(courseId: number, examId: number, exerciseGroup: ExerciseGroup): Observable<EntityResponseType> {
    return this.http.post<ExerciseGroup>(
        `${this.resourceUrl}/${courseId}/exams/${examId}/exercise-groups`,
        exerciseGroup,
        { observe: 'response' }
    ).pipe(
        timeoutWith(30000, throwError(() => new Error('Request timed out'))),
        catchError(error => {
            console.error('Error creating exercise group:', error);
            return throwError(() => error);
        })
    );
}

Also applies to: 29-33, 38-41, 49-58, 64-67


Line range hint 6-7: Enhance type safety with more specific response types

Consider creating more specific response types that include potential error states and null checks.

type ApiError = {
    message: string;
    status: number;
};

type EntityResponse<T> = HttpResponse<T | null>;
type EntityResponseType = EntityResponse<ExerciseGroup>;
type EntityArrayResponseType = EntityResponse<ExerciseGroup[]>;
src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.ts (2)

1-1: Consider migrating to standalone component

Since the PR objectives include migration to standalone components, consider updating this component to use the standalone architecture. This would involve:

  • Adding standalone: true to the component decorator
  • Moving imports to the component's imports array
  • Removing the component from its NgModule declarations

41-50: Consider extracting URL creation logic

The buildPlanUrl creation logic is duplicated for both solution and template participations. Consider extracting this into a helper method to improve maintainability and reduce duplication.

+ private updateParticipationBuildPlanUrl(participation: any, projectKey: string, urlTemplate: string): void {
+   if (participation?.buildPlanId) {
+     participation.buildPlanUrl = createBuildPlanUrl(urlTemplate, projectKey, participation.buildPlanId);
+   }
+ }

  ngOnInit(): void {
    this.profileService.getProfileInfo().subscribe((profileInfo) => {
      this.localVCEnabled = profileInfo.activeProfiles.includes(PROFILE_LOCALVC);
      this.onlineIdeEnabled = profileInfo.activeProfiles.includes(PROFILE_THEIA);

      const projectKey = this.exercise()?.projectKey;
      if (projectKey) {
-       const solutionParticipation = this.exercise()?.solutionParticipation;
-       if (solutionParticipation?.buildPlanId) {
-         solutionParticipation.buildPlanUrl = createBuildPlanUrl(profileInfo.buildPlanURLTemplate, projectKey, solutionParticipation.buildPlanId);
-       }
-       const templateParticipation = this.exercise()?.templateParticipation;
-       if (templateParticipation?.buildPlanId) {
-         templateParticipation.buildPlanUrl = createBuildPlanUrl(profileInfo.buildPlanURLTemplate, projectKey, templateParticipation.buildPlanId);
-       }
+       this.updateParticipationBuildPlanUrl(this.exercise()?.solutionParticipation, projectKey, profileInfo.buildPlanURLTemplate);
+       this.updateParticipationBuildPlanUrl(this.exercise()?.templateParticipation, projectKey, profileInfo.buildPlanURLTemplate);
      }
    });
  }
src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.html (1)

10-20: Consider DRY principle for repository link handling

The template and solution repository sections have very similar code structures. Consider extracting this pattern into a reusable template.

+ <!-- Create a new ng-template -->
+ <ng-template #repositoryLink let-type="type" let-uri="uri">
+   @if (!localVCEnabled) {
+     <span>
+       <a [href]="uri" target="_blank">{{type}}</a>
+     </span>
+   } @else {
+     <a [routerLink]="" (click)="downloadRepository(type)">
+       <fa-icon [icon]="faDownload" /> {{type}}
+     </a>
+   }
+ </ng-template>

+ <!-- Use it in both places -->
+ <ng-container 
+   *ngTemplateOutlet="repositoryLink;
+   context: { 
+     type: 'TEMPLATE', 
+     uri: exercise().templateParticipation?.repositoryUri 
+   }">
+ </ng-container>

Also applies to: 30-38

src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group-update.component.spec.ts (1)

Line range hint 69-124: Consider adding tests for edge cases

While the existing test cases cover the main scenarios well, consider adding tests for:

  1. Validation error scenarios
  2. Component initialization failures
  3. Edge cases in exercise group data

This would provide more comprehensive test coverage for the component.

Example test case structure:

it('should handle initialization errors', fakeAsync(() => {
    const error = new Error('Initialization failed');
    route.data = throwError(() => error);
    
    fixture = TestBed.createComponent(ExerciseGroupUpdateComponent);
    fixture.detectChanges();
    tick();
    
    expect(alertServiceStub).toHaveBeenCalledOnce();
}));
src/test/javascript/spec/component/exam/manage/exercise-groups/programming-exercise-group-cell.component.spec.ts (3)

64-64: Good use of setInput, but consider improving test module setup.

While the change to use setInput is correct, there are some improvements that could be made to the test setup:

  1. Consider using MockModule instead of importing the full ArtemisTestModule
  2. Explicitly mock HTTP calls in ProgrammingExerciseService

Example refactor:

// Replace full module import with mock
TestBed.configureTestingModule({
    imports: [NgMocks.mockModule(ArtemisTestModule)],
    // ... rest of the configuration
});

// Explicitly mock HTTP calls
mockProgrammingExerciseService = {
    exportInstructorRepository: jest.fn().mockReturnValue(of(new HttpResponse<Blob>())),
    // ... other methods
};

Line range hint 77-91: Enhance test assertions for better specificity.

While the input setting is correct, the assertions could be more specific according to our testing guidelines:

  1. Use toBeTrue()/toBeFalse() for boolean checks
  2. Use toEqual for exact matches
  3. Consider using toHaveText() for more precise text content matching

Example improvements:

// For the short name test
expect(div).not.toBeNull();
expect(div.nativeElement).toHaveText(exercise.shortName);

// For the repository URL test
expect(span).toBeDefined();
expect(span.nativeElement).toHaveText('Template');
expect(span.nativeElement.getAttribute('href')).toEqual(exercise.templateParticipation!.repositoryUri);

Line range hint 94-107: Improve editor mode flags test structure and translations.

The test could be improved in several ways:

  1. Use translation service properly for text expectations
  2. Reduce duplication in assertions
  3. Make text content checks more precise

Suggested refactor:

it('should display editor mode flags', () => {
    const translateService = TestBed.inject(TranslateService);
    fixture.componentRef.setInput('displayEditorModus', true);
    fixture.detectChanges();

    const testCases = [
        { selector: 'div:first-child', mode: 'offlineIde', value: exercise.allowOfflineIde },
        { selector: 'div:nth-child(2)', mode: 'onlineEditor', value: exercise.allowOnlineEditor },
        { selector: 'div:nth-child(3)', mode: 'onlineIde', value: exercise.allowOnlineIde }
    ];

    testCases.forEach(({ selector, mode, value }) => {
        const div = fixture.debugElement.query(By.css(`div > div > ${selector}`));
        const expectedText = translateService.instant(`artemisApp.programmingExercise.${mode}`) + 
                           translateService.instant(`artemisApp.exercise.${value ? 'yes' : 'no'}`);
        expect(div).not.toBeNull();
        expect(div.nativeElement).toHaveText(expectedText);
    });
});
src/main/webapp/app/exam/manage/exercise-groups/exercise-groups.component.ts (4)

43-51: Consider adding readonly modifier to injected services

While the migration to inject() function is good, consider adding readonly modifier to prevent accidental reassignment of these service references.

-    private route = inject(ActivatedRoute);
+    private readonly route = inject(ActivatedRoute);
-    private exerciseGroupService = inject(ExerciseGroupService);
+    private readonly exerciseGroupService = inject(ExerciseGroupService);
-    exerciseService = inject(ExerciseService);
+    readonly exerciseService = inject(ExerciseService);

Line range hint 89-102: Add subscription management to prevent memory leaks

The forkJoin subscription in ngOnInit should be managed to prevent memory leaks. Consider using the takeUntilDestroyed operator or implementing OnDestroy.

+    private readonly destroy$ = new Subject<void>();
+
     ngOnInit(): void {
         this.courseId = Number(this.route.snapshot.paramMap.get('courseId'));
         this.examId = Number(this.route.snapshot.paramMap.get('examId'));
         // Only take action when a response was received for both requests
         forkJoin([this.loadExerciseGroups(), this.loadLatestIndividualEndDateOfExam()]).subscribe({
             next: ([examRes, examInfoDTO]) => {
                 this.exam = examRes.body!;
                 this.exerciseGroups = this.exam.exerciseGroups;
                 this.course = this.exam.course!;
                 this.latestIndividualEndDate = examInfoDTO ? examInfoDTO.body!.latestIndividualEndDate : undefined;
                 this.setupExerciseGroupToExerciseTypesDict();
             },
             error: (res: HttpErrorResponse) => onError(this.alertService, res),
-        });
+        }).pipe(takeUntil(this.destroy$));
+
+    ngOnDestroy(): void {
+        this.destroy$.next();
+        this.destroy$.complete();
+    }

Line range hint 89-102: Consider migrating to Angular signals for better state management

As mentioned in the PR objectives about migrating to signals, this component's state management could benefit from using signals for exerciseGroups, exam, and other state properties.

Example implementation:

export class ExerciseGroupsComponent implements OnInit {
    private readonly exerciseGroups = signal<ExerciseGroup[]>([]);
    private readonly exam = signal<Exam | null>(null);
    
    // Computed values
    readonly hasExerciseGroups = computed(() => this.exerciseGroups().length > 0);
    
    ngOnInit(): void {
        forkJoin([/*...*/]).subscribe({
            next: ([examRes, examInfoDTO]) => {
                this.exam.set(examRes.body!);
                this.exerciseGroups.set(examRes.body!.exerciseGroups);
                // ...
            }
        });
    }
}

Line range hint 196-208: Improve error handling in deleteExerciseGroup

The error handling in deleteExerciseGroup updates UI state before the server request completes. Consider moving the state updates to the success callback.

     deleteExerciseGroup(exerciseGroupId: number, event: { [key: string]: boolean }) {
         this.exerciseGroupService.delete(this.courseId, this.examId, exerciseGroupId, event.deleteStudentReposBuildPlans, event.deleteBaseReposBuildPlans).subscribe({
             next: () => {
+                this.exerciseGroups = this.exerciseGroups!.filter((exerciseGroup) => exerciseGroup.id !== exerciseGroupId);
+                this.exerciseGroupToExerciseTypesDict.delete(exerciseGroupId);
                 this.eventManager.broadcast({
                     name: 'exerciseGroupOverviewModification',
                     content: 'Deleted an exercise group',
                 });
                 this.dialogErrorSource.next('');
-                this.exerciseGroups = this.exerciseGroups!.filter((exerciseGroup) => exerciseGroup.id !== exerciseGroupId);
-                this.exerciseGroupToExerciseTypesDict.delete(exerciseGroupId);
             },
             error: (error: HttpErrorResponse) => this.dialogErrorSource.next(error.message),
         });
     }
📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 4eaba4e and 65b0e6b.

📒 Files selected for processing (17)
  • src/main/webapp/app/exam/manage/exercise-groups/exercise-group-update.component.ts (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/exercise-group.service.ts (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/exercise-groups.component.ts (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.html (1 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.ts (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.html (1 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.ts (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.html (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.ts (3 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.html (1 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.ts (2 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group-update.component.spec.ts (2 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group.service.spec.ts (1 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/file-upload-exercise-group-cell.component.spec.ts (1 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/modeling-exercise-group-cell.component.spec.ts (1 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/programming-exercise-group-cell.component.spec.ts (3 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/quiz-exercise-group-cell.component.spec.ts (1 hunks)
🧰 Additional context used
📓 Path-based instructions (17)
src/main/webapp/app/exam/manage/exercise-groups/exercise-group-update.component.ts (1)

Pattern src/main/webapp/**/*.ts: angular_style:https://angular.io/guide/styleguide;methods_in_html:false;lazy_loading:true;code_reuse:true;tests:meaningful;types:PascalCase;enums:PascalCase;funcs:camelCase;props:camelCase;no_priv_prefix:true;strings:single_quotes;localize:true;btns:functionality;links:navigation;icons_text:newline;labels:associate;code_style:arrow_funcs,curly_braces,open_braces_same_line,indent_4;memory_leak_prevention:true;routes:naming_schema;chart_framework:ngx-charts;responsive_layout:true

src/main/webapp/app/exam/manage/exercise-groups/exercise-group.service.ts (1)

Pattern src/main/webapp/**/*.ts: angular_style:https://angular.io/guide/styleguide;methods_in_html:false;lazy_loading:true;code_reuse:true;tests:meaningful;types:PascalCase;enums:PascalCase;funcs:camelCase;props:camelCase;no_priv_prefix:true;strings:single_quotes;localize:true;btns:functionality;links:navigation;icons_text:newline;labels:associate;code_style:arrow_funcs,curly_braces,open_braces_same_line,indent_4;memory_leak_prevention:true;routes:naming_schema;chart_framework:ngx-charts;responsive_layout:true

src/main/webapp/app/exam/manage/exercise-groups/exercise-groups.component.ts (1)

Pattern src/main/webapp/**/*.ts: angular_style:https://angular.io/guide/styleguide;methods_in_html:false;lazy_loading:true;code_reuse:true;tests:meaningful;types:PascalCase;enums:PascalCase;funcs:camelCase;props:camelCase;no_priv_prefix:true;strings:single_quotes;localize:true;btns:functionality;links:navigation;icons_text:newline;labels:associate;code_style:arrow_funcs,curly_braces,open_braces_same_line,indent_4;memory_leak_prevention:true;routes:naming_schema;chart_framework:ngx-charts;responsive_layout:true

src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.html (1)

Pattern src/main/webapp/**/*.html: @if and @for are new and valid Angular syntax replacing *ngIf and *ngFor. They should always be used over the old style.

src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.ts (1)

Pattern src/main/webapp/**/*.ts: angular_style:https://angular.io/guide/styleguide;methods_in_html:false;lazy_loading:true;code_reuse:true;tests:meaningful;types:PascalCase;enums:PascalCase;funcs:camelCase;props:camelCase;no_priv_prefix:true;strings:single_quotes;localize:true;btns:functionality;links:navigation;icons_text:newline;labels:associate;code_style:arrow_funcs,curly_braces,open_braces_same_line,indent_4;memory_leak_prevention:true;routes:naming_schema;chart_framework:ngx-charts;responsive_layout:true

src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.html (1)

Pattern src/main/webapp/**/*.html: @if and @for are new and valid Angular syntax replacing *ngIf and *ngFor. They should always be used over the old style.

src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.ts (1)

Pattern src/main/webapp/**/*.ts: angular_style:https://angular.io/guide/styleguide;methods_in_html:false;lazy_loading:true;code_reuse:true;tests:meaningful;types:PascalCase;enums:PascalCase;funcs:camelCase;props:camelCase;no_priv_prefix:true;strings:single_quotes;localize:true;btns:functionality;links:navigation;icons_text:newline;labels:associate;code_style:arrow_funcs,curly_braces,open_braces_same_line,indent_4;memory_leak_prevention:true;routes:naming_schema;chart_framework:ngx-charts;responsive_layout:true

src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.html (1)

Pattern src/main/webapp/**/*.html: @if and @for are new and valid Angular syntax replacing *ngIf and *ngFor. They should always be used over the old style.

src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.ts (1)

Pattern src/main/webapp/**/*.ts: angular_style:https://angular.io/guide/styleguide;methods_in_html:false;lazy_loading:true;code_reuse:true;tests:meaningful;types:PascalCase;enums:PascalCase;funcs:camelCase;props:camelCase;no_priv_prefix:true;strings:single_quotes;localize:true;btns:functionality;links:navigation;icons_text:newline;labels:associate;code_style:arrow_funcs,curly_braces,open_braces_same_line,indent_4;memory_leak_prevention:true;routes:naming_schema;chart_framework:ngx-charts;responsive_layout:true

src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.html (1)

Pattern src/main/webapp/**/*.html: @if and @for are new and valid Angular syntax replacing *ngIf and *ngFor. They should always be used over the old style.

src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.ts (1)

Pattern src/main/webapp/**/*.ts: angular_style:https://angular.io/guide/styleguide;methods_in_html:false;lazy_loading:true;code_reuse:true;tests:meaningful;types:PascalCase;enums:PascalCase;funcs:camelCase;props:camelCase;no_priv_prefix:true;strings:single_quotes;localize:true;btns:functionality;links:navigation;icons_text:newline;labels:associate;code_style:arrow_funcs,curly_braces,open_braces_same_line,indent_4;memory_leak_prevention:true;routes:naming_schema;chart_framework:ngx-charts;responsive_layout:true

src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group-update.component.spec.ts (1)

Pattern src/test/javascript/spec/**/*.ts: jest: true; mock: NgMocks; bad_practices: avoid_full_module_import; perf_improvements: mock_irrelevant_deps; service_testing: mock_http_for_logic; no_schema: avoid_NO_ERRORS_SCHEMA; expectation_specificity: true; solutions: {boolean: toBeTrue/False, reference: toBe, existence: toBeNull/NotNull, undefined: toBeUndefined, class_obj: toContainEntries/toEqual, spy_calls: {not_called: not.toHaveBeenCalled, once: toHaveBeenCalledOnce, with_value: toHaveBeenCalledWith|toHaveBeenCalledExactlyOnceWith}}

src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group.service.spec.ts (1)

Pattern src/test/javascript/spec/**/*.ts: jest: true; mock: NgMocks; bad_practices: avoid_full_module_import; perf_improvements: mock_irrelevant_deps; service_testing: mock_http_for_logic; no_schema: avoid_NO_ERRORS_SCHEMA; expectation_specificity: true; solutions: {boolean: toBeTrue/False, reference: toBe, existence: toBeNull/NotNull, undefined: toBeUndefined, class_obj: toContainEntries/toEqual, spy_calls: {not_called: not.toHaveBeenCalled, once: toHaveBeenCalledOnce, with_value: toHaveBeenCalledWith|toHaveBeenCalledExactlyOnceWith}}

src/test/javascript/spec/component/exam/manage/exercise-groups/file-upload-exercise-group-cell.component.spec.ts (1)

Pattern src/test/javascript/spec/**/*.ts: jest: true; mock: NgMocks; bad_practices: avoid_full_module_import; perf_improvements: mock_irrelevant_deps; service_testing: mock_http_for_logic; no_schema: avoid_NO_ERRORS_SCHEMA; expectation_specificity: true; solutions: {boolean: toBeTrue/False, reference: toBe, existence: toBeNull/NotNull, undefined: toBeUndefined, class_obj: toContainEntries/toEqual, spy_calls: {not_called: not.toHaveBeenCalled, once: toHaveBeenCalledOnce, with_value: toHaveBeenCalledWith|toHaveBeenCalledExactlyOnceWith}}

src/test/javascript/spec/component/exam/manage/exercise-groups/modeling-exercise-group-cell.component.spec.ts (1)

Pattern src/test/javascript/spec/**/*.ts: jest: true; mock: NgMocks; bad_practices: avoid_full_module_import; perf_improvements: mock_irrelevant_deps; service_testing: mock_http_for_logic; no_schema: avoid_NO_ERRORS_SCHEMA; expectation_specificity: true; solutions: {boolean: toBeTrue/False, reference: toBe, existence: toBeNull/NotNull, undefined: toBeUndefined, class_obj: toContainEntries/toEqual, spy_calls: {not_called: not.toHaveBeenCalled, once: toHaveBeenCalledOnce, with_value: toHaveBeenCalledWith|toHaveBeenCalledExactlyOnceWith}}

src/test/javascript/spec/component/exam/manage/exercise-groups/programming-exercise-group-cell.component.spec.ts (1)

Pattern src/test/javascript/spec/**/*.ts: jest: true; mock: NgMocks; bad_practices: avoid_full_module_import; perf_improvements: mock_irrelevant_deps; service_testing: mock_http_for_logic; no_schema: avoid_NO_ERRORS_SCHEMA; expectation_specificity: true; solutions: {boolean: toBeTrue/False, reference: toBe, existence: toBeNull/NotNull, undefined: toBeUndefined, class_obj: toContainEntries/toEqual, spy_calls: {not_called: not.toHaveBeenCalled, once: toHaveBeenCalledOnce, with_value: toHaveBeenCalledWith|toHaveBeenCalledExactlyOnceWith}}

src/test/javascript/spec/component/exam/manage/exercise-groups/quiz-exercise-group-cell.component.spec.ts (1)

Pattern src/test/javascript/spec/**/*.ts: jest: true; mock: NgMocks; bad_practices: avoid_full_module_import; perf_improvements: mock_irrelevant_deps; service_testing: mock_http_for_logic; no_schema: avoid_NO_ERRORS_SCHEMA; expectation_specificity: true; solutions: {boolean: toBeTrue/False, reference: toBe, existence: toBeNull/NotNull, undefined: toBeUndefined, class_obj: toContainEntries/toEqual, spy_calls: {not_called: not.toHaveBeenCalled, once: toHaveBeenCalledOnce, with_value: toHaveBeenCalledWith|toHaveBeenCalledExactlyOnceWith}}

🔇 Additional comments (24)
src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.html (1)

1-1: LGTM! Correct usage of new Angular control flow syntax.

The @if syntax follows the new Angular control flow guidelines, replacing the older *ngIf directive.

src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.html (1)

1-1: LGTM! Correct usage of new @if syntax.

The code correctly uses the new Angular @if syntax as per guidelines, replacing the older *ngIf directive.

src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.html (1)

1-1: LGTM! Correctly using new @if syntax

The code follows the new Angular syntax guidelines by using @if instead of *ngIf.

src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.ts (2)

1-1: LGTM! Good migration to the new input API.

The change from Input decorator to the new input function aligns with Angular's modern practices for handling component inputs.


12-12: Well-structured input declaration with type safety.

The migration to input.required<QuizExercise>() brings several benefits:

  • Ensures type safety with explicit QuizExercise type
  • Makes the input requirement explicit at compile time
  • Reduces boilerplate compared to the traditional decorator approach

Let's verify the template usage of this input:

✅ Verification successful

Input usage verified and correctly implemented in template

The template correctly uses the new input syntax by calling exercise() as a function in both instances:

  • Condition check: exercise().type === exerciseType.QUIZ
  • Property access: exercise().quizQuestions?.length
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if the template correctly uses the new input syntax
# Expect: Template should call exercise() as a function rather than accessing it as a property

rg -t html "exercise\(" "src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell"

Length of output: 400

src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.ts (2)

1-1: LGTM! Modern dependency injection approach

The change from @Input decorator to the new input function aligns with Angular's modern dependency injection patterns.


12-12: LGTM! Verify template updates

The new input.required<ModelingExercise>() implementation provides better type safety and removes the need for explicit type casting.

Let's verify that all template usages have been updated:

✅ Verification successful

Let me check the template file specifically to verify if it's using the new exercise() syntax.


Template updates are correctly implemented

The template is properly using the new exercise() signal syntax, and the remaining modelingExercise references found are just translation keys in a different component, which is expected and correct.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if the template file has been updated to use exercise() instead of modelingExercise
# and ensure no other files are still referencing modelingExercise

# Check the component's template file
rg -A 2 "modelingExercise" src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/

# Check for any remaining references in other files
rg "modelingExercise" src/main/webapp/app/exam/manage/exercise-groups/

Length of output: 789


Script:

#!/bin/bash
# Check the component's template file specifically
fd modeling-exercise-group-cell.component.html -t f --exec cat {}

Length of output: 201

src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.ts (2)

12-12: Verify template updates for the new input syntax.

The migration to input.required looks good, but ensure that the corresponding template file has been updated to use exercise() instead of direct property access.

Let's verify the template updates:

✅ Verification successful

Template is correctly using the new input syntax

The template is already using the correct exercise() function call syntax in both places:

  • exercise().type
  • exercise().filePattern
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if the template has been properly updated
# Expected: Template should use exercise() function calls

# Check the template file for any remaining direct property access
rg "fileUploadExercise" "src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.html"

Length of output: 144


Script:

#!/bin/bash
# Let's first check if the template file exists and see its content
fd "file-upload-exercise-group-cell.component.html" --type f --exec cat {}

# As a fallback, let's check for any references to the exercise property in the template
rg "exercise" "src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.html"

Length of output: 397


1-1: LGTM! Verify other components in the exercise group module.

The migration from Input decorator to input function aligns with Angular's new input system using signals, which can improve performance through better change detection.

Let's verify consistent migration across related components:

✅ Verification successful

Migration to input() signal is consistently applied across exercise group components

The verification shows that all exercise cell components in the exercise groups module (file-upload, modeling, programming, and quiz) have been properly migrated to use the new input() signal from @angular/core. Let's check the remaining components to ensure complete migration:


Migration to input() signal is correctly implemented

Based on the verification results, all exercise cell components (file-upload, modeling, programming, and quiz) have been properly migrated to use the new input() signal from @angular/core. No legacy @Input decorators were found in the module, confirming consistent implementation of the signals architecture migration.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if other exercise group components still use the old Input decorator
# Expected: No components should use the old @Input decorator

# Search for old @Input usage in exercise group components
rg -l "@Input\(\)" "src/main/webapp/app/exam/manage/exercise-groups"

Length of output: 68


Script:

#!/bin/bash
# Let's try a broader search to find any Input usage, including variations
rg -i "input.*from.*@angular/core|@Input|input\(" "src/main/webapp/app/exam/manage/exercise-groups" -A 2

Length of output: 3171


Script:

#!/bin/bash
# Let's also check for component files that might need migration
fd ".component.ts$" "src/main/webapp/app/exam/manage/exercise-groups"

Length of output: 685


Script:

#!/bin/bash
# Check the specific pattern of input usage from @angular/core
ast-grep --pattern 'import { $$$, input } from "@angular/core"'

Length of output: 65


Script:

#!/bin/bash
# Check if exercise-group-update and exercise-groups components still use @Input
rg -i "@Input|input.*from.*@angular/core" "src/main/webapp/app/exam/manage/exercise-groups/exercise-group" -A 2

Length of output: 214


Script:

#!/bin/bash
# Check the remaining components with correct paths
rg -i "@Input|input.*from.*@angular/core" "src/main/webapp/app/exam/manage/exercise-groups/exercise-group.*\.component\.ts" -A 2

Length of output: 250


Script:

#!/bin/bash
# Double check for any remaining @Input decorators in the module
rg "@Input" "src/main/webapp/app/exam/manage/exercise-groups/" --type ts

Length of output: 70

src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group.service.spec.ts (1)

1-4: LGTM! Imports follow best practices.

The imports are specific and include the necessary testing utilities for HTTP testing in Angular.

src/main/webapp/app/exam/manage/exercise-groups/exercise-group-update.component.ts (1)

17-20: 🛠️ Refactor suggestion

Verify subscription cleanup to prevent memory leaks

The inject pattern is correctly implemented. However, since we're subscribing to route.data in ngOnInit, we should ensure proper cleanup.

Consider implementing OnDestroy and managing subscriptions:

-export class ExerciseGroupUpdateComponent implements OnInit {
+export class ExerciseGroupUpdateComponent implements OnInit, OnDestroy {
+    private destroy$ = new Subject<void>();
+
     private route = inject(ActivatedRoute);
     private router = inject(Router);
     private exerciseGroupService = inject(ExerciseGroupService);
     private alertService = inject(AlertService);

     ngOnInit(): void {
         this.courseId = Number(this.route.snapshot.paramMap.get('courseId'));
-        this.route.data.subscribe(({ exam, exerciseGroup }) => {
+        this.route.data.pipe(
+            takeUntil(this.destroy$)
+        ).subscribe(({ exam, exerciseGroup }) => {
             this.exam = exam;
             this.exerciseGroup = exerciseGroup;
         });
     }
+
+    ngOnDestroy(): void {
+        this.destroy$.next();
+        this.destroy$.complete();
+    }
src/main/webapp/app/exam/manage/exercise-groups/exercise-group.service.ts (1)

1-2: LGTM! Good modernization of dependency injection

The transition from constructor-based DI to the inject() function follows modern Angular best practices and reduces boilerplate code. The removal of unused Router dependency also helps maintain a clean codebase.

Also applies to: 11-11

src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.ts (3)

19-21: LGTM! Clean migration to inject()

The migration from constructor injection to the new inject() function follows Angular's modern best practices and reduces boilerplate code.


28-32: LGTM! Proper input() usage

The migration from @input() decorators to input() function is well implemented with:

  • Correct default values for boolean flags
  • Proper type safety using input.required for the exercise property

Line range hint 63-71: LGTM! Well-documented and properly implemented

The downloadRepository method:

  • Correctly uses the new exercise() accessor
  • Includes proper error handling
  • Has clear documentation about repository type limitations
src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.html (3)

1-6: LGTM! Proper usage of new Angular syntax

The changes correctly implement the new @if syntax and handle null safety appropriately.


Line range hint 84-101: LGTM! Well-structured editor modes section

The section properly handles different IDE modes with appropriate translations and feature flags.


21-27: Verify status component integration

The programming exercise instructor status component is used with different participations. Let's verify its integration.

Also applies to: 39-45

✅ Verification successful

Based on the search results, I can now generate the final response as we have sufficient information about the component's integration:

Programming Exercise Instructor Status component is properly integrated

The component is consistently used across the codebase with the same input pattern for different participation types:

  • In programming exercise management view for template and solution participations
  • In programming test status details
  • In exam exercise groups for template and solution participations (current usage)

All usages properly provide the required inputs: participationType, participation, and exercise.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for the component definition and its usage
ast-grep --pattern 'class ProgrammingExerciseInstructorStatus {
  $$$
}'

# Check for other usages of the component
rg "programming-exercise-instructor-status" -A 5

Length of output: 10035

src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group-update.component.spec.ts (3)

1-2: LGTM! Modern HTTP testing configuration

The switch to provideHttpClient() and provideHttpClientTesting() follows Angular's recommended testing practices, providing better control over HTTP testing and aligning with the modernization objectives.

Also applies to: 50-51


Line range hint 47-58: LGTM! Well-structured test module configuration

The test module setup follows best practices with:

  • Proper component and directive mocking
  • Clear separation of concerns
  • Avoiding NO_ERRORS_SCHEMA as per guidelines

Line range hint 1-124: Verify signal testing coverage

Given that this PR involves migrating to signals, please ensure that signal-related functionality is properly tested.

src/main/webapp/app/exam/manage/exercise-groups/exercise-groups.component.ts (3)

Line range hint 1-32: LGTM! Well-organized imports following Angular best practices

The imports are properly organized and follow tree-shaking friendly patterns.


Line range hint 34-42: LGTM! Component follows Angular architectural patterns

The component structure follows Angular best practices with proper separation of concerns.


45-45: Review public access modifier for exerciseService

The exerciseService is marked as public while other services are private. Please verify if this service needs to be accessed from the template, otherwise consider making it private for better encapsulation.

✅ Verification successful

Keep exerciseService public as it's used in the template

The exerciseService needs to remain public since it's being accessed directly in the template to call isIncludedInScore() method on an exercise.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check if exerciseService is used in the template
rg "exerciseService\." src/main/webapp/app/exam/manage/exercise-groups/exercise-groups.component.html

Length of output: 219

@coolchock coolchock force-pushed the chore/exercise-groups-client-migration branch from 70c5db4 to 5d48e0c Compare January 2, 2025 18:23
Copy link

@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

🔭 Outside diff range comments (1)
src/test/javascript/spec/component/exam/manage/exercise-groups/file-upload-exercise-group-cell.component.spec.ts (1)

Line range hint 24-32: Improve test object creation and assertions.

Several improvements can be made to enhance type safety and test clarity:

  1. Avoid unsafe type assertions
  2. Create a complete test object
  3. Use more specific assertions
-    const exercise: FileUploadExercise = {
-        id: 1,
-        type: ExerciseType.FILE_UPLOAD,
-        filePattern: '*.pdf',
-    } as any as FileUploadExercise;
+    const exercise = new FileUploadExercise();
+    exercise.id = 1;
+    exercise.type = ExerciseType.FILE_UPLOAD;
+    exercise.filePattern = '*.pdf';
+    exercise.title = 'Test Exercise';

     fixture.componentRef.setInput('exercise', exercise);
     fixture.detectChanges();
-    expect(fixture.nativeElement.textContent).toContain(exercise.filePattern);
+    expect(fixture.nativeElement.textContent.trim()).toEqual(expect.stringContaining(exercise.filePattern));
♻️ Duplicate comments (1)
src/test/javascript/spec/component/exam/manage/exercise-groups/file-upload-exercise-group-cell.component.spec.ts (1)

36-44: 🛠️ Refactor suggestion

Previous review feedback remains unaddressed.

The test still contains the issues highlighted in the previous review regarding type safety and test clarity.

Additional note: Per the coding guidelines for test files, use more specific assertions:

     fixture.componentRef.setInput('exercise', exercise);
     fixture.detectChanges();
-    expect(fixture.nativeElement.textContent).toBe('');
+    expect(fixture.nativeElement.textContent.trim()).toEqual('');
🧹 Nitpick comments (1)
src/test/javascript/spec/component/exam/manage/exercise-groups/file-upload-exercise-group-cell.component.spec.ts (1)

Line range hint 1-46: Consider adding test cases for improved coverage.

The test suite would benefit from additional test cases:

  1. Exercise with undefined/null file pattern
  2. Exercise with empty file pattern
  3. Error handling scenarios

Example additional test:

it('should handle undefined file pattern gracefully', () => {
    const exercise = new FileUploadExercise();
    exercise.id = 1;
    exercise.type = ExerciseType.FILE_UPLOAD;
    exercise.title = 'Test Exercise';
    // file pattern intentionally left undefined

    fixture.componentRef.setInput('exercise', exercise);
    fixture.detectChanges();
    expect(fixture.nativeElement.textContent.trim()).toEqual('');
});
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 70c5db4 and 5d48e0c.

📒 Files selected for processing (21)
  • src/main/webapp/app/exam/manage/exam-management.module.ts (1 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/exercise-group-update.component.ts (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/exercise-group.service.ts (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/exercise-groups.component.ts (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.html (1 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.ts (1 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.html (1 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.ts (1 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.html (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.ts (2 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.html (1 hunks)
  • src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.ts (1 hunks)
  • src/main/webapp/app/exercises/shared/exam-exercise-row-buttons/exam-exercise-row-buttons.component.ts (1 hunks)
  • src/test/javascript/spec/component/exam/manage/exam-exercise-row-buttons.component.spec.ts (0 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group-update.component.spec.ts (2 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group.service.spec.ts (1 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-groups.component.spec.ts (4 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/file-upload-exercise-group-cell.component.spec.ts (2 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/modeling-exercise-group-cell.component.spec.ts (2 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/programming-exercise-group-cell.component.spec.ts (4 hunks)
  • src/test/javascript/spec/component/exam/manage/exercise-groups/quiz-exercise-group-cell.component.spec.ts (1 hunks)
💤 Files with no reviewable changes (1)
  • src/test/javascript/spec/component/exam/manage/exam-exercise-row-buttons.component.spec.ts
🚧 Files skipped from review as they are similar to previous changes (18)
  • src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.html
  • src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.html
  • src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.html
  • src/main/webapp/app/exam/manage/exercise-groups/exercise-group-update.component.ts
  • src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group.service.spec.ts
  • src/test/javascript/spec/component/exam/manage/exercise-groups/quiz-exercise-group-cell.component.spec.ts
  • src/test/javascript/spec/component/exam/manage/exercise-groups/modeling-exercise-group-cell.component.spec.ts
  • src/main/webapp/app/exercises/shared/exam-exercise-row-buttons/exam-exercise-row-buttons.component.ts
  • src/main/webapp/app/exam/manage/exercise-groups/file-upload-exercise-cell/file-upload-exercise-group-cell.component.ts
  • src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-groups.component.spec.ts
  • src/test/javascript/spec/component/exam/manage/exercise-groups/exercise-group-update.component.spec.ts
  • src/main/webapp/app/exam/manage/exercise-groups/exercise-group.service.ts
  • src/main/webapp/app/exam/manage/exercise-groups/modeling-exercise-cell/modeling-exercise-group-cell.component.ts
  • src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.html
  • src/main/webapp/app/exam/manage/exercise-groups/quiz-exercise-cell/quiz-exercise-group-cell.component.ts
  • src/test/javascript/spec/component/exam/manage/exercise-groups/programming-exercise-group-cell.component.spec.ts
  • src/main/webapp/app/exam/manage/exercise-groups/exercise-groups.component.ts
  • src/main/webapp/app/exam/manage/exercise-groups/programming-exercise-cell/programming-exercise-group-cell.component.ts
🧰 Additional context used
📓 Path-based instructions (2)
src/test/javascript/spec/component/exam/manage/exercise-groups/file-upload-exercise-group-cell.component.spec.ts (1)

Pattern src/test/javascript/spec/**/*.ts: jest: true; mock: NgMocks; bad_practices: avoid_full_module_import; perf_improvements: mock_irrelevant_deps; service_testing: mock_http_for_logic; no_schema: avoid_NO_ERRORS_SCHEMA; expectation_specificity: true; solutions: {boolean: toBeTrue/False, reference: toBe, existence: toBeNull/NotNull, undefined: toBeUndefined, class_obj: toContainEntries/toEqual, spy_calls: {not_called: not.toHaveBeenCalled, once: toHaveBeenCalledOnce, with_value: toHaveBeenCalledWith|toHaveBeenCalledExactlyOnceWith}}

src/main/webapp/app/exam/manage/exam-management.module.ts (1)

Pattern src/main/webapp/**/*.ts: angular_style:https://angular.io/guide/styleguide;methods_in_html:false;lazy_loading:true;code_reuse:true;tests:meaningful;types:PascalCase;enums:PascalCase;funcs:camelCase;props:camelCase;no_priv_prefix:true;strings:single_quotes;localize:true;btns:functionality;links:navigation;icons_text:newline;labels:associate;code_style:arrow_funcs,curly_braces,open_braces_same_line,indent_4;memory_leak_prevention:true;routes:naming_schema;chart_framework:ngx-charts;responsive_layout:true

🔇 Additional comments (3)
src/test/javascript/spec/component/exam/manage/exercise-groups/file-upload-exercise-group-cell.component.spec.ts (1)

14-14: LGTM! Declarations updated for standalone component.

The removal of FileUploadExerciseGroupCellComponent from declarations is consistent with the migration to standalone components.

src/main/webapp/app/exam/manage/exam-management.module.ts (2)

133-133: Verify the necessity of exporting shared modules.

The addition of ArtemisSharedModule and ArtemisSharedComponentModule to exports might create circular dependencies if these modules are already imported elsewhere. Consider whether these exports are necessary or if they can be imported directly by the modules that need them.

Run the following script to check for potential circular dependencies:

✅ Verification successful

Exporting shared modules is safe in this context

Based on the analysis of the imports across the codebase:

  1. The ArtemisExamManagementModule is only imported lazily via loadChildren in:

    • src/main/webapp/app/course/manage/course-management.route.ts
    • src/main/webapp/app/app-routing.module.ts
  2. Both ArtemisSharedModule and ArtemisSharedComponentModule are widely used across the application and are already imported directly by many feature modules.

  3. No circular dependencies are introduced because:

    • The exam management module is loaded lazily
    • The shared modules are designed to be imported by feature modules
    • No evidence of the shared modules importing the exam management module

The exports of shared modules in the exam management module are safe and follow Angular's best practices for module organization.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check for potential circular dependencies with shared modules

# Search for imports of ArtemisExamManagementModule
echo "Modules importing ArtemisExamManagementModule:"
rg "import.*ArtemisExamManagementModule" -A 5

# Search for imports of shared modules in those files
echo "\nShared module imports in dependent modules:"
rg "import.*ArtemisSharedModule|import.*ArtemisSharedComponentModule" -A 5

Length of output: 68580


Line range hint 1-134: Verify the migration of exercise group components.

The removal of exercise group components from declarations suggests they've been converted to standalone components. This aligns with the PR objective of migrating to signals and standalone architecture.

Run the following script to verify the proper migration of removed components:

✅ Verification successful

Exercise group components have been properly migrated to standalone components

The verification confirms that the exercise group components have been successfully migrated to standalone architecture:

  • ExerciseGroupsComponent is marked as standalone with proper imports
  • All related cell components are also standalone:
    • ProgrammingExerciseGroupCellComponent
    • QuizExerciseGroupCellComponent
    • ModelingExerciseGroupCellComponent
    • FileUploadExerciseGroupCellComponent

The components are properly configured with the required imports and dependencies for standalone operation.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify that removed components are properly converted to standalone

# Search for the removed components and verify they are marked as standalone
echo "Checking standalone configuration in exercise group components:"
ast-grep --pattern 'Component({
  $$$
  standalone: true,
  $$$
})'

# Check for proper dependency injection using inject()
echo "\nChecking inject() usage in exercise group components:"
rg "inject\(" -A 5

Length of output: 66139

coderabbitai[bot]
coderabbitai bot previously approved these changes Jan 2, 2025
coderabbitai[bot]
coderabbitai bot previously approved these changes Jan 2, 2025
ole-ve
ole-ve previously approved these changes Jan 3, 2025
Copy link
Contributor

@ole-ve ole-ve left a comment

Choose a reason for hiding this comment

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

changes look good, just one smaller question regarding exporting the ArtemisSharedModule

src/main/webapp/app/exam/manage/exam-management.module.ts Outdated Show resolved Hide resolved
@coolchock coolchock dismissed stale reviews from ole-ve and coderabbitai[bot] via 7e121d9 January 3, 2025 17:28
Copy link
Contributor

@ole-ve ole-ve 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 👍

Copy link

@Cathy0123456789 Cathy0123456789 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 described.
image
image

Copy link

@vinceclifford vinceclifford 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 TS5, works as expected

Copy link

There hasn't been any activity on this pull request recently. Therefore, this pull request has been automatically marked as stale and will be closed if no further activity occurs within seven days. Thank you for your contributions.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
client Pull requests that update TypeScript code. (Added Automatically!) tests
Projects
Status: Ready For Review
Development

Successfully merging this pull request may close these issues.

8 participants