From 49a91a46da4d38e21ea1d92cadcb4822c75dcbc6 Mon Sep 17 00:00:00 2001 From: Benjamin Schmitz <66966223+bensofficial@users.noreply.github.com> Date: Thu, 2 Jan 2025 19:38:33 +0100 Subject: [PATCH 01/34] Development: Enable gzip compression for nginx (#10092) --- docker/nginx/nginx.conf | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/docker/nginx/nginx.conf b/docker/nginx/nginx.conf index a825fdf6a98f..f7c2dde8b0d8 100644 --- a/docker/nginx/nginx.conf +++ b/docker/nginx/nginx.conf @@ -32,7 +32,8 @@ http { keepalive_timeout 65; - #gzip on; + gzip on; + gzip_types text/plain application/javascript application/x-javascript text/javascript text/xml text/css; # specific Artemis value from the Artemis ansible collection server_names_hash_bucket_size 256; From b26821cb17d91fa9e04930d4ac4fa49e00f4be10 Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Mon, 30 Dec 2024 13:48:53 +0100 Subject: [PATCH 02/34] Development: Improve client config --- LICENSE | 2 +- angular.json | 2 +- docs/conf.py | 2 +- gradle/test.gradle | 3 +- package-lock.json | 227 ++++++++++++++++++ package.json | 1 + .../apollon-diagram-detail.component.ts | 6 +- .../conversation-add-users-form.component.ts | 7 +- .../lecture-unit/lecture-unit.component.ts | 19 +- tsconfig.json | 2 + 10 files changed, 249 insertions(+), 22 deletions(-) diff --git a/LICENSE b/LICENSE index 507cc7deda09..bcc6ca1809f7 100644 --- a/LICENSE +++ b/LICENSE @@ -1,6 +1,6 @@ MIT License -Copyright (c) 2024 TUM Applied Education Technologies +Copyright (c) 2025 TUM Applied Education Technologies Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal diff --git a/angular.json b/angular.json index 02cb18df601c..24f70a21a733 100644 --- a/angular.json +++ b/angular.json @@ -226,7 +226,7 @@ "packageManager": "npm", "cache": { "enabled": true, - "path": ".cache", + "path": "./build/angular/", "environment": "all" }, "schematicCollections": [ diff --git a/docs/conf.py b/docs/conf.py index 705a2a5e18db..090ac361cddb 100644 --- a/docs/conf.py +++ b/docs/conf.py @@ -18,7 +18,7 @@ # -- Project information ----------------------------------------------------- project = 'Artemis' -copyright = '2024, Applied Education Technologies, Technical University of Munich' +copyright = '2025, Applied Education Technologies, Technical University of Munich' author = 'Applied Education Technologies, Technical University of Munich' diff --git a/gradle/test.gradle b/gradle/test.gradle index d09c3a9c61c4..8e84af376e9f 100644 --- a/gradle/test.gradle +++ b/gradle/test.gradle @@ -57,7 +57,8 @@ test { useJUnitPlatform() exclude "**/*IT*", "**/*IntTest*" } - + testClassesDirs = testing.suites.test.sources.output.classesDirs + classpath = testing.suites.test.sources.runtimeClasspath testLogging { events "FAILED", "SKIPPED" } diff --git a/package-lock.json b/package-lock.json index df82b26a8c98..3f71e3b6a204 100644 --- a/package-lock.json +++ b/package-lock.json @@ -131,6 +131,7 @@ "sass": "1.83.0", "ts-jest": "29.2.5", "typescript": "5.5.4", + "typescript-eslint": "8.18.2", "vite-tsconfig-paths": "5.1.4", "vitest": "2.1.8", "weak-napi": "2.0.2" @@ -20795,6 +20796,232 @@ "typescript-logic": "^0.0.0" } }, + "node_modules/typescript-eslint": { + "version": "8.18.2", + "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.18.2.tgz", + "integrity": "sha512-KuXezG6jHkvC3MvizeXgupZzaG5wjhU3yE8E7e6viOvAvD9xAWYp8/vy0WULTGe9DYDWcQu7aW03YIV3mSitrQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/eslint-plugin": "8.18.2", + "@typescript-eslint/parser": "8.18.2", + "@typescript-eslint/utils": "8.18.2" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/typescript-eslint/node_modules/@typescript-eslint/eslint-plugin": { + "version": "8.18.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.18.2.tgz", + "integrity": "sha512-adig4SzPLjeQ0Tm+jvsozSGiCliI2ajeURDGHjZ2llnA+A67HihCQ+a3amtPhUakd1GlwHxSRvzOZktbEvhPPg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/regexpp": "^4.10.0", + "@typescript-eslint/scope-manager": "8.18.2", + "@typescript-eslint/type-utils": "8.18.2", + "@typescript-eslint/utils": "8.18.2", + "@typescript-eslint/visitor-keys": "8.18.2", + "graphemer": "^1.4.0", + "ignore": "^5.3.1", + "natural-compare": "^1.4.0", + "ts-api-utils": "^1.3.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^8.0.0 || ^8.0.0-alpha.0", + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/typescript-eslint/node_modules/@typescript-eslint/parser": { + "version": "8.18.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.18.2.tgz", + "integrity": "sha512-y7tcq4StgxQD4mDr9+Jb26dZ+HTZ/SkfqpXSiqeUXZHxOUyjWDKsmwKhJ0/tApR08DgOhrFAoAhyB80/p3ViuA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/scope-manager": "8.18.2", + "@typescript-eslint/types": "8.18.2", + "@typescript-eslint/typescript-estree": "8.18.2", + "@typescript-eslint/visitor-keys": "8.18.2", + "debug": "^4.3.4" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/typescript-eslint/node_modules/@typescript-eslint/scope-manager": { + "version": "8.18.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.18.2.tgz", + "integrity": "sha512-YJFSfbd0CJjy14r/EvWapYgV4R5CHzptssoag2M7y3Ra7XNta6GPAJPPP5KGB9j14viYXyrzRO5GkX7CRfo8/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.18.2", + "@typescript-eslint/visitor-keys": "8.18.2" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/typescript-eslint/node_modules/@typescript-eslint/type-utils": { + "version": "8.18.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.18.2.tgz", + "integrity": "sha512-AB/Wr1Lz31bzHfGm/jgbFR0VB0SML/hd2P1yxzKDM48YmP7vbyJNHRExUE/wZsQj2wUCvbWH8poNHFuxLqCTnA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/typescript-estree": "8.18.2", + "@typescript-eslint/utils": "8.18.2", + "debug": "^4.3.4", + "ts-api-utils": "^1.3.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/typescript-eslint/node_modules/@typescript-eslint/types": { + "version": "8.18.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.2.tgz", + "integrity": "sha512-Z/zblEPp8cIvmEn6+tPDIHUbRu/0z5lqZ+NvolL5SvXWT5rQy7+Nch83M0++XzO0XrWRFWECgOAyE8bsJTl1GQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/typescript-eslint/node_modules/@typescript-eslint/typescript-estree": { + "version": "8.18.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.18.2.tgz", + "integrity": "sha512-WXAVt595HjpmlfH4crSdM/1bcsqh+1weFRWIa9XMTx/XHZ9TCKMcr725tLYqWOgzKdeDrqVHxFotrvWcEsk2Tg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.18.2", + "@typescript-eslint/visitor-keys": "8.18.2", + "debug": "^4.3.4", + "fast-glob": "^3.3.2", + "is-glob": "^4.0.3", + "minimatch": "^9.0.4", + "semver": "^7.6.0", + "ts-api-utils": "^1.3.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/typescript-eslint/node_modules/@typescript-eslint/utils": { + "version": "8.18.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.18.2.tgz", + "integrity": "sha512-Cr4A0H7DtVIPkauj4sTSXVl+VBWewE9/o40KcF3TV9aqDEOWoXF3/+oRXNby3DYzZeCATvbdksYsGZzplwnK/Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.4.0", + "@typescript-eslint/scope-manager": "8.18.2", + "@typescript-eslint/types": "8.18.2", + "@typescript-eslint/typescript-estree": "8.18.2" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/typescript-eslint/node_modules/@typescript-eslint/visitor-keys": { + "version": "8.18.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.18.2.tgz", + "integrity": "sha512-zORcwn4C3trOWiCqFQP1x6G3xTRyZ1LYydnj51cRnJ6hxBlr/cKPckk+PKPUw/fXmvfKTcw7bwY3w9izgx5jZw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.18.2", + "eslint-visitor-keys": "^4.2.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/typescript-eslint/node_modules/eslint-visitor-keys": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", + "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/typescript-eslint/node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, "node_modules/typescript-logic": { "version": "0.0.0", "resolved": "https://registry.npmjs.org/typescript-logic/-/typescript-logic-0.0.0.tgz", diff --git a/package.json b/package.json index 7c7e3e405edd..1248fce42f27 100644 --- a/package.json +++ b/package.json @@ -165,6 +165,7 @@ "sass": "1.83.0", "ts-jest": "29.2.5", "typescript": "5.5.4", + "typescript-eslint": "8.18.2", "vite-tsconfig-paths": "5.1.4", "vitest": "2.1.8", "weak-napi": "2.0.2" diff --git a/src/main/webapp/app/exercises/quiz/manage/apollon-diagrams/apollon-diagram-detail.component.ts b/src/main/webapp/app/exercises/quiz/manage/apollon-diagrams/apollon-diagram-detail.component.ts index 9505c20fe759..ef93f826cc49 100644 --- a/src/main/webapp/app/exercises/quiz/manage/apollon-diagrams/apollon-diagram-detail.component.ts +++ b/src/main/webapp/app/exercises/quiz/manage/apollon-diagrams/apollon-diagram-detail.component.ts @@ -27,10 +27,8 @@ export class ApollonDiagramDetailComponent implements OnInit, OnDestroy { @ViewChild('editorContainer', { static: false }) editorContainer: ElementRef; @ViewChild('titleField') titleField?: NgModel; - @Input() - private courseId: number; - @Input() - private apollonDiagramId: number; + @Input() courseId: number; + @Input() apollonDiagramId: number; @Output() closeEdit = new EventEmitter(); @Output() closeModal = new EventEmitter(); diff --git a/src/main/webapp/app/overview/course-conversations/dialogs/conversation-add-users-dialog/add-users-form/conversation-add-users-form.component.ts b/src/main/webapp/app/overview/course-conversations/dialogs/conversation-add-users-dialog/add-users-form/conversation-add-users-form.component.ts index 08c13cd628ab..f2cec3a67fc5 100644 --- a/src/main/webapp/app/overview/course-conversations/dialogs/conversation-add-users-dialog/add-users-form/conversation-add-users-form.component.ts +++ b/src/main/webapp/app/overview/course-conversations/dialogs/conversation-add-users-dialog/add-users-form/conversation-add-users-form.component.ts @@ -19,13 +19,12 @@ export interface AddUsersFormData { }) export class ConversationAddUsersFormComponent implements OnInit, OnChanges { @Output() formSubmitted: EventEmitter = new EventEmitter(); + @Input() courseId: number; @Input() maxSelectable?: number = undefined; + @Input() activeConversation: ConversationDTO; - @Input() - activeConversation: ConversationDTO; - - protected readonly isLoading = input(false); + isLoading = input(false); form: FormGroup; diff --git a/src/main/webapp/app/overview/course-lectures/lecture-unit/lecture-unit.component.ts b/src/main/webapp/app/overview/course-lectures/lecture-unit/lecture-unit.component.ts index aa7735766df1..b4cf868d5d5a 100644 --- a/src/main/webapp/app/overview/course-lectures/lecture-unit/lecture-unit.component.ts +++ b/src/main/webapp/app/overview/course-lectures/lecture-unit/lecture-unit.component.ts @@ -16,21 +16,20 @@ export class LectureUnitComponent { protected faSquareCheck = faSquareCheck; protected faSquare = faSquare; - readonly lectureUnit = input.required(); - protected readonly icon = input.required(); + lectureUnit = input.required(); + icon = input.required(); - readonly showViewIsolatedButton = input(false); - readonly viewIsolatedButtonLabel = input('artemisApp.textUnit.isolated'); - readonly viewIsolatedButtonIcon = input(faExternalLinkAlt); - readonly onShowIsolated = output(); + showViewIsolatedButton = input(false); + viewIsolatedButtonLabel = input('artemisApp.textUnit.isolated'); + viewIsolatedButtonIcon = input(faExternalLinkAlt); + isPresentationMode = input.required(); - readonly isCollapsed = signal(true); + readonly onShowIsolated = output(); readonly onCollapse = output(); - - readonly isPresentationMode = input.required(); - readonly onCompletion = output(); + readonly isCollapsed = signal(true); + readonly isVisibleToStudents = computed(() => this.lectureUnit().visibleToStudents); toggleCompletion(event: Event) { diff --git a/tsconfig.json b/tsconfig.json index d6cc333517c6..8dcb8e56932a 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -33,6 +33,8 @@ "angularCompilerOptions": { "genDir": "build/resources/main/aot", "skipMetadataEmit": true, + "strictInjectionParameters": true, + "strictInputAccessModifiers": true, "fullTemplateTypeCheck": true, "strictTemplates": true, "preserveWhitespaces": true, From 7d59fe01b6baa534844cea655d595ace611e9788 Mon Sep 17 00:00:00 2001 From: Ajayvir Singh <38434017+AjayvirS@users.noreply.github.com> Date: Thu, 2 Jan 2025 20:45:33 +0100 Subject: [PATCH 03/34] Development: Update liquibase property name in gradle build config (#10096) --- gradle/liquibase.gradle | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/gradle/liquibase.gradle b/gradle/liquibase.gradle index dd48fa3cb338..70ed9b74070c 100644 --- a/gradle/liquibase.gradle +++ b/gradle/liquibase.gradle @@ -19,12 +19,12 @@ ext.isWindows = OperatingSystem.current().isWindows() if (isWindows) { tasks.register("pathingLiquibaseJar", Jar) { - dependsOn configurations.liquibase + dependsOn configurations.liquibaseRuntime archiveAppendix = "pathingLiquibase" doFirst { manifest { - attributes "Class-Path": (sourceSets.main.runtimeClasspath + configurations.liquibase).collect { + attributes "Class-Path": (sourceSets.main.runtimeClasspath + configurations.liquibaseRuntime).collect { it.toURI().toURL().toString().replaceFirst(/file:\/+/, "/") }.join(" ") } @@ -46,7 +46,7 @@ def liquibaseCommand(command) { classpath tasks.named("pathingLiquibaseJar").get().outputs.files } else { classpath sourceSets.main.runtimeClasspath - classpath configurations.liquibase + classpath configurations.liquibaseRuntime } mainClass = "liquibase.integration.commandline.Main" From 82bf67cf8ec6e1b6d36a233b90efc96d4d0e39e4 Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Thu, 2 Jan 2025 21:27:17 +0100 Subject: [PATCH 04/34] Development: Fix an issue with LTI authorization types (#10098) --- .../artemis/communication/dto/ChannelDTO.java | 5 + .../artemis/lti/config/Lti13LaunchFilter.java | 4 +- .../tum/cit/aet/artemis/lti/dto/Claims.java | 2 +- .../aet/artemis/lti/dto/Lti13AgsClaim.java | 2 + ....java => Lti13AuthenticationResponse.java} | 5 +- .../lti/dto/Lti13ClientRegistration.java | 270 +----------------- .../dto/Lti13ClientRegistrationFactory.java | 48 ++++ .../lti/dto/Lti13DeepLinkingResponse.java | 3 +- .../artemis/lti/dto/Lti13LaunchRequest.java | 2 + .../cit/aet/artemis/lti/dto/Lti13Message.java | 12 + .../lti/dto/Lti13PlatformConfiguration.java | 2 + .../lti/dto/Lti13ToolConfiguration.java | 14 + .../LtiDynamicRegistrationService.java | 8 +- .../OnlineCourseConfigurationService.java | 4 +- .../{ => lti}/Lti13LaunchFilterTest.java | 6 +- .../security/lti/Lti13TokenRetrieverTest.java | 4 +- .../LtiDynamicRegistrationServiceTest.java | 5 +- .../OnlineCourseConfigurationServiceTest.java | 4 +- 18 files changed, 118 insertions(+), 282 deletions(-) rename src/main/java/de/tum/cit/aet/artemis/lti/dto/{LtiAuthenticationResponse.java => Lti13AuthenticationResponse.java} (58%) create mode 100644 src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ClientRegistrationFactory.java create mode 100644 src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13Message.java create mode 100644 src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ToolConfiguration.java rename src/test/java/de/tum/cit/aet/artemis/core/security/{ => lti}/Lti13LaunchFilterTest.java (98%) diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/dto/ChannelDTO.java b/src/main/java/de/tum/cit/aet/artemis/communication/dto/ChannelDTO.java index 6e5738d3904a..6fd8d9c89fa2 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/dto/ChannelDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/dto/ChannelDTO.java @@ -197,6 +197,11 @@ else if (channel.getExam() != null) { } } + /** + * Converts the DTO to a channel entity + * + * @return the created channel entity based on the attributes in the DTO + */ public Channel toChannel() { Channel channel = new Channel(); channel.setName(this.name); diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/config/Lti13LaunchFilter.java b/src/main/java/de/tum/cit/aet/artemis/lti/config/Lti13LaunchFilter.java index 6929eee2d537..65624c32a3e5 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/config/Lti13LaunchFilter.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/config/Lti13LaunchFilter.java @@ -26,7 +26,7 @@ import de.tum.cit.aet.artemis.core.exception.LtiEmailAlreadyInUseException; import de.tum.cit.aet.artemis.core.security.SecurityUtils; import de.tum.cit.aet.artemis.lti.dto.Claims; -import de.tum.cit.aet.artemis.lti.dto.LtiAuthenticationResponse; +import de.tum.cit.aet.artemis.lti.dto.Lti13AuthenticationResponse; import de.tum.cit.aet.artemis.lti.service.Lti13Service; import uk.ac.ox.ctl.lti13.security.oauth2.client.lti.authentication.OidcAuthenticationToken; import uk.ac.ox.ctl.lti13.security.oauth2.client.lti.web.OAuth2LoginAuthenticationFilter; @@ -125,7 +125,7 @@ private void writeResponse(String targetLinkUri, OidcIdToken ltiIdToken, String log.info("User is authenticated, building LTI response"); lti13Service.buildLtiResponse(uriBuilder, response); } - LtiAuthenticationResponse jsonResponse = new LtiAuthenticationResponse(uriBuilder.build().toUriString(), ltiIdToken.getTokenValue(), clientRegistrationId); + Lti13AuthenticationResponse jsonResponse = new Lti13AuthenticationResponse(uriBuilder.build().toUriString(), ltiIdToken.getTokenValue(), clientRegistrationId); response.setContentType("application/json"); response.setCharacterEncoding("UTF-8"); diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Claims.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Claims.java index 41659ae1210b..38a1b7a24bba 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Claims.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Claims.java @@ -1,6 +1,6 @@ package de.tum.cit.aet.artemis.lti.dto; -public class Claims extends uk.ac.ox.ctl.lti13.lti.Claims { +public final class Claims extends uk.ac.ox.ctl.lti13.lti.Claims { /** * Constant for LTI Assignment and Grade Services (AGS) claim endpoint. diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13AgsClaim.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13AgsClaim.java index c3ba3dab86d0..0b417baa473e 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13AgsClaim.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13AgsClaim.java @@ -6,12 +6,14 @@ import org.springframework.security.oauth2.core.oidc.OidcIdToken; +import com.fasterxml.jackson.annotation.JsonInclude; import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.ObjectMapper; /** * A wrapper record for an LTI 1.3 Assignment and Grading Services Claim. We support the Score Publishing Service in order to transmit scores. */ +@JsonInclude(JsonInclude.Include.NON_EMPTY) public record Lti13AgsClaim(List scope, String lineItem) { /** diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/LtiAuthenticationResponse.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13AuthenticationResponse.java similarity index 58% rename from src/main/java/de/tum/cit/aet/artemis/lti/dto/LtiAuthenticationResponse.java rename to src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13AuthenticationResponse.java index 49470f2fdc5b..0a6de2a3c212 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/dto/LtiAuthenticationResponse.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13AuthenticationResponse.java @@ -1,5 +1,7 @@ package de.tum.cit.aet.artemis.lti.dto; +import com.fasterxml.jackson.annotation.JsonInclude; + /** * Holds LTI authentication response details. * @@ -7,5 +9,6 @@ * @param ltiIdToken LTI service provided ID token. * @param clientRegistrationId Client's registration ID with LTI service. */ -public record LtiAuthenticationResponse(String targetLinkUri, String ltiIdToken, String clientRegistrationId) { +@JsonInclude(JsonInclude.Include.NON_EMPTY) +public record Lti13AuthenticationResponse(String targetLinkUri, String ltiIdToken, String clientRegistrationId) { } diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ClientRegistration.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ClientRegistration.java index f61dc9af1367..ef651272c086 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ClientRegistration.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ClientRegistration.java @@ -1,276 +1,20 @@ package de.tum.cit.aet.artemis.lti.dto; -import java.util.Arrays; import java.util.List; -import org.springframework.security.oauth2.core.AuthorizationGrantType; - +import com.fasterxml.jackson.annotation.JsonInclude; import com.fasterxml.jackson.annotation.JsonProperty; -import de.tum.cit.aet.artemis.lti.config.CustomLti13Configurer; - /** * Represents the client registration details for an LTI 1.3 integration. * This class encapsulates information required for LTI 1.3 client registration, * including response types, grant types, redirect URIs, and tool configuration. */ -public class Lti13ClientRegistration { - - @JsonProperty("client_id") - private String clientId; - - @JsonProperty("response_types") - private List responseTypes; - - @JsonProperty("grant_types") - private List grantTypes; - - @JsonProperty("initiate_login_uri") - private String initiateLoginUri; - - @JsonProperty("redirect_uris") - private List redirectUris; - - @JsonProperty("client_name") - private String clientName; - - @JsonProperty("jwks_uri") - private String jwksUri; - - @JsonProperty("logo_uri") - private String logoUri; - - @JsonProperty("token_endpoint_auth_method") - private String tokenEndpointAuthMethod; - - private String scope; - - @JsonProperty("https://purl.imsglobal.org/spec/lti-tool-configuration") - private Lti13ToolConfiguration lti13ToolConfiguration; - - /** - * Default constructor necessary for conversion. - */ - public Lti13ClientRegistration() { // Necessary for conversion - } - - /** - * Constructs a new Lti13ClientRegistration with specified server URL and client registration ID. - * Initializes various properties such as grant types, response types, and tool configurations. - * - * @param serverUrl The server URL for LTI configuration. - * @param clientRegistrationId The client registration ID for LTI configuration. - */ - public Lti13ClientRegistration(String serverUrl, String clientRegistrationId) { - this.setGrantTypes(Arrays.asList(AuthorizationGrantType.CLIENT_CREDENTIALS.getValue(), AuthorizationGrantType.AUTHORIZATION_CODE.getValue())); - this.setResponseTypes(List.of("id_token")); - this.setClientName("Artemis - " + serverUrl); - this.setTokenEndpointAuthMethod("private_key_jwt"); - this.setScope(String.join(" ", List.of(Scopes.AGS_SCORE, Scopes.AGS_RESULT))); - this.setRedirectUris(List.of(serverUrl + "/" + CustomLti13Configurer.LTI13_LOGIN_REDIRECT_PROXY_PATH)); - this.setInitiateLoginUri(serverUrl + "/" + CustomLti13Configurer.LTI13_LOGIN_INITIATION_PATH + "/" + clientRegistrationId); - this.setJwksUri(serverUrl + "/.well-known/jwks.json"); - this.setLogoUri(serverUrl + "/public/images/logo.png"); - - Lti13ToolConfiguration toolConfiguration = getLti13ToolConfiguration(serverUrl); - this.setLti13ToolConfiguration(toolConfiguration); - } - - private static Lti13ToolConfiguration getLti13ToolConfiguration(String serverUrl) { - Lti13ToolConfiguration toolConfiguration = new Lti13ToolConfiguration(); - - // Extracting the domain from the server URL - String[] urlParts = serverUrl.split("://"); - String domain = ""; - if (urlParts.length >= 1) { - domain = urlParts[1]; // Domain cannot include protocol - } - toolConfiguration.setDomain(domain); - toolConfiguration.setTargetLinkUri(serverUrl + "/courses"); - toolConfiguration.setDescription("Artemis: Interactive Learning with Individual Feedback"); - toolConfiguration.setClaims(Arrays.asList("iss", "email", "sub", "name", "given_name", "family_name")); - Message deepLinkingMessage = new Message(CustomLti13Configurer.LTI13_DEEPLINK_MESSAGE_REQUEST, serverUrl + "/" + CustomLti13Configurer.LTI13_DEEPLINK_REDIRECT_PATH); - toolConfiguration.setMessages(List.of(deepLinkingMessage)); - return toolConfiguration; - } - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } - - public List getResponseTypes() { - return responseTypes; - } - - public void setResponseTypes(List responseTypes) { - this.responseTypes = responseTypes; - } - - public List getGrantTypes() { - return grantTypes; - } - - public void setGrantTypes(List grantTypes) { - this.grantTypes = grantTypes; - } - - public String getInitiateLoginUri() { - return initiateLoginUri; - } - - public void setInitiateLoginUri(String initiateLoginUri) { - this.initiateLoginUri = initiateLoginUri; - } - - public List getRedirectUris() { - return redirectUris; - } - - public void setRedirectUris(List redirectUris) { - this.redirectUris = redirectUris; - } - - public String getClientName() { - return clientName; - } - - public void setClientName(String clientName) { - this.clientName = clientName; - } - - public String getJwksUri() { - return jwksUri; - } - - public void setJwksUri(String jwksUri) { - this.jwksUri = jwksUri; - } - - public String getLogoUri() { - return logoUri; - } - - public void setLogoUri(String logoUri) { - this.logoUri = logoUri; - } - - public String getTokenEndpointAuthMethod() { - return tokenEndpointAuthMethod; - } - - public void setTokenEndpointAuthMethod(String tokenEndpointAuthMethod) { - this.tokenEndpointAuthMethod = tokenEndpointAuthMethod; - } - - public String getScope() { - return scope; - } - - public void setScope(String scope) { - this.scope = scope; - } - - public Lti13ToolConfiguration getLti13ToolConfiguration() { - return lti13ToolConfiguration; - } - - public void setLti13ToolConfiguration(Lti13ToolConfiguration lti13ToolConfiguration) { - this.lti13ToolConfiguration = lti13ToolConfiguration; - } - - /** - * Inner class representing the LTI 1.3 tool configuration. - */ - public static class Lti13ToolConfiguration { - - private String domain; - - @JsonProperty("target_link_uri") - private String targetLinkUri; - - private String description; - - private List messages; - - private List claims; - - public String getDomain() { - return domain; - } - - public void setDomain(String domain) { - this.domain = domain; - } - - public String getTargetLinkUri() { - return targetLinkUri; - } - - public void setTargetLinkUri(String targetLinkUri) { - this.targetLinkUri = targetLinkUri; - } - - public String getDescription() { - return description; - } - - public void setDescription(String description) { - this.description = description; - } - - public List getMessages() { - return messages; - } - - public void setMessages(List messages) { - this.messages = messages; - } - - public List getClaims() { - return claims; - } - - public void setClaims(List claims) { - this.claims = claims; - } - } - - /** - * Inner class representing a message in LTI 1.3 tool configuration. - */ - public static class Message { - - private String type; - - @JsonProperty("target_link_uri") - private String targetLinkUri; - - public Message() {// Necessary for conversion - } - - public Message(String type, String targetLinkUri) { - this.type = type; - this.targetLinkUri = targetLinkUri; - } - - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } - - public String getTargetLinkUri() { - return targetLinkUri; - } +@JsonInclude(JsonInclude.Include.NON_EMPTY) +public record Lti13ClientRegistration(@JsonProperty("client_id") String clientId, @JsonProperty("response_types") List responseTypes, + @JsonProperty("grant_types") List grantTypes, @JsonProperty("initiate_login_uri") String initiateLoginUri, @JsonProperty("redirect_uris") List redirectUris, + @JsonProperty("client_name") String clientName, @JsonProperty("jwks_uri") String jwksUri, @JsonProperty("logo_uri") String logoUri, + @JsonProperty("token_endpoint_auth_method") String tokenEndpointAuthMethod, String scope, + @JsonProperty("https://purl.imsglobal.org/spec/lti-tool-configuration") Lti13ToolConfiguration lti13ToolConfiguration) { - public void setTargetLinkUri(String targetLinkUri) { - this.targetLinkUri = targetLinkUri; - } - } } diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ClientRegistrationFactory.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ClientRegistrationFactory.java new file mode 100644 index 000000000000..e7ffd354871e --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ClientRegistrationFactory.java @@ -0,0 +1,48 @@ +package de.tum.cit.aet.artemis.lti.dto; + +import java.util.Arrays; +import java.util.List; + +import org.springframework.security.oauth2.core.AuthorizationGrantType; + +import de.tum.cit.aet.artemis.lti.config.CustomLti13Configurer; +import uk.ac.ox.ctl.lti13.security.oauth2.client.lti.web.LTIAuthorizationGrantType; + +public class Lti13ClientRegistrationFactory { + + /** + * Constructs a new Lti13ClientRegistration with specified server URL and client registration ID. + * Initializes various properties such as grant types, response types, and tool configurations. + * + * @param serverUrl The server URL for LTI configuration. + * @param clientRegistrationId The client registration ID for LTI configuration. + * @return A new Lti13ClientRegistration object with certain default values for Artemis + */ + public static Lti13ClientRegistration createRegistration(String serverUrl, String clientRegistrationId) { + var grantTypes = Arrays.asList(AuthorizationGrantType.CLIENT_CREDENTIALS.getValue(), LTIAuthorizationGrantType.IMPLICIT.getValue()); + var responseTypes = List.of("id_token"); + var clientName = "Artemis - " + serverUrl; + var tokenEndpointAuthMethod = "private_key_jwt"; + var scope = String.join(" ", List.of(Scopes.AGS_SCORE, Scopes.AGS_RESULT)); + var redirectUris = List.of(serverUrl + "/" + CustomLti13Configurer.LTI13_LOGIN_REDIRECT_PROXY_PATH); + var initiateLoginUri = serverUrl + "/" + CustomLti13Configurer.LTI13_LOGIN_INITIATION_PATH + "/" + clientRegistrationId; + var jwksUri = serverUrl + "/.well-known/jwks.json"; + var logoUri = serverUrl + "/public/images/logo.png"; + + return new Lti13ClientRegistration(clientRegistrationId, responseTypes, grantTypes, initiateLoginUri, redirectUris, clientName, jwksUri, logoUri, tokenEndpointAuthMethod, + scope, createLti13ToolConfiguration(serverUrl)); + } + + private static Lti13ToolConfiguration createLti13ToolConfiguration(String serverUrl) { + + // Extracting the domain from the server URL + String[] urlParts = serverUrl.split("://"); + String domain = ""; + if (urlParts.length >= 1) { + domain = urlParts[1]; // Domain cannot include protocol + } + var claims = Arrays.asList("iss", "email", "sub", "name", "given_name", "family_name"); + var deepLinkingMessage = new Lti13Message(CustomLti13Configurer.LTI13_DEEPLINK_MESSAGE_REQUEST, serverUrl + "/" + CustomLti13Configurer.LTI13_DEEPLINK_REDIRECT_PATH); + return new Lti13ToolConfiguration(domain, serverUrl + "/courses", "Artemis: Interactive Learning with Individual Feedback", List.of(deepLinkingMessage), claims); + } +} diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13DeepLinkingResponse.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13DeepLinkingResponse.java index 8d9986d433c0..e622dee0ca5a 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13DeepLinkingResponse.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13DeepLinkingResponse.java @@ -7,6 +7,7 @@ import org.springframework.security.oauth2.core.oidc.IdTokenClaimNames; import org.springframework.security.oauth2.core.oidc.OidcIdToken; +import com.fasterxml.jackson.annotation.JsonInclude; import com.fasterxml.jackson.annotation.JsonProperty; import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.ObjectMapper; @@ -34,6 +35,7 @@ * @param clientRegistrationId The client registration ID. * @param returnUrl The URL to return to after deep linking is completed. */ +@JsonInclude(JsonInclude.Include.NON_EMPTY) public record Lti13DeepLinkingResponse(@JsonProperty(IdTokenClaimNames.AUD) String aud, @JsonProperty(IdTokenClaimNames.ISS) String iss, @JsonProperty(IdTokenClaimNames.EXP) String exp, @JsonProperty(IdTokenClaimNames.IAT) String iat, @JsonProperty(IdTokenClaimNames.NONCE) String nonce, @JsonProperty(Claims.MSG) String message, @JsonProperty(Claims.LTI_DEPLOYMENT_ID) String deploymentId, @JsonProperty(Claims.MESSAGE_TYPE) String messageType, @@ -105,7 +107,6 @@ public Map getClaims() { claims.put(Claims.MESSAGE_TYPE, messageType()); claims.put(Claims.LTI_VERSION, ltiVersion()); claims.put(Claims.CONTENT_ITEMS, contentItems()); - return claims; } diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13LaunchRequest.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13LaunchRequest.java index 8576a7dff782..e33f0b62c007 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13LaunchRequest.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13LaunchRequest.java @@ -4,6 +4,7 @@ import org.springframework.security.oauth2.core.oidc.OidcIdToken; import org.springframework.util.Assert; +import com.fasterxml.jackson.annotation.JsonInclude; import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.ObjectMapper; @@ -19,6 +20,7 @@ * @param agsClaim An optional {@link Lti13AgsClaim} representing the Assignment and Grade Services claim, if present. * @param clientRegistrationId The client registration ID, identifying the tool registration with the platform. */ +@JsonInclude(JsonInclude.Include.NON_EMPTY) public record Lti13LaunchRequest(String iss, String sub, String deploymentId, String resourceLinkId, String targetLinkUri, Lti13AgsClaim agsClaim, String clientRegistrationId) { /** diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13Message.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13Message.java new file mode 100644 index 000000000000..43eb826f353e --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13Message.java @@ -0,0 +1,12 @@ +package de.tum.cit.aet.artemis.lti.dto; + +import com.fasterxml.jackson.annotation.JsonInclude; +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * Inner class representing a message in LTI 1.3 tool configuration. + */ +@JsonInclude(JsonInclude.Include.NON_EMPTY) +public record Lti13Message(String type, @JsonProperty("target_link_uri") String targetLinkUri) { + +} diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13PlatformConfiguration.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13PlatformConfiguration.java index e1a54be22f60..23e0aa772dfa 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13PlatformConfiguration.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13PlatformConfiguration.java @@ -1,5 +1,6 @@ package de.tum.cit.aet.artemis.lti.dto; +import com.fasterxml.jackson.annotation.JsonInclude; import com.fasterxml.jackson.annotation.JsonProperty; /** @@ -11,6 +12,7 @@ * @param jwksUri The URI for the JSON Web Key Set (JWKS). * @param registrationEndpoint The endpoint URL for registration. */ +@JsonInclude(JsonInclude.Include.NON_EMPTY) public record Lti13PlatformConfiguration(@JsonProperty("issuer") String issuer, @JsonProperty("token_endpoint") String tokenEndpoint, @JsonProperty("authorization_endpoint") String authorizationEndpoint, @JsonProperty("jwks_uri") String jwksUri, @JsonProperty("registration_endpoint") String registrationEndpoint) { diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ToolConfiguration.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ToolConfiguration.java new file mode 100644 index 000000000000..99a4d8a907f3 --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Lti13ToolConfiguration.java @@ -0,0 +1,14 @@ +package de.tum.cit.aet.artemis.lti.dto; + +import java.util.List; + +import com.fasterxml.jackson.annotation.JsonInclude; +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * Inner class representing the LTI 1.3 tool configuration. + */ +@JsonInclude(JsonInclude.Include.NON_EMPTY) +public record Lti13ToolConfiguration(String domain, @JsonProperty("target_link_uri") String targetLinkUri, String description, List messages, List claims) { + +} diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/service/LtiDynamicRegistrationService.java b/src/main/java/de/tum/cit/aet/artemis/lti/service/LtiDynamicRegistrationService.java index bb4ba0abf16e..636877628224 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/service/LtiDynamicRegistrationService.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/service/LtiDynamicRegistrationService.java @@ -20,6 +20,7 @@ import de.tum.cit.aet.artemis.core.exception.BadRequestAlertException; import de.tum.cit.aet.artemis.lti.domain.LtiPlatformConfiguration; import de.tum.cit.aet.artemis.lti.dto.Lti13ClientRegistration; +import de.tum.cit.aet.artemis.lti.dto.Lti13ClientRegistrationFactory; import de.tum.cit.aet.artemis.lti.dto.Lti13PlatformConfiguration; import de.tum.cit.aet.artemis.lti.repository.LtiPlatformConfigurationRepository; @@ -62,8 +63,7 @@ public void performDynamicRegistration(String openIdConfigurationUrl, String reg throw new BadRequestAlertException("Invalid platform configuration", "LTI", "invalidPlatformConfiguration"); } - Lti13ClientRegistration clientRegistrationResponse = postClientRegistrationToPlatform(platformConfiguration.registrationEndpoint(), clientRegistrationId, - registrationToken); + var clientRegistrationResponse = postClientRegistrationToPlatform(platformConfiguration.registrationEndpoint(), clientRegistrationId, registrationToken); LtiPlatformConfiguration ltiPlatformConfiguration = updateLtiPlatformConfiguration(clientRegistrationId, platformConfiguration, clientRegistrationResponse); ltiPlatformConfigurationRepository.save(ltiPlatformConfiguration); @@ -100,7 +100,7 @@ private Lti13ClientRegistration postClientRegistrationToPlatform(String registra headers.setBearerAuth(registrationToken); } - Lti13ClientRegistration lti13ClientRegistration = new Lti13ClientRegistration(artemisServerUrl, clientRegistrationId); + Lti13ClientRegistration lti13ClientRegistration = Lti13ClientRegistrationFactory.createRegistration(artemisServerUrl, clientRegistrationId); Lti13ClientRegistration registrationResponse = null; try { ResponseEntity response = restTemplate.postForEntity(registrationEndpoint, new HttpEntity<>(lti13ClientRegistration, headers), @@ -123,7 +123,7 @@ private LtiPlatformConfiguration updateLtiPlatformConfiguration(String registrat Lti13ClientRegistration clientRegistrationResponse) { LtiPlatformConfiguration ltiPlatformConfiguration = new LtiPlatformConfiguration(); ltiPlatformConfiguration.setRegistrationId(registrationId); - ltiPlatformConfiguration.setClientId(clientRegistrationResponse.getClientId()); + ltiPlatformConfiguration.setClientId(clientRegistrationResponse.clientId()); ltiPlatformConfiguration.setAuthorizationUri(platformConfiguration.authorizationEndpoint()); ltiPlatformConfiguration.setJwkSetUri(platformConfiguration.jwksUri()); ltiPlatformConfiguration.setTokenUri(platformConfiguration.tokenEndpoint()); diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/service/OnlineCourseConfigurationService.java b/src/main/java/de/tum/cit/aet/artemis/lti/service/OnlineCourseConfigurationService.java index ffbf6db62f82..e70a464b8900 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/service/OnlineCourseConfigurationService.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/service/OnlineCourseConfigurationService.java @@ -15,7 +15,6 @@ import org.springframework.context.annotation.Profile; import org.springframework.security.oauth2.client.registration.ClientRegistration; import org.springframework.security.oauth2.client.registration.ClientRegistrationRepository; -import org.springframework.security.oauth2.core.AuthorizationGrantType; import org.springframework.stereotype.Service; import de.tum.cit.aet.artemis.core.domain.Course; @@ -24,6 +23,7 @@ import de.tum.cit.aet.artemis.lti.domain.LtiPlatformConfiguration; import de.tum.cit.aet.artemis.lti.domain.OnlineCourseConfiguration; import de.tum.cit.aet.artemis.lti.repository.LtiPlatformConfigurationRepository; +import uk.ac.ox.ctl.lti13.security.oauth2.client.lti.web.LTIAuthorizationGrantType; /** * Service Implementation for OnlineCourseConfiguration. @@ -104,7 +104,7 @@ public ClientRegistration getClientRegistration(LtiPlatformConfiguration ltiPlat .tokenUri(ltiPlatformConfiguration.getTokenUri()) // .redirectUri(artemisServerUrl + "/" + CustomLti13Configurer.LTI13_LOGIN_REDIRECT_PROXY_PATH) // .scope("openid") // - .authorizationGrantType(AuthorizationGrantType.AUTHORIZATION_CODE) // + .authorizationGrantType(LTIAuthorizationGrantType.IMPLICIT) // .build(); } catch (IllegalArgumentException e) { diff --git a/src/test/java/de/tum/cit/aet/artemis/core/security/Lti13LaunchFilterTest.java b/src/test/java/de/tum/cit/aet/artemis/core/security/lti/Lti13LaunchFilterTest.java similarity index 98% rename from src/test/java/de/tum/cit/aet/artemis/core/security/Lti13LaunchFilterTest.java rename to src/test/java/de/tum/cit/aet/artemis/core/security/lti/Lti13LaunchFilterTest.java index 4bc65daa9a32..0d2de6aef218 100644 --- a/src/test/java/de/tum/cit/aet/artemis/core/security/Lti13LaunchFilterTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/core/security/lti/Lti13LaunchFilterTest.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.core.security; +package de.tum.cit.aet.artemis.core.security.lti; import static org.assertj.core.api.Assertions.assertThat; import static org.mockito.Mockito.any; @@ -230,7 +230,7 @@ void emailAddressAlreadyInUseServiceLaunchFailed() throws IOException, ServletEx verify(httpResponse).setStatus(HttpStatus.UNAUTHORIZED.value()); assertThat((responseJsonBody.get("targetLinkUri").toString())).as("Response body contains the expected targetLinkUri") .contains("https://any-artemis-domain.org/course/123/exercise/1234"); - assertThat(responseJsonBody.get("ltiIdToken").isNull()).isTrue(); + assertThat(responseJsonBody.get("ltiIdToken")).isNull(); assertThat((responseJsonBody.get("clientRegistrationId").toString())).as("Response body contains the expected clientRegistrationId").contains("some-registration"); } @@ -247,7 +247,7 @@ void emailAddressAlreadyInUseServiceDeepLinkingFailed() throws ServletException, verify(httpResponse).setStatus(HttpStatus.UNAUTHORIZED.value()); assertThat((responseJsonBody.get("targetLinkUri").toString())).as("Response body contains the expected targetLinkUri").contains("/lti/select-course"); - assertThat(responseJsonBody.get("ltiIdToken").isNull()).isTrue(); + assertThat(responseJsonBody.get("ltiIdToken")).isNull(); assertThat((responseJsonBody.get("clientRegistrationId").toString())).as("Response body contains the expected clientRegistrationId").contains("some-registration"); } diff --git a/src/test/java/de/tum/cit/aet/artemis/core/security/lti/Lti13TokenRetrieverTest.java b/src/test/java/de/tum/cit/aet/artemis/core/security/lti/Lti13TokenRetrieverTest.java index 764824a74fe1..02eb2e3fa4f8 100644 --- a/src/test/java/de/tum/cit/aet/artemis/core/security/lti/Lti13TokenRetrieverTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/core/security/lti/Lti13TokenRetrieverTest.java @@ -32,7 +32,6 @@ import org.springframework.security.crypto.keygen.Base64StringKeyGenerator; import org.springframework.security.crypto.keygen.StringKeyGenerator; import org.springframework.security.oauth2.client.registration.ClientRegistration; -import org.springframework.security.oauth2.core.AuthorizationGrantType; import org.springframework.web.client.HttpClientErrorException; import org.springframework.web.client.RestTemplate; @@ -50,6 +49,7 @@ import de.tum.cit.aet.artemis.lti.config.Lti13TokenRetriever; import de.tum.cit.aet.artemis.lti.dto.Scopes; import de.tum.cit.aet.artemis.lti.service.OAuth2JWKSService; +import uk.ac.ox.ctl.lti13.security.oauth2.client.lti.web.LTIAuthorizationGrantType; class Lti13TokenRetrieverTest { @@ -72,7 +72,7 @@ void init() { lti13TokenRetriever = new Lti13TokenRetriever(oAuth2JWKSService, restTemplate); clientRegistration = ClientRegistration.withRegistrationId("regId") // - .authorizationGrantType(AuthorizationGrantType.AUTHORIZATION_CODE) // + .authorizationGrantType(LTIAuthorizationGrantType.IMPLICIT) // .redirectUri("redirectUri") // .authorizationUri("authUri") // .tokenUri("tokenUri").clientId("clientId") // diff --git a/src/test/java/de/tum/cit/aet/artemis/lti/LtiDynamicRegistrationServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/lti/LtiDynamicRegistrationServiceTest.java index 036a4034325d..fb348724ce61 100644 --- a/src/test/java/de/tum/cit/aet/artemis/lti/LtiDynamicRegistrationServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/lti/LtiDynamicRegistrationServiceTest.java @@ -8,6 +8,8 @@ import static org.mockito.Mockito.verify; import static org.mockito.Mockito.when; +import java.util.UUID; + import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; @@ -23,6 +25,7 @@ import de.tum.cit.aet.artemis.core.exception.BadRequestAlertException; import de.tum.cit.aet.artemis.lti.domain.OnlineCourseConfiguration; import de.tum.cit.aet.artemis.lti.dto.Lti13ClientRegistration; +import de.tum.cit.aet.artemis.lti.dto.Lti13ClientRegistrationFactory; import de.tum.cit.aet.artemis.lti.dto.Lti13PlatformConfiguration; import de.tum.cit.aet.artemis.lti.service.LtiDynamicRegistrationService; import de.tum.cit.aet.artemis.lti.service.OAuth2JWKSService; @@ -66,7 +69,7 @@ void init() { registrationToken = "token"; platformConfiguration = new Lti13PlatformConfiguration(null, "token", "auth", "jwks", "register"); - clientRegistrationResponse = new Lti13ClientRegistration(); + clientRegistrationResponse = Lti13ClientRegistrationFactory.createRegistration("http://artemis.com", "artemis-" + UUID.randomUUID()); } @AfterEach diff --git a/src/test/java/de/tum/cit/aet/artemis/lti/service/OnlineCourseConfigurationServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/lti/service/OnlineCourseConfigurationServiceTest.java index 28217ced4ec3..0ecb0491005f 100644 --- a/src/test/java/de/tum/cit/aet/artemis/lti/service/OnlineCourseConfigurationServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/lti/service/OnlineCourseConfigurationServiceTest.java @@ -16,13 +16,13 @@ import org.springframework.beans.factory.annotation.Value; import org.springframework.security.core.context.SecurityContextHolder; import org.springframework.security.oauth2.client.registration.ClientRegistration; -import org.springframework.security.oauth2.core.AuthorizationGrantType; import de.tum.cit.aet.artemis.core.exception.BadRequestAlertException; import de.tum.cit.aet.artemis.core.exception.EntityNotFoundException; import de.tum.cit.aet.artemis.lti.domain.LtiPlatformConfiguration; import de.tum.cit.aet.artemis.lti.domain.OnlineCourseConfiguration; import de.tum.cit.aet.artemis.lti.test_repository.LtiPlatformConfigurationTestRepository; +import uk.ac.ox.ctl.lti13.security.oauth2.client.lti.web.LTIAuthorizationGrantType; class OnlineCourseConfigurationServiceTest { @@ -65,7 +65,7 @@ void getClientRegistrationSuccess() { ClientRegistration clientRegistration = onlineCourseConfigurationService.getClientRegistration(ltiPlatformConfiguration); - assertThat(clientRegistration.getAuthorizationGrantType()).isEqualTo(AuthorizationGrantType.AUTHORIZATION_CODE); + assertThat(clientRegistration.getAuthorizationGrantType()).isEqualTo(LTIAuthorizationGrantType.IMPLICIT); assertThat(clientRegistration.getScopes()).hasSize(1).contains("openid"); assertThat(clientRegistration.getRegistrationId()).isEqualTo("reg"); assertThat(clientRegistration.getRedirectUri()).isEqualTo(artemisServerUrl + "/api/public/lti13/auth-callback"); From 22d38a81305d97454068102b018e1c7024a37bfc Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Sat, 4 Jan 2025 11:21:35 +0100 Subject: [PATCH 05/34] Development: Remove testwise coverage support (#9993) --- .github/issue-labeler.yml | 1 - docs/admin/setup/programming-exercises.rst | 3 - ...tegrated_Code_Lifecycle_Build_Job_Item.svg | 2 +- .../programming-exercise-features.inc | 79 ++- .../exercises/programming-exercise-setup.inc | 3 - gradle/test.gradle | 4 +- .../artemis/assessment/domain/Feedback.java | 2 +- .../aet/artemis/assessment/domain/Result.java | 19 - .../assessment/service/ResultService.java | 4 +- .../web/open/PublicResultResource.java | 13 +- .../service/LearningObjectImportService.java | 3 +- .../artemis/buildagent/dto/BuildConfig.java | 6 +- .../artemis/buildagent/dto/BuildResult.java | 165 +----- .../artemis/buildagent/dto/LocalCIJobDTO.java | 26 + .../buildagent/dto/LocalCITestJobDTO.java | 18 + .../service/BuildJobExecutionService.java | 28 +- .../service/SharedQueueProcessingService.java | 3 +- .../service/TestResultXmlParser.java | 25 +- .../conversation/ConversationService.java | 4 +- .../core/repository/UserRepository.java | 2 +- .../service/TitleCacheEvictionService.java | 10 - .../exam/service/ExamImportService.java | 2 +- .../exercise/dto/ExerciseDetailsDTO.java | 6 +- .../service/ParticipationService.java | 12 +- .../exercise/web/ExerciseResource.java | 15 +- .../tum/cit/aet/artemis/lti/dto/Scopes.java | 2 +- ...ProgrammingPlagiarismDetectionService.java | 2 +- .../domain/ProgrammingExercise.java | 16 +- .../ProgrammingExerciseBuildConfig.java | 18 +- .../ProgrammingExerciseGitDiffEntry.java | 2 +- .../ProgrammingExerciseGitDiffReport.java | 3 +- .../{hestia => }/ProgrammingExerciseTask.java | 19 +- .../domain/ProgrammingExerciseTestCase.java | 32 +- .../ProgrammingExerciseTestCaseType.java | 4 +- .../programming/domain/hestia/CodeHint.java | 69 --- .../domain/hestia/CoverageFileReport.java | 86 --- .../domain/hestia/CoverageReport.java | 75 --- .../domain/hestia/ExerciseHint.java | 192 ------ .../domain/hestia/ExerciseHintActivation.java | 67 --- .../ProgrammingExerciseSolutionEntry.java | 135 ----- .../hestia/TestwiseCoverageReportEntry.java | 72 --- ...OInterface.java => BuildJobInterface.java} | 6 +- ...nDTO.java => BuildResultNotification.java} | 39 +- .../programming/dto/ConsistencyErrorDTO.java | 58 +- .../CoverageReportAndSubmissionDateDTO.java | 11 - .../ProgrammingExerciseGitDiffEntryDTO.java | 2 +- .../ProgrammingExerciseGitDiffReportDTO.java | 2 +- ...TestCaseBaseDTO.java => TestCaseBase.java} | 6 +- .../programming/dto/aeolus/Action.java | 30 + .../aeolus/AeolusRepository.java | 2 +- .../{service => dto}/aeolus/AeolusResult.java | 2 +- .../{service => dto}/aeolus/DockerConfig.java | 2 +- .../dto/aeolus/PlatformAction.java | 17 + .../{service => dto}/aeolus/ScriptAction.java | 18 +- .../programming/dto/aeolus/Windfile.java | 86 +++ .../aeolus/WindfileMetadata.java | 2 +- ...ammingExerciseGitDiffReportRepository.java | 4 +- .../ProgrammingExerciseRepository.java | 9 +- .../ProgrammingExerciseTaskRepository.java | 21 +- ...ProgrammingExerciseTestCaseRepository.java | 35 -- .../repository/hestia/CodeHintRepository.java | 50 -- .../hestia/CoverageFileReportRepository.java | 15 - .../hestia/CoverageReportRepository.java | 101 ---- .../ExerciseHintActivationRepository.java | 41 -- .../hestia/ExerciseHintRepository.java | 50 -- ...ammingExerciseSolutionEntryRepository.java | 83 --- ...TestwiseCoverageReportEntryRepository.java | 18 - .../service/BuildScriptProviderService.java | 14 +- .../service/CommitHistoryService.java | 5 +- .../GenericBuildScriptGenerationService.java | 2 +- .../GitDiffReportParserService.java | 4 +- .../ProgrammingExerciseExportService.java | 1 - ...ammingExerciseFeedbackCreationService.java | 15 +- ...ogrammingExerciseGitDiffReportService.java | 10 +- .../ProgrammingExerciseGradingService.java | 21 +- ...ProgrammingExerciseImportBasicService.java | 182 ++---- .../ProgrammingExerciseImportService.java | 1 - .../ProgrammingExerciseRepositoryService.java | 2 - .../service/ProgrammingExerciseService.java | 39 +- .../ProgrammingExerciseTaskService.java | 46 +- .../ProgrammingExerciseTestCaseService.java | 1 - .../service/ProgrammingLanguageFeature.java | 3 +- .../service/ProgrammingSubmissionService.java | 1 - .../programming/service/aeolus/Action.java | 86 --- .../service/aeolus/ActionDeserializer.java | 4 + .../aeolus/AeolusBuildPlanService.java | 2 + .../AeolusBuildScriptGenerationService.java | 6 +- .../service/aeolus/AeolusTemplateService.java | 35 +- .../service/aeolus/PlatformAction.java | 33 -- .../programming/service/aeolus/Windfile.java | 123 ---- ...actContinuousIntegrationResultService.java | 64 +- .../ContinuousIntegrationResultService.java | 6 +- .../ci/notification/dto/TestCaseDTO.java | 11 +- .../ci/notification/dto/TestResultsDTO.java | 139 +---- .../ci/notification/dto/TestSuiteDTO.java | 10 +- .../dto/TestwiseCoverageReportDTO.java | 115 ---- ...abCIProgrammingLanguageFeatureService.java | 8 +- .../gitlabci/GitLabCIResultService.java | 16 +- .../service/gitlabci/GitLabCIService.java | 2 +- .../service/hestia/CodeHintService.java | 187 ------ .../service/hestia/ExerciseHintService.java | 326 ----------- .../hestia/TestwiseCoverageService.java | 314 ---------- .../behavioral/BehavioralBlackboard.java | 58 -- ...ioralSolutionEntryGenerationException.java | 18 - .../behavioral/BehavioralTestCaseService.java | 201 ------- .../hestia/behavioral/GroupedFile.java | 150 ----- ...ddUncoveredLinesAsPotentialCodeBlocks.java | 110 ---- .../BehavioralKnowledgeSource.java | 27 - .../knowledgesource/CombineChangeBlocks.java | 61 -- .../CreateCommonChangeBlocks.java | 64 -- .../CreateSolutionEntries.java | 57 -- .../DropRemovedGitDiffEntries.java | 35 -- .../knowledgesource/ExtractChangedLines.java | 35 -- .../knowledgesource/ExtractCoveredLines.java | 36 -- .../knowledgesource/FindCommonLines.java | 35 -- ...dCoverageEntriesByFilePathAndTestCase.java | 70 --- .../knowledgesource/InsertFileContents.java | 34 -- .../structural/StructuralAttribute.java | 79 --- .../hestia/structural/StructuralClass.java | 145 ----- .../structural/StructuralClassElements.java | 66 --- .../structural/StructuralConstructor.java | 77 --- .../hestia/structural/StructuralElement.java | 135 ----- .../hestia/structural/StructuralMethod.java | 140 ----- ...turalSolutionEntryGenerationException.java | 18 - .../structural/StructuralTestCaseService.java | 254 -------- ...kinsProgrammingLanguageFeatureService.java | 28 +- .../service/jenkins/JenkinsResultService.java | 16 +- .../service/jenkins/JenkinsService.java | 2 +- .../build_plan/JenkinsBuildPlanService.java | 12 +- .../JenkinsPipelineScriptCreator.java | 9 +- .../LocalCIBuildConfigurationService.java | 10 +- ...alCIProgrammingLanguageFeatureService.java | 34 +- .../localci/LocalCIQueueWebsocketService.java | 4 +- .../service/localci/LocalCIResultService.java | 12 +- .../service/localci/LocalCIService.java | 2 +- .../localci/LocalCITriggerService.java | 19 +- ...ogrammingExerciseExportImportResource.java | 6 +- ...grammingExerciseGitDiffReportResource.java | 6 +- .../web/ProgrammingExerciseResource.java | 40 +- .../ProgrammingExerciseTaskResource.java | 6 +- .../web/hestia/CodeHintResource.java | 124 ---- .../web/hestia/CoverageReportResource.java | 68 --- .../web/hestia/ExerciseHintResource.java | 349 ----------- ...grammingExerciseSolutionEntryResource.java | 337 ----------- .../web/localci/AeolusTemplateResource.java | 37 +- ...grammingExerciseParticipationResource.java | 20 - .../java/plain_gradle_static_coverage.sh | 34 -- .../java/plain_gradle_static_coverage.yaml | 29 - .../java/plain_maven_static_coverage.sh | 40 -- .../java/plain_maven_static_coverage.yaml | 30 - .../aeolus/kotlin/default_coverage.sh | 26 - .../aeolus/kotlin/default_coverage.yaml | 15 - src/main/resources/templates/iris/hestia.hbs | 80 --- .../maven_maven/test/projectTemplate/pom.xml | 58 -- .../test/gradle/projectTemplate/build.gradle | 27 - .../java/test/maven/projectTemplate/pom.xml | 58 -- .../java/gradle/regularRuns/pipeline.groovy | 20 +- .../java/maven/regularRuns/pipeline.groovy | 20 +- .../kotlin/regularRuns/pipeline.groovy | 20 +- .../kotlin/test/maven/projectTemplate/pom.xml | 58 -- .../course/manage/course-management.module.ts | 2 - .../course/manage/course-update.component.ts | 2 +- .../manage/detail/course-detail.component.ts | 6 - ...rogramming-diff-report-detail.component.ts | 6 +- .../app/detail-overview-list/detail.model.ts | 2 +- .../app/entities/hestia/code-hint-model.ts | 12 - .../hestia/coverage-file-report.model.ts | 13 - .../entities/hestia/coverage-report.model.ts | 11 - .../entities/hestia/exercise-hint.model.ts | 20 - ...ogramming-exercise-solution-entry.model.ts | 14 - .../testwise-coverage-report-entry.model.ts | 12 - ...ogramming-exercise-git-diff-entry.model.ts | 0 ...gramming-exercise-git-diff-report.model.ts | 2 +- .../programming-exercise-task.model.ts | 0 .../programming/build-config.model.ts | 1 - .../programming-exercise-build.config.ts | 2 - .../programming-exercise-test-case.model.ts | 2 - .../programming/programming-exercise.model.ts | 11 +- .../app/exam/manage/exam-management.module.ts | 2 +- ...rogramming-exercise-exam-diff.component.ts | 4 +- .../student-exam-timeline.component.ts | 2 +- .../git-diff-file-panel-title.component.html | 0 .../git-diff-file-panel-title.component.scss | 0 .../git-diff-file-panel-title.component.ts | 0 .../git-diff-file-panel.component.html | 0 .../git-diff-file-panel.component.scss | 0 .../git-diff-file-panel.component.ts | 8 +- .../git-diff-file.component.html | 0 .../git-diff-file.component.ts | 2 +- .../git-diff-line-stat.component.html | 0 .../git-diff-line-stat.component.scss | 0 .../git-diff-line-stat.component.ts | 0 .../git-diff-report-modal.component.html | 0 .../git-diff-report-modal.component.ts | 4 +- .../git-diff-report.component.html | 0 .../git-diff-report.component.ts | 8 +- ...de-hint-generation-overview.component.html | 52 -- ...de-hint-generation-overview.component.scss | 49 -- ...code-hint-generation-overview.component.ts | 95 --- .../code-hint-generation-overview.module.ts | 53 -- ...hint-generation-status-step.component.html | 14 - ...e-hint-generation-status-step.component.ts | 24 - ...code-hint-generation-status.component.html | 45 -- ...code-hint-generation-status.component.scss | 65 -- .../code-hint-generation-status.component.ts | 26 - ...lution-entry-creation-modal.component.html | 43 -- ...solution-entry-creation-modal.component.ts | 85 --- ...olution-entry-details-modal.component.html | 21 - .../solution-entry-details-modal.component.ts | 42 -- .../code-hint-generation-step.component.html | 74 --- .../code-hint-generation-step.component.ts | 67 --- .../coverage-generation-step.component.html | 13 - .../coverage-generation-step.component.ts | 47 -- .../diff-generation-step.component.html | 15 - .../diff-generation-step.component.ts | 54 -- ...ution-entry-generation-step.component.html | 97 --- ...olution-entry-generation-step.component.ts | 178 ------ .../testwise-coverage-file.component.html | 11 - .../testwise-coverage-file.component.scss | 20 - .../testwise-coverage-file.component.ts | 91 --- ...twise-coverage-report-modal.component.html | 11 - ...estwise-coverage-report-modal.component.ts | 21 - .../testwise-coverage-report.component.html | 17 - .../testwise-coverage-report.component.ts | 75 --- .../testwise-coverage-report.module.ts | 14 - ...-exercise-grading-tasks-table.component.ts | 2 +- .../programming-exercise-task.service.ts | 2 +- .../tasks/programming-exercise-task.ts | 2 +- ...programming-exercise-detail.component.html | 30 - .../programming-exercise-detail.component.ts | 105 +--- ...ming-exercise-management-routing.module.ts | 13 - .../programming-exercise-management.module.ts | 4 - .../programming-exercise.component.html | 9 - .../services/programming-exercise.service.ts | 46 +- .../programming-exercise-creation-config.ts | 1 - .../programming-exercise-update.component.ts | 6 - ...cise-custom-aeolus-build-plan.component.ts | 23 +- ...ng-exercise-custom-build-plan.component.ts | 54 +- ...ogramming-exercise-language.component.html | 18 +- ...de-editor-student-container.component.html | 5 - ...code-editor-student-container.component.ts | 37 -- .../programming-participation.module.ts | 2 - .../programming-repository.module.ts | 2 +- .../shared/service/aeolus.service.ts | 19 +- .../programming-language-feature.service.ts | 1 - .../exercise-hint-detail.component.html | 44 -- .../manage/exercise-hint-detail.component.ts | 48 -- .../manage/exercise-hint-management.module.ts | 32 - .../exercise-hint-update.component.html | 91 --- .../manage/exercise-hint-update.component.ts | 196 ------- .../manage/exercise-hint.component.html | 110 ---- .../manage/exercise-hint.component.ts | 127 ---- .../manage/exercise-hint.route.ts | 96 --- ...xercise-hint-button-overlay.component.html | 13 - ...xercise-hint-button-overlay.component.scss | 20 - .../exercise-hint-button-overlay.component.ts | 41 -- .../exercise-hint-expandable.component.html | 56 -- .../exercise-hint-expandable.component.scss | 31 - .../exercise-hint-expandable.component.ts | 58 -- .../exercise-hint-participation.module.ts | 26 - ...xercise-hint-student-dialog.component.html | 32 - .../exercise-hint-student-dialog.component.ts | 26 - .../exercise-hint-student-dialog.scss | 4 - .../services/code-hint-cast.pipe.ts | 16 - .../services/code-hint.service.ts | 68 --- ...ramming-exercise-solution-entry.service.ts | 97 --- .../shared/code-hint-container.component.html | 25 - .../shared/code-hint-container.component.ts | 68 --- .../shared/exercise-hint-shared.module.ts | 14 - .../shared/exercise-hint.service.ts | 179 ------ .../shared/solution-entry.component.html | 12 - .../shared/solution-entry.component.scss | 3 - .../shared/solution-entry.component.ts | 47 -- .../shared/exercise/exercise.service.ts | 6 - .../commit-details-view.component.ts | 2 +- .../course-exercise-details.component.html | 6 - .../course-exercise-details.component.ts | 35 -- .../course-exercise-details.module.ts | 2 - .../webapp/app/shared/http/file.service.ts | 23 +- .../layouts/navbar/entity-title.service.ts | 4 - .../shared/layouts/navbar/navbar.component.ts | 8 - .../markdown-editor-monaco.component.ts | 6 +- .../content/scss/themes/_dark-variables.scss | 1 - .../scss/themes/_default-variables.scss | 1 - src/main/webapp/i18n/de/codeHint.json | 105 ---- src/main/webapp/i18n/de/error.json | 1 - src/main/webapp/i18n/de/iris.json | 4 +- .../webapp/i18n/de/programmingExercise.json | 26 - src/main/webapp/i18n/en/codeHint.json | 105 ---- src/main/webapp/i18n/en/error.json | 1 - src/main/webapp/i18n/en/iris.json | 4 +- .../webapp/i18n/en/programmingExercise.json | 26 - .../service/BuildAgentDockerServiceTest.java | 3 +- .../buildagent/service/BuildResultTest.java | 4 +- .../service/TestResultXmlParserTest.java | 42 +- .../TitleCacheEvictionServiceTest.java | 39 -- .../artemis/core/util/CourseTestService.java | 2 +- .../PlagiarismDetectionServiceTest.java | 8 +- ...ProgrammingIntegrationIndependentTest.java | 38 +- ...mingIntegrationLocalCILocalVCTestBase.java | 34 +- .../programming/BuildPlanIntegrationTest.java | 1 - ...gExerciseIntegrationJenkinsGitlabTest.java | 23 +- ...rammingExerciseIntegrationTestService.java | 93 +-- ...rammingExerciseLocalVCIntegrationTest.java | 6 - ...gExerciseResultJenkinsIntegrationTest.java | 2 +- ...rammingExerciseServiceIntegrationTest.java | 30 +- ...ammingExerciseTemplateIntegrationTest.java | 45 +- ...AndResultGitlabJenkinsIntegrationTest.java | 20 +- .../RepositoryIntegrationTest.java | 7 - .../StaticCodeAnalysisIntegrationTest.java | 4 +- ...eolusBuildScriptGenerationServiceTest.java | 11 +- .../programming/aelous/AeolusServiceTest.java | 33 +- .../aelous/AeolusTemplateResourceTest.java | 23 +- .../programming/aelous/AeolusTest.java | 118 ++-- .../hestia/CodeHintIntegrationTest.java | 147 ----- .../hestia/CodeHintServiceTest.java | 249 -------- .../hestia/ExerciseHintIntegrationTest.java | 457 --------------- .../hestia/ExerciseHintServiceTest.java | 304 ---------- .../hestia/HestiaDatabaseTest.java | 165 ------ ...gExerciseGitDiffReportIntegrationTest.java | 223 ------- ...mmingExerciseGitDiffReportServiceTest.java | 179 ------ ...gExerciseSolutionEntryIntegrationTest.java | 245 -------- ...rogrammingExerciseTaskIntegrationTest.java | 222 ------- .../ProgrammingExerciseTaskServiceTest.java | 402 ------------- .../hestia/StructuralTestCaseServiceTest.java | 554 ------------------ .../TestwiseCoverageIntegrationTest.java | 114 ---- .../TestwiseCoverageReportServiceTest.java | 102 ---- ...coveredLinesAsPotentialCodeBlocksTest.java | 122 ---- ...oralTestCaseServiceLocalCILocalVCTest.java | 149 ----- .../behavioral/CombineChangeBlocksTest.java | 84 --- .../CreateCommonChangeBlocksTest.java | 58 -- .../behavioral/CreateSolutionEntriesTest.java | 106 ---- .../DropRemovedGitDiffEntriesTest.java | 54 -- .../behavioral/ExtractChangedLinesTest.java | 71 --- .../behavioral/ExtractCoveredLinesTest.java | 71 --- .../behavioral/FindCommonLinesTest.java | 56 -- ...erageEntriesByFilePathAndTestCaseTest.java | 117 ---- .../behavioral/InsertFileContentsTest.java | 60 -- .../hestia/util/TestwiseCoverageTestUtil.java | 49 -- .../icl/LocalCIIntegrationTest.java | 4 +- .../icl/LocalCIResourceIntegrationTest.java | 6 +- .../programming/icl/LocalCIServiceTest.java | 7 +- .../icl/LocalVCLocalCITestService.java | 4 +- .../service/ConsistencyCheckTestService.java | 2 +- ...ngExerciseFeedbackCreationServiceTest.java | 6 +- .../JenkinsPipelineScriptCreatorTest.java | 5 +- ...ProgrammingExerciseTaskTestRepository.java | 37 +- ...rammingExerciseTestCaseTestRepository.java | 17 - .../ProgrammingExerciseTestRepository.java | 6 +- .../util/ProgrammingExerciseFactory.java | 21 +- .../ProgrammingExerciseResultTestService.java | 77 +-- .../util/ProgrammingExerciseTestService.java | 35 +- .../util/ProgrammingExerciseUtilService.java | 182 ++---- ...issionAndResultIntegrationTestService.java | 2 +- .../ProgrammingUtilTestService.java} | 12 +- ...AbstractModuleServiceArchitectureTest.java | 3 +- ...editor-student-container.component.spec.ts | 2 - .../programming-exam-diff.component.spec.ts | 6 +- .../exercise-hint-detail.component.spec.ts | 37 -- .../exercise-hint-update.component.spec.ts | 239 -------- .../manage/exercise-hint.component.spec.ts | 90 --- ...cise-hint-button-overlay.component.spec.ts | 57 -- ...exercise-hint-expandable.component.spec.ts | 82 --- .../code-hint-container.component.spec.ts | 91 --- .../shared/solution-entry.component.spec.ts | 55 -- ...it-diff-file-panel-title.component.spec.ts | 4 +- .../git-diff-file-panel.component.spec.ts | 14 +- .../git-diff-file.component.spec.ts | 8 +- .../git-diff-line-stat.component.spec.ts | 6 +- .../git-diff-modal.component.spec.ts | 14 +- .../git-diff-report.component.spec.ts | 16 +- ...hint-generation-overview.component.spec.ts | 176 ------ ...e-hint-generation-status.component.spec.ts | 28 - ...ode-hint-generation-step.component.spec.ts | 69 --- ...coverage-generation-step.component.spec.ts | 65 -- .../diff-generation-step.component.spec.ts | 67 --- ...ion-entry-creation-modal.component.spec.ts | 97 --- ...tion-entry-details-modal.component.spec.ts | 62 -- ...on-entry-generation-step.component.spec.ts | 210 ------- .../testwise-coverage-file.component.spec.ts | 116 ---- ...testwise-coverage-report.component.spec.ts | 133 ----- .../build-agent-details.component.spec.ts | 1 - .../build-agent-summary.component.spec.ts | 1 - .../build-agents/build-agents.service.spec.ts | 1 - .../build-queue/build-queue.component.spec.ts | 4 - .../build-queue/build-queue.service.spec.ts | 1 - .../commit-details-view.component.spec.ts | 4 +- .../course-exercise-details.component.spec.ts | 28 - ...mming-diff-report-detail.component.spec.ts | 2 +- ...custom-aeolus-build-plan.component.spec.ts | 4 - ...ercise-custom-build-plan.component.spec.ts | 4 - ...gramming-exercise-detail.component.spec.ts | 45 +- ...gramming-exercise-update.component.spec.ts | 10 - ...ogramming-exercise-creation-config-mock.ts | 1 - .../shared/http/file.service.spec.ts | 41 +- .../component/shared/navbar.component.spec.ts | 58 +- .../service/mock-exercise-hint.service.ts | 49 -- .../mock-programming-exercise.service.ts | 6 - .../code-editor-student.integration.spec.ts | 2 - .../spec/service/entity-title.service.spec.ts | 4 - .../service/exercise-hint.service.spec.ts | 111 ---- .../spec/service/profile.service.spec.ts | 10 - .../programming-exercise-task.service.spec.ts | 2 +- .../programming-exercise.service.spec.ts | 28 +- .../ProgrammingExerciseAssessment.spec.ts | 1 - .../exercise/programming/python/template.json | 1 - .../support/requests/ExerciseAPIRequests.ts | 4 - .../manage_programming_exercise.py | 1 - 408 files changed, 1003 insertions(+), 17785 deletions(-) create mode 100644 src/main/java/de/tum/cit/aet/artemis/buildagent/dto/LocalCIJobDTO.java create mode 100644 src/main/java/de/tum/cit/aet/artemis/buildagent/dto/LocalCITestJobDTO.java rename src/main/java/de/tum/cit/aet/artemis/programming/domain/{hestia => }/ProgrammingExerciseGitDiffEntry.java (98%) rename src/main/java/de/tum/cit/aet/artemis/programming/domain/{hestia => }/ProgrammingExerciseGitDiffReport.java (95%) rename src/main/java/de/tum/cit/aet/artemis/programming/domain/{hestia => }/ProgrammingExerciseTask.java (73%) rename src/main/java/de/tum/cit/aet/artemis/programming/domain/{hestia => }/ProgrammingExerciseTestCaseType.java (77%) delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CodeHint.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CoverageFileReport.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CoverageReport.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ExerciseHint.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ExerciseHintActivation.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseSolutionEntry.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/TestwiseCoverageReportEntry.java rename src/main/java/de/tum/cit/aet/artemis/programming/dto/{BuildJobDTOInterface.java => BuildJobInterface.java} (74%) rename src/main/java/de/tum/cit/aet/artemis/programming/dto/{AbstractBuildResultNotificationDTO.java => BuildResultNotification.java} (54%) delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/dto/CoverageReportAndSubmissionDateDTO.java rename src/main/java/de/tum/cit/aet/artemis/programming/dto/{TestCaseBaseDTO.java => TestCaseBase.java} (85%) create mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/Action.java rename src/main/java/de/tum/cit/aet/artemis/programming/{service => dto}/aeolus/AeolusRepository.java (85%) rename src/main/java/de/tum/cit/aet/artemis/programming/{service => dto}/aeolus/AeolusResult.java (83%) rename src/main/java/de/tum/cit/aet/artemis/programming/{service => dto}/aeolus/DockerConfig.java (95%) create mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/PlatformAction.java rename src/main/java/de/tum/cit/aet/artemis/programming/{service => dto}/aeolus/ScriptAction.java (60%) create mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/Windfile.java rename src/main/java/de/tum/cit/aet/artemis/programming/{service => dto}/aeolus/WindfileMetadata.java (90%) rename src/main/java/de/tum/cit/aet/artemis/programming/repository/{hestia => }/ProgrammingExerciseGitDiffReportRepository.java (88%) rename src/main/java/de/tum/cit/aet/artemis/programming/repository/{hestia => }/ProgrammingExerciseTaskRepository.java (74%) delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CodeHintRepository.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CoverageFileReportRepository.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CoverageReportRepository.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ExerciseHintActivationRepository.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ExerciseHintRepository.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ProgrammingExerciseSolutionEntryRepository.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/TestwiseCoverageReportEntryRepository.java rename src/main/java/de/tum/cit/aet/artemis/programming/{web => service}/GitDiffReportParserService.java (98%) rename src/main/java/de/tum/cit/aet/artemis/programming/service/{hestia => }/ProgrammingExerciseGitDiffReportService.java (97%) rename src/main/java/de/tum/cit/aet/artemis/programming/service/{hestia => }/ProgrammingExerciseTaskService.java (91%) delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/Action.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/PlatformAction.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/Windfile.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestwiseCoverageReportDTO.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/CodeHintService.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/ExerciseHintService.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/TestwiseCoverageService.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralBlackboard.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralSolutionEntryGenerationException.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralTestCaseService.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/GroupedFile.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/AddUncoveredLinesAsPotentialCodeBlocks.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/BehavioralKnowledgeSource.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CombineChangeBlocks.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CreateCommonChangeBlocks.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CreateSolutionEntries.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/DropRemovedGitDiffEntries.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/ExtractChangedLines.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/ExtractCoveredLines.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/FindCommonLines.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/GroupGitDiffAndCoverageEntriesByFilePathAndTestCase.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/InsertFileContents.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralAttribute.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralClass.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralClassElements.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralConstructor.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralElement.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralMethod.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralSolutionEntryGenerationException.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralTestCaseService.java rename src/main/java/de/tum/cit/aet/artemis/programming/web/{hestia => }/ProgrammingExerciseGitDiffReportResource.java (98%) rename src/main/java/de/tum/cit/aet/artemis/programming/web/{hestia => }/ProgrammingExerciseTaskResource.java (94%) delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/CodeHintResource.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/CoverageReportResource.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ExerciseHintResource.java delete mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ProgrammingExerciseSolutionEntryResource.java delete mode 100644 src/main/resources/templates/aeolus/java/plain_gradle_static_coverage.sh delete mode 100644 src/main/resources/templates/aeolus/java/plain_gradle_static_coverage.yaml delete mode 100644 src/main/resources/templates/aeolus/java/plain_maven_static_coverage.sh delete mode 100644 src/main/resources/templates/aeolus/java/plain_maven_static_coverage.yaml delete mode 100644 src/main/resources/templates/aeolus/kotlin/default_coverage.sh delete mode 100644 src/main/resources/templates/aeolus/kotlin/default_coverage.yaml delete mode 100644 src/main/resources/templates/iris/hestia.hbs delete mode 100644 src/main/webapp/app/entities/hestia/code-hint-model.ts delete mode 100644 src/main/webapp/app/entities/hestia/coverage-file-report.model.ts delete mode 100644 src/main/webapp/app/entities/hestia/coverage-report.model.ts delete mode 100644 src/main/webapp/app/entities/hestia/exercise-hint.model.ts delete mode 100644 src/main/webapp/app/entities/hestia/programming-exercise-solution-entry.model.ts delete mode 100644 src/main/webapp/app/entities/hestia/testwise-coverage-report-entry.model.ts rename src/main/webapp/app/entities/{hestia => }/programming-exercise-git-diff-entry.model.ts (100%) rename src/main/webapp/app/entities/{hestia => }/programming-exercise-git-diff-report.model.ts (85%) rename src/main/webapp/app/entities/{hestia => }/programming-exercise-task.model.ts (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-file-panel-title.component.html (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-file-panel-title.component.scss (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-file-panel-title.component.ts (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-file-panel.component.html (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-file-panel.component.scss (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-file-panel.component.ts (90%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-file.component.html (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-file.component.ts (92%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-line-stat.component.html (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-line-stat.component.scss (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-line-stat.component.ts (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-report-modal.component.html (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-report-modal.component.ts (97%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-report.component.html (100%) rename src/main/webapp/app/exercises/programming/{hestia => }/git-diff-report/git-diff-report.component.ts (95%) delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.scss delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.module.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status-step.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status-step.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.scss delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/solution-entry-details-modal/solution-entry-details-modal.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/solution-entry-details-modal/solution-entry-details-modal.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/code-hint-generation-step/code-hint-generation-step.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/code-hint-generation-step/code-hint-generation-step.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/coverage-generation-step/coverage-generation-step.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/coverage-generation-step/coverage-generation-step.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/diff-generation-step/diff-generation-step.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/diff-generation-step/diff-generation-step.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/solution-entry-generation-step/solution-entry-generation-step.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/solution-entry-generation-step/solution-entry-generation-step.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.scss delete mode 100644 src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report-modal.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report-modal.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.component.html delete mode 100644 src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.component.ts delete mode 100644 src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.module.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint-detail.component.html delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint-detail.component.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint-management.module.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint-update.component.html delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint-update.component.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.component.html delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.component.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.route.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.html delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.scss delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.html delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.scss delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-participation.module.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component.html delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.scss delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/services/code-hint-cast.pipe.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/services/code-hint.service.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/services/programming-exercise-solution-entry.service.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/shared/code-hint-container.component.html delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/shared/code-hint-container.component.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/shared/exercise-hint-shared.module.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/shared/exercise-hint.service.ts delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.html delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.scss delete mode 100644 src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.ts delete mode 100644 src/main/webapp/i18n/de/codeHint.json delete mode 100644 src/main/webapp/i18n/en/codeHint.json delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/CodeHintIntegrationTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/CodeHintServiceTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/ExerciseHintIntegrationTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/ExerciseHintServiceTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/HestiaDatabaseTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseGitDiffReportIntegrationTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseGitDiffReportServiceTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseSolutionEntryIntegrationTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseTaskIntegrationTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseTaskServiceTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/StructuralTestCaseServiceTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/TestwiseCoverageIntegrationTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/TestwiseCoverageReportServiceTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/AddUncoveredLinesAsPotentialCodeBlocksTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/BehavioralTestCaseServiceLocalCILocalVCTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CombineChangeBlocksTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CreateCommonChangeBlocksTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CreateSolutionEntriesTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/DropRemovedGitDiffEntriesTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/ExtractChangedLinesTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/ExtractCoveredLinesTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/FindCommonLinesTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/GroupGitDiffAndCoverageEntriesByFilePathAndTestCaseTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/InsertFileContentsTest.java delete mode 100644 src/test/java/de/tum/cit/aet/artemis/programming/hestia/util/TestwiseCoverageTestUtil.java rename src/test/java/de/tum/cit/aet/artemis/programming/{hestia/util/HestiaUtilTestService.java => util/ProgrammingUtilTestService.java} (96%) delete mode 100644 src/test/javascript/spec/component/exercise-hint/manage/exercise-hint-detail.component.spec.ts delete mode 100644 src/test/javascript/spec/component/exercise-hint/manage/exercise-hint-update.component.spec.ts delete mode 100644 src/test/javascript/spec/component/exercise-hint/manage/exercise-hint.component.spec.ts delete mode 100644 src/test/javascript/spec/component/exercise-hint/participate/exercise-hint-button-overlay.component.spec.ts delete mode 100644 src/test/javascript/spec/component/exercise-hint/participate/exercise-hint-expandable.component.spec.ts delete mode 100644 src/test/javascript/spec/component/exercise-hint/shared/code-hint-container.component.spec.ts delete mode 100644 src/test/javascript/spec/component/exercise-hint/shared/solution-entry.component.spec.ts rename src/test/javascript/spec/component/{hestia => }/git-diff-report/git-diff-file-panel-title.component.spec.ts (89%) rename src/test/javascript/spec/component/{hestia => }/git-diff-report/git-diff-file-panel.component.spec.ts (85%) rename src/test/javascript/spec/component/{hestia => }/git-diff-report/git-diff-file.component.spec.ts (83%) rename src/test/javascript/spec/component/{hestia => }/git-diff-report/git-diff-line-stat.component.spec.ts (81%) rename src/test/javascript/spec/component/{hestia => }/git-diff-report/git-diff-modal.component.spec.ts (91%) rename src/test/javascript/spec/component/{hestia => }/git-diff-report/git-diff-report.component.spec.ts (90%) delete mode 100644 src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-overview.component.spec.ts delete mode 100644 src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-status.component.spec.ts delete mode 100644 src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-step.component.spec.ts delete mode 100644 src/test/javascript/spec/component/hestia/generation-overview/coverage-generation-step.component.spec.ts delete mode 100644 src/test/javascript/spec/component/hestia/generation-overview/diff-generation-step.component.spec.ts delete mode 100644 src/test/javascript/spec/component/hestia/generation-overview/manual-solution-entry-creation-modal.component.spec.ts delete mode 100644 src/test/javascript/spec/component/hestia/generation-overview/solution-entry-details-modal.component.spec.ts delete mode 100644 src/test/javascript/spec/component/hestia/generation-overview/solution-entry-generation-step.component.spec.ts delete mode 100644 src/test/javascript/spec/component/hestia/testwise-coverage-report/testwise-coverage-file.component.spec.ts delete mode 100644 src/test/javascript/spec/component/hestia/testwise-coverage-report/testwise-coverage-report.component.spec.ts delete mode 100644 src/test/javascript/spec/helpers/mocks/service/mock-exercise-hint.service.ts delete mode 100644 src/test/javascript/spec/service/exercise-hint.service.spec.ts diff --git a/.github/issue-labeler.yml b/.github/issue-labeler.yml index 34ca1e28321c..8fc424b1e785 100644 --- a/.github/issue-labeler.yml +++ b/.github/issue-labeler.yml @@ -111,7 +111,6 @@ programming: - build plan - code hint - \b(? -
LocalCIBuildJobQueueItem
id: String
name: String
buildAgentAddress: String
participationId: long
courseId: long
exerciseId: long
retryCount: int
priority: int
status: BuildJobResult
RepositoryInfo
repositoryName: String
repositoryType: RepositoryType
triggeredByPushTo: RepositoryType
assignmentRepositoryUri: String
testRepositoryUri: String
solutionRepositoryUri: String
auxiliaryRepositoryUris: String []
auxiliaryRepositoryCheckoutDirectories: String []
<<Enumeration>>
BuildJobResult
SUCCESSFUL
FAILED
CANCELLED
<<Enumeration>>
RepositoryType
TEMPLATE
SOLUTION
TESTS
AUXILIARY
USER
JobTimingInfo
submissionDate: ZonedDateTime
buildStartDate: ZonedDateTime
buildCompletionDate: ZonedDateTime
1
BuildConfig
buildScript: String
dockerImage: String
commitHash: String
commitHash: String
branch: String
programmingLanguage: ProgrammingLanguage
projectType: ProjectType
scaEnabled: boolean
sequentialTestRunsEnabled: boolean
testwiseCoverageEnabled: boolean
resultPaths: String []
1
1
\ No newline at end of file +
LocalCIBuildJobQueueItem
id: String
name: String
buildAgentAddress: String
participationId: long
courseId: long
exerciseId: long
retryCount: int
priority: int
status: BuildJobResult
RepositoryInfo
repositoryName: String
repositoryType: RepositoryType
triggeredByPushTo: RepositoryType
assignmentRepositoryUri: String
testRepositoryUri: String
solutionRepositoryUri: String
auxiliaryRepositoryUris: String []
auxiliaryRepositoryCheckoutDirectories: String []
<<Enumeration>>
BuildJobResult
SUCCESSFUL
FAILED
CANCELLED
<<Enumeration>>
RepositoryType
TEMPLATE
SOLUTION
TESTS
AUXILIARY
USER
JobTimingInfo
submissionDate: ZonedDateTime
buildStartDate: ZonedDateTime
buildCompletionDate: ZonedDateTime
1
BuildConfig
buildScript: String
dockerImage: String
commitHash: String
commitHash: String
branch: String
programmingLanguage: ProgrammingLanguage
projectType: ProjectType
scaEnabled: boolean
sequentialTestRunsEnabled: boolean
resultPaths: String []
1
1
diff --git a/docs/user/exercises/programming-exercise-features.inc b/docs/user/exercises/programming-exercise-features.inc index 18434aaed70f..1b43ab3cae7b 100644 --- a/docs/user/exercises/programming-exercise-features.inc +++ b/docs/user/exercises/programming-exercise-features.inc @@ -54,43 +54,43 @@ Instructors can still use those templates to generate programming exercises and In case a feature has different support for different continuous integration systems, the table shows the differences between Local CI **(L)** and Jenkins **(J)**. (Note that Gitlab CI is experimental and therefore has a limited set of features which are not mentioned here to keep the overview simpler.) - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | Programming Language | Sequential Test Runs | Static Code Analysis | Plagiarism Check | Package Name | Project Type | Solution Repository Checkout | Testwise Coverage Analysis | Auxiliary repositories | - +======================+======================+======================+=====================+==============+==========================================+==============================+============================+========================+ - | Java | yes | yes | yes | yes | Gradle, Maven, J: `DejaGnu`_ | no | J: yes; L: no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | Python | L: yes; J: no | no | yes | no | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | C | no | L: yes; J: no | yes | no | FACT, GCC | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | C (FACT framework) | no | L: yes; J: no | yes | no | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | Haskell | L: yes; J: no | no | no | no | n/a | L: yes; J: no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | Kotlin | yes | no | yes | yes | n/a | no | J: yes; L: no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | VHDL | no | no | no | no | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | Assembler | no | no | no | no | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | Swift | no | yes | yes | yes | Plain (Xcode: not supported) | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | OCaml | no | no | no | no | n/a | yes | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | Rust | no | no | yes | no | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | JavaScript | no | no | yes | no | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | R | no | no | yes | no | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | C++ | no | no | yes | no | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | TypeScript | no | no | yes | no | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | C# | no | no | yes | no | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ - | Go | no | no | yes | yes | n/a | no | no | L: yes, J: no | - +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+----------------------------+------------------------+ + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | Programming Language | Sequential Test Runs | Static Code Analysis | Plagiarism Check | Package Name | Project Type | Solution Repository Checkout | Auxiliary repositories | + +======================+======================+======================+=====================+==============+==========================================+==============================+========================+ + | Java | yes | yes | yes | yes | Gradle, Maven, J: `DejaGnu`_ | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | Python | L: yes; J: no | no | yes | no | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | C | no | L: yes; J: no | yes | no | FACT, GCC | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | C (FACT framework) | no | L: yes; J: no | yes | no | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | Haskell | L: yes; J: no | no | no | no | n/a | L: yes; J: no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | Kotlin | yes | no | yes | yes | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | VHDL | no | no | no | no | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | Assembler | no | no | no | no | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | Swift | no | yes | yes | yes | Plain (Xcode: not supported) | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | OCaml | no | no | no | no | n/a | yes | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | Rust | no | no | yes | no | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | JavaScript | no | no | yes | no | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | R | no | no | yes | no | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | C++ | no | no | yes | no | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | TypeScript | no | no | yes | no | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | C# | no | no | yes | no | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ + | Go | no | no | yes | yes | n/a | no | L: yes, J: no | + +----------------------+----------------------+----------------------+---------------------+--------------+------------------------------------------+------------------------------+------------------------+ - *Sequential Test Runs*: ``Artemis`` can generate a build plan which first executes structural and then behavioral tests. This feature can help students to better concentrate on the immediate challenge at hand. - *Static Code Analysis*: ``Artemis`` can generate a build plan which additionally executes static code analysis tools. @@ -98,8 +98,6 @@ Instructors can still use those templates to generate programming exercises and - *Plagiarism Checks*: ``Artemis`` is able to automatically calculate the similarity between student submissions. A side-by-side view of similar submissions is available to confirm the plagiarism suspicion. - *Package Name*: A package name has to be provided - *Solution Repository Checkout*: Instructors are able to compare a student submission against a sample solution in the solution repository - - *Testwise Coverage Analysis*: ``Artemis`` can generate a build plan which additionally executes a testwise coverage analysis. - ``Artemis`` aggregates the recorded data into different metrics. This feature allows instructors to check which code in the solution submission is how often executed by the test cases. .. note:: Only some ``templates`` for ``LocalCI`` support ``Sequential Test Runs`` at the moment. @@ -107,9 +105,6 @@ Instructors can still use those templates to generate programming exercises and .. note:: Static Code Analysis for ``C`` exercises is only supported for ``LocalCI`` at the moment. -.. note:: - Testwise Coverage Analysis is currently not supported, but will be again supported in future versions. - .. note:: Instructors are still able to extend the generated programming exercises with additional features that are not available in one specific template. diff --git a/docs/user/exercises/programming-exercise-setup.inc b/docs/user/exercises/programming-exercise-setup.inc index 563f65f48361..b95420105763 100644 --- a/docs/user/exercises/programming-exercise-setup.inc +++ b/docs/user/exercises/programming-exercise-setup.inc @@ -106,9 +106,6 @@ Generate programming exercise - **Sequential Test Runs:** Activate this option to first run structural and then behavior tests. This feature allows students to better concentrate on the immediate challenge at hand. Not supported together with static code analysis. Cannot be changed after the exercise creation. - - **Record Testwise Coverage:** Activate this option to record the testwise coverage for the solution repository. - This is necessary when working with Hestia to generate code-based hints. - This option is only available for Java/Kotlin-exercises with non-sequential test runs. - **Customize Build Plan** Activate this option if you want to customize the build plan of your exercise. This feature is available for all programming languages, and works with LocalCI and Jenkins, Artemis provides templates for the build plan configuration. The build plan can also be customized after the exercise creation. diff --git a/gradle/test.gradle b/gradle/test.gradle index 8e84af376e9f..52ef3fe25ea5 100644 --- a/gradle/test.gradle +++ b/gradle/test.gradle @@ -108,13 +108,13 @@ jacocoTestCoverageVerification { counter = "INSTRUCTION" value = "COVEREDRATIO" // TODO: in the future the following value should become higher than 0.92 - minimum = 0.895 + minimum = 0.891 } limit { counter = "CLASS" value = "MISSEDCOUNT" // TODO: in the future the following value should become less than 10 - maximum = 56 + maximum = 55 } } } diff --git a/src/main/java/de/tum/cit/aet/artemis/assessment/domain/Feedback.java b/src/main/java/de/tum/cit/aet/artemis/assessment/domain/Feedback.java index de095b019216..c07579401ff0 100644 --- a/src/main/java/de/tum/cit/aet/artemis/assessment/domain/Feedback.java +++ b/src/main/java/de/tum/cit/aet/artemis/assessment/domain/Feedback.java @@ -79,7 +79,7 @@ public class Feedback extends DomainObject { * null if the feedback was not created by an automatic test case. */ @ManyToOne(fetch = FetchType.LAZY) - @JsonIgnoreProperties({ "tasks", "solutionEntries", "exercise", "coverageEntries" }) + @JsonIgnoreProperties({ "tasks", "exercise" }) private ProgrammingExerciseTestCase testCase; /** diff --git a/src/main/java/de/tum/cit/aet/artemis/assessment/domain/Result.java b/src/main/java/de/tum/cit/aet/artemis/assessment/domain/Result.java index 88d5d44e5271..ad45634f2d74 100644 --- a/src/main/java/de/tum/cit/aet/artemis/assessment/domain/Result.java +++ b/src/main/java/de/tum/cit/aet/artemis/assessment/domain/Result.java @@ -10,10 +10,8 @@ import java.util.ArrayList; import java.util.HashMap; import java.util.List; -import java.util.Map; import java.util.Objects; import java.util.Optional; -import java.util.Set; import java.util.stream.Collectors; import jakarta.persistence.CascadeType; @@ -28,7 +26,6 @@ import jakarta.persistence.OneToMany; import jakarta.persistence.OrderColumn; import jakarta.persistence.Table; -import jakarta.persistence.Transient; import jakarta.validation.constraints.NotNull; import org.apache.commons.lang3.StringUtils; @@ -53,7 +50,6 @@ import de.tum.cit.aet.artemis.exercise.domain.participation.Participation; import de.tum.cit.aet.artemis.exercise.service.ExerciseDateService; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageFileReport; import de.tum.cit.aet.artemis.programming.dto.ResultDTO; import de.tum.cit.aet.artemis.quiz.config.QuizView; import de.tum.cit.aet.artemis.quiz.domain.QuizExercise; @@ -152,13 +148,6 @@ public class Result extends DomainObject implements Comparable { @JsonIgnore private Instant lastModifiedDate; - // This attribute is required to forward the coverage file reports after creating the build result. This is required in order to - // delay referencing the corresponding test cases from the entries because the test cases are not saved in the database - // at this point of time but the required test case name would be lost, otherwise. - @Transient - @JsonIgnore - private Map> fileReportsByTestCaseName; - public ZonedDateTime getCompletionDate() { return completionDate; } @@ -500,14 +489,6 @@ public void setCodeIssueCount(int codeIssueCount) { this.codeIssueCount = Math.min(codeIssueCount, SIZE_OF_UNSIGNED_TINYINT); } - public Map> getCoverageFileReportsByTestCaseName() { - return fileReportsByTestCaseName; - } - - public void setCoverageFileReportsByTestCaseName(Map> fileReportsByTestCaseName) { - this.fileReportsByTestCaseName = fileReportsByTestCaseName; - } - /** * Checks the initialization status of the assessment note before returning. Only a single element is returned instead of the list, * because it is modelled that way on the client-side. Jackson therefore needs a single object for the (de-)serialization. diff --git a/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java b/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java index 1dce0090c001..80d564e4695a 100644 --- a/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java +++ b/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java @@ -67,16 +67,16 @@ import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseStudentParticipation; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; import de.tum.cit.aet.artemis.programming.domain.build.BuildPlanType; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.repository.BuildJobRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseStudentParticipationRepository; import de.tum.cit.aet.artemis.programming.repository.SolutionProgrammingExerciseParticipationRepository; import de.tum.cit.aet.artemis.programming.repository.TemplateProgrammingExerciseParticipationRepository; import de.tum.cit.aet.artemis.programming.service.BuildLogEntryService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseTaskService; +import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseTaskService; @Profile(PROFILE_CORE) @Service diff --git a/src/main/java/de/tum/cit/aet/artemis/assessment/web/open/PublicResultResource.java b/src/main/java/de/tum/cit/aet/artemis/assessment/web/open/PublicResultResource.java index a346594c409f..c6b7a6ceb082 100644 --- a/src/main/java/de/tum/cit/aet/artemis/assessment/web/open/PublicResultResource.java +++ b/src/main/java/de/tum/cit/aet/artemis/assessment/web/open/PublicResultResource.java @@ -30,7 +30,6 @@ import de.tum.cit.aet.artemis.programming.service.ProgrammingMessagingService; import de.tum.cit.aet.artemis.programming.service.ProgrammingTriggerService; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationService; -import de.tum.cit.aet.artemis.programming.service.hestia.TestwiseCoverageService; /** * REST controller for receiving build results. @@ -51,19 +50,15 @@ public class PublicResultResource { private final ResultService resultService; - private final TestwiseCoverageService testwiseCoverageService; - private final ProgrammingTriggerService programmingTriggerService; private final ProgrammingMessagingService programmingMessagingService; public PublicResultResource(Optional continuousIntegrationService, ProgrammingExerciseGradingService programmingExerciseGradingService, - ResultService resultService, TestwiseCoverageService testwiseCoverageService, ProgrammingTriggerService programmingTriggerService, - ProgrammingMessagingService programmingMessagingService) { + ResultService resultService, ProgrammingTriggerService programmingTriggerService, ProgrammingMessagingService programmingMessagingService) { this.continuousIntegrationService = continuousIntegrationService; this.programmingExerciseGradingService = programmingExerciseGradingService; this.resultService = resultService; - this.testwiseCoverageService = testwiseCoverageService; this.programmingTriggerService = programmingTriggerService; this.programmingMessagingService = programmingMessagingService; } @@ -125,12 +120,6 @@ public ResponseEntity processNewProgrammingExerciseResult(@RequestHeader(" // This method will return without triggering the build if the submission is not of type TEST. var programmingSubmission = (ProgrammingSubmission) result.getSubmission(); triggerTemplateBuildIfTestCasesChanged(participation.getProgrammingExercise().getId(), programmingSubmission); - - // the test cases and the submission have been saved to the database previously, therefore we can add the reference to the coverage reports - if (Boolean.TRUE.equals(participation.getProgrammingExercise().getBuildConfig().isTestwiseCoverageEnabled()) && Boolean.TRUE.equals(result.isSuccessful())) { - testwiseCoverageService.createTestwiseCoverageReport(result.getCoverageFileReportsByTestCaseName(), participation.getProgrammingExercise(), - programmingSubmission); - } } programmingMessagingService.notifyUserAboutNewResult(result, participation); diff --git a/src/main/java/de/tum/cit/aet/artemis/atlas/service/LearningObjectImportService.java b/src/main/java/de/tum/cit/aet/artemis/atlas/service/LearningObjectImportService.java index d20c671193de..19f5081bb032 100644 --- a/src/main/java/de/tum/cit/aet/artemis/atlas/service/LearningObjectImportService.java +++ b/src/main/java/de/tum/cit/aet/artemis/atlas/service/LearningObjectImportService.java @@ -55,7 +55,7 @@ import de.tum.cit.aet.artemis.plagiarism.service.PlagiarismDetectionConfigHelper; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseTaskRepository; +import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTaskRepository; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseImportService; import de.tum.cit.aet.artemis.quiz.domain.QuizExercise; import de.tum.cit.aet.artemis.quiz.repository.QuizExerciseRepository; @@ -261,7 +261,6 @@ else if (foundByShortName.isPresent()) { private void clearProgrammingExerciseAttributes(ProgrammingExercise programmingExercise) { programmingExercise.setTasks(null); - programmingExercise.setExerciseHints(new HashSet<>()); programmingExercise.setTestCases(new HashSet<>()); programmingExercise.setStaticCodeAnalysisCategories(new HashSet<>()); programmingExercise.setTeams(new HashSet<>()); diff --git a/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/BuildConfig.java b/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/BuildConfig.java index 34d139aa6f19..4f8df8d2f49d 100644 --- a/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/BuildConfig.java +++ b/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/BuildConfig.java @@ -11,12 +11,12 @@ // NOTE: this data structure is used in shared code between core and build agent nodes. Changing it requires that the shared data structures in Hazelcast (or potentially Redis) // in the future are migrated or cleared. Changes should be communicated in release notes as potentially breaking changes. +// TODO: reduce the amount of parameters and combine some in smaller record DTOs @JsonIgnoreProperties(ignoreUnknown = true) @JsonInclude(JsonInclude.Include.NON_EMPTY) public record BuildConfig(String buildScript, String dockerImage, String commitHashToBuild, String assignmentCommitHash, String testCommitHash, String branch, - ProgrammingLanguage programmingLanguage, ProjectType projectType, boolean scaEnabled, boolean sequentialTestRunsEnabled, boolean testwiseCoverageEnabled, - List resultPaths, int timeoutSeconds, String assignmentCheckoutPath, String testCheckoutPath, String solutionCheckoutPath, DockerRunConfig dockerRunConfig) - implements Serializable { + ProgrammingLanguage programmingLanguage, ProjectType projectType, boolean scaEnabled, boolean sequentialTestRunsEnabled, List resultPaths, int timeoutSeconds, + String assignmentCheckoutPath, String testCheckoutPath, String solutionCheckoutPath, DockerRunConfig dockerRunConfig) implements Serializable { @Override public String dockerImage() { diff --git a/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/BuildResult.java b/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/BuildResult.java index 9d78a5fe4984..b9ab91a21301 100644 --- a/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/BuildResult.java +++ b/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/BuildResult.java @@ -3,20 +3,15 @@ import java.io.Serializable; import java.time.ZonedDateTime; import java.util.ArrayList; -import java.util.Collections; import java.util.List; - -import org.springframework.util.ObjectUtils; +import java.util.Objects; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonInclude; import de.tum.cit.aet.artemis.programming.domain.build.BuildLogEntry; -import de.tum.cit.aet.artemis.programming.dto.AbstractBuildResultNotificationDTO; -import de.tum.cit.aet.artemis.programming.dto.BuildJobDTOInterface; +import de.tum.cit.aet.artemis.programming.dto.BuildResultNotification; import de.tum.cit.aet.artemis.programming.dto.StaticCodeAnalysisReportDTO; -import de.tum.cit.aet.artemis.programming.dto.TestCaseBaseDTO; -import de.tum.cit.aet.artemis.programming.service.ci.notification.dto.TestwiseCoverageReportDTO; /** * Represents all the information returned by the local CI system about a build. @@ -27,88 +22,30 @@ // in the future are migrated or cleared. Changes should be communicated in release notes as potentially breaking changes. @JsonIgnoreProperties(ignoreUnknown = true) @JsonInclude(JsonInclude.Include.NON_EMPTY) -// TODO: this should be a record in the future -public class BuildResult extends AbstractBuildResultNotificationDTO implements Serializable { - - private final String assignmentRepoBranchName; - - private final String assignmentRepoCommitHash; - - private final String testsRepoCommitHash; - - private final boolean isBuildSuccessful; - - private final ZonedDateTime buildRunDate; - - private final List jobs; - - private List buildLogEntries = new ArrayList<>(); - - private final List staticCodeAnalysisReports; - - private boolean hasLogs = false; - - public BuildResult(String assignmentRepoBranchName, String assignmentRepoCommitHash, String testsRepoCommitHash, boolean isBuildSuccessful, ZonedDateTime buildRunDate, - List jobs, List staticCodeAnalysisReports) { - this.assignmentRepoBranchName = assignmentRepoBranchName; - this.assignmentRepoCommitHash = assignmentRepoCommitHash; - this.testsRepoCommitHash = testsRepoCommitHash; - this.isBuildSuccessful = isBuildSuccessful; - this.buildRunDate = buildRunDate; - this.jobs = jobs; - this.staticCodeAnalysisReports = staticCodeAnalysisReports; - } - - public BuildResult(String assignmentRepoBranchName, String assignmentRepoCommitHash, String testsRepoCommitHash, boolean isBuildSuccessful) { - this.assignmentRepoBranchName = assignmentRepoBranchName; - this.assignmentRepoCommitHash = assignmentRepoCommitHash; - this.testsRepoCommitHash = testsRepoCommitHash; - this.isBuildSuccessful = isBuildSuccessful; - this.buildRunDate = ZonedDateTime.now(); - this.jobs = new ArrayList<>(); - this.staticCodeAnalysisReports = new ArrayList<>(); - } +public record BuildResult(String assignmentRepoBranchName, String assignmentRepoCommitHash, String testsRepoCommitHash, boolean isBuildSuccessful, ZonedDateTime buildRunDate, + List jobs, List buildLogEntries, List staticCodeAnalysisReports, boolean hasLogs) + implements BuildResultNotification, Serializable { - @Override - public ZonedDateTime getBuildRunDate() { - return buildRunDate; - } - - @Override - protected String getCommitHashFromAssignmentRepo() { - if (ObjectUtils.isEmpty(assignmentRepoCommitHash)) { - return null; - } - return assignmentRepoCommitHash; - } - - @Override - protected String getCommitHashFromTestsRepo() { - if (ObjectUtils.isEmpty(testsRepoCommitHash)) { - return null; - } - return testsRepoCommitHash; + public BuildResult { + buildRunDate = Objects.requireNonNullElse(buildRunDate, ZonedDateTime.now()); + jobs = Objects.requireNonNullElse(jobs, new ArrayList<>()); + staticCodeAnalysisReports = Objects.requireNonNullElse(staticCodeAnalysisReports, new ArrayList<>()); + buildLogEntries = Objects.requireNonNullElse(buildLogEntries, new ArrayList<>()); + hasLogs = !buildLogEntries.isEmpty(); } - @Override - public String getBranchNameFromAssignmentRepo() { - return assignmentRepoBranchName; - } - - @Override - public boolean isBuildSuccessful() { - return isBuildSuccessful; + public BuildResult(String branch, String assignmentRepoCommitHash, String testsRepoCommitHash, List buildLogs, boolean isBuildSuccessful) { + this(branch, assignmentRepoCommitHash, testsRepoCommitHash, isBuildSuccessful, null, null, buildLogs, null, buildLogs != null && !buildLogs.isEmpty()); } @Override - public Double getBuildScore() { + public Double buildScore() { // the real score is calculated in the grading service return 0D; } /** - * Local CI does not support checking for artifacts as of now. - * TODO LOCALVC_CI: Figure out in the build process whether an artifact was created, and return true here if yes. + * NOTE: Local CI does not support checking for artifacts as of now. * * @return will always return false because LocalCI does not support checking for artifacts. */ @@ -117,81 +54,9 @@ public boolean hasArtifact() { return false; } - @Override - public boolean hasLogs() { - return hasLogs; - } - @Override public List extractBuildLogs() { // convert the buildLogEntry DTOs to BuildLogEntry objects return buildLogEntries.stream().map(log -> new BuildLogEntry(log.time(), log.log())).toList(); } - - /** - * Setter for the buildLogEntries - * - * @param buildLogEntries the buildLogEntries to be set - */ - public void setBuildLogEntries(List buildLogEntries) { - this.buildLogEntries = buildLogEntries; - hasLogs = true; - } - - @Override - public List getBuildJobs() { - return jobs; - } - - @Override - public List getStaticCodeAnalysisReports() { - return staticCodeAnalysisReports; - } - - @Override - public List getTestwiseCoverageReports() { - // TODO LOCALVC_CI: Implement testwise coverage and return the reports here. - return Collections.emptyList(); - } - - /** - * Represents all the information returned by the local CI system about a job. - * In the current implementation of local CI, there is always one job per build. - * - * @param failedTests list of failed tests. - * @param successfulTests list of successful tests. - */ - @JsonInclude(JsonInclude.Include.NON_EMPTY) - public record LocalCIJobDTO(List failedTests, List successfulTests) implements BuildJobDTOInterface, Serializable { - - @Override - public List getFailedTests() { - return failedTests; - } - - @Override - public List getSuccessfulTests() { - return successfulTests; - } - } - - /** - * Represents the information about one test case, including the test case's name and potential error messages that indicate what went wrong. - * - * @param name name of the test case. - * @param errors list of error messages. - */ - @JsonInclude(JsonInclude.Include.NON_EMPTY) - public record LocalCITestJobDTO(String name, List errors) implements TestCaseBaseDTO, Serializable { - - @Override - public String getName() { - return name; - } - - @Override - public List getTestMessages() { - return errors; - } - } } diff --git a/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/LocalCIJobDTO.java b/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/LocalCIJobDTO.java new file mode 100644 index 000000000000..27410b0b60f7 --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/LocalCIJobDTO.java @@ -0,0 +1,26 @@ +package de.tum.cit.aet.artemis.buildagent.dto; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.List; +import java.util.Objects; + +import com.fasterxml.jackson.annotation.JsonInclude; + +import de.tum.cit.aet.artemis.programming.dto.BuildJobInterface; + +/** + * Represents all the information returned by the local CI system about a job. + * In the current implementation of local CI, there is always one job per build. + * + * @param failedTests list of failed tests. + * @param successfulTests list of successful tests. + */ +@JsonInclude(JsonInclude.Include.NON_EMPTY) +public record LocalCIJobDTO(List failedTests, List successfulTests) implements BuildJobInterface, Serializable { + + public LocalCIJobDTO { + failedTests = Objects.requireNonNullElse(failedTests, new ArrayList<>()); + successfulTests = Objects.requireNonNullElse(successfulTests, new ArrayList<>()); + } +} diff --git a/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/LocalCITestJobDTO.java b/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/LocalCITestJobDTO.java new file mode 100644 index 000000000000..3232501a8737 --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/buildagent/dto/LocalCITestJobDTO.java @@ -0,0 +1,18 @@ +package de.tum.cit.aet.artemis.buildagent.dto; + +import java.io.Serializable; +import java.util.List; + +import com.fasterxml.jackson.annotation.JsonInclude; + +import de.tum.cit.aet.artemis.programming.dto.TestCaseBase; + +/** + * Represents the information about one test case, including the test case's name and potential error messages that indicate what went wrong. + * + * @param name name of the test case. + * @param testMessages list of error messages. + */ +@JsonInclude(JsonInclude.Include.NON_EMPTY) +public record LocalCITestJobDTO(String name, List testMessages) implements TestCaseBase, Serializable { +} diff --git a/src/main/java/de/tum/cit/aet/artemis/buildagent/service/BuildJobExecutionService.java b/src/main/java/de/tum/cit/aet/artemis/buildagent/service/BuildJobExecutionService.java index e7fd13c7b97e..c77f26b426d1 100644 --- a/src/main/java/de/tum/cit/aet/artemis/buildagent/service/BuildJobExecutionService.java +++ b/src/main/java/de/tum/cit/aet/artemis/buildagent/service/BuildJobExecutionService.java @@ -40,7 +40,10 @@ import com.github.dockerjava.api.exception.NotFoundException; import de.tum.cit.aet.artemis.buildagent.dto.BuildJobQueueItem; +import de.tum.cit.aet.artemis.buildagent.dto.BuildLogDTO; import de.tum.cit.aet.artemis.buildagent.dto.BuildResult; +import de.tum.cit.aet.artemis.buildagent.dto.LocalCIJobDTO; +import de.tum.cit.aet.artemis.buildagent.dto.LocalCITestJobDTO; import de.tum.cit.aet.artemis.core.exception.EntityNotFoundException; import de.tum.cit.aet.artemis.core.exception.GitException; import de.tum.cit.aet.artemis.core.exception.LocalCIException; @@ -327,9 +330,9 @@ private BuildResult runScriptAndParseResults(BuildJobQueueItem buildJob, String try { testResultsTarInputStream = buildJobContainerService.getArchiveFromContainer(containerId, LOCALCI_WORKING_DIRECTORY + LOCALCI_RESULTS_DIRECTORY); + var buildLogs = buildLogsMap.getAndTruncateBuildLogs(buildJob.id()); buildResult = parseTestResults(testResultsTarInputStream, buildJob.buildConfig().branch(), assignmentRepoCommitHash, testRepoCommitHash, buildCompletedDate, - buildJob.id()); - buildResult.setBuildLogEntries(buildLogsMap.getAndTruncateBuildLogs(buildJob.id())); + buildJob.id(), buildLogs); } catch (NotFoundException e) { msg = "Could not find test results in container " + containerName; @@ -389,10 +392,10 @@ private BuildResult runScriptAndParseResults(BuildJobQueueItem buildJob, String // --- Helper methods ---- private BuildResult parseTestResults(TarArchiveInputStream testResultsTarInputStream, String assignmentRepoBranchName, String assignmentRepoCommitHash, - String testsRepoCommitHash, ZonedDateTime buildCompletedDate, String buildJobId) throws IOException { + String testsRepoCommitHash, ZonedDateTime buildCompletedDate, String buildJobId, List buildLogs) throws IOException { - List failedTests = new ArrayList<>(); - List successfulTests = new ArrayList<>(); + List failedTests = new ArrayList<>(); + List successfulTests = new ArrayList<>(); List staticCodeAnalysisReports = new ArrayList<>(); TarArchiveEntry tarEntry; @@ -434,7 +437,7 @@ private BuildResult parseTestResults(TarArchiveInputStream testResultsTarInputSt } return constructBuildResult(failedTests, successfulTests, assignmentRepoBranchName, assignmentRepoCommitHash, testsRepoCommitHash, !failedTests.isEmpty(), - buildCompletedDate, staticCodeAnalysisReports); + buildCompletedDate, staticCodeAnalysisReports, buildLogs); } private boolean isValidTestResultFile(TarArchiveEntry tarArchiveEntry) { @@ -499,7 +502,7 @@ private String readTarEntryContent(TarArchiveInputStream tarArchiveInputStream) */ private BuildResult constructFailedBuildResult(String assignmentRepoBranchName, @Nullable String assignmentRepoCommitHash, @Nullable String testsRepoCommitHash, ZonedDateTime buildRunDate) { - return constructBuildResult(List.of(), List.of(), assignmentRepoBranchName, assignmentRepoCommitHash, testsRepoCommitHash, false, buildRunDate, List.of()); + return constructBuildResult(List.of(), List.of(), assignmentRepoBranchName, assignmentRepoCommitHash, testsRepoCommitHash, false, buildRunDate, List.of(), null); } /** @@ -513,14 +516,15 @@ private BuildResult constructFailedBuildResult(String assignmentRepoBranchName, * @param isBuildSuccessful Whether the build was successful or not. * @param buildRunDate The date when the build was completed. * @param staticCodeAnalysisReports The static code analysis reports + * @param buildLogs the build logs * @return a {@link BuildResult} */ - private BuildResult constructBuildResult(List failedTests, List successfulTests, String assignmentRepoBranchName, + private BuildResult constructBuildResult(List failedTests, List successfulTests, String assignmentRepoBranchName, String assignmentRepoCommitHash, String testsRepoCommitHash, boolean isBuildSuccessful, ZonedDateTime buildRunDate, - List staticCodeAnalysisReports) { - BuildResult.LocalCIJobDTO job = new BuildResult.LocalCIJobDTO(failedTests, successfulTests); - - return new BuildResult(assignmentRepoBranchName, assignmentRepoCommitHash, testsRepoCommitHash, isBuildSuccessful, buildRunDate, List.of(job), staticCodeAnalysisReports); + List staticCodeAnalysisReports, List buildLogs) { + LocalCIJobDTO job = new LocalCIJobDTO(failedTests, successfulTests); + return new BuildResult(assignmentRepoBranchName, assignmentRepoCommitHash, testsRepoCommitHash, isBuildSuccessful, buildRunDate, List.of(job), buildLogs, + staticCodeAnalysisReports, true); } private Path cloneRepository(VcsRepositoryUri repositoryUri, @Nullable String commitHash, boolean checkout, String buildJobId) { diff --git a/src/main/java/de/tum/cit/aet/artemis/buildagent/service/SharedQueueProcessingService.java b/src/main/java/de/tum/cit/aet/artemis/buildagent/service/SharedQueueProcessingService.java index cc9c853ec806..c833a52704ec 100644 --- a/src/main/java/de/tum/cit/aet/artemis/buildagent/service/SharedQueueProcessingService.java +++ b/src/main/java/de/tum/cit/aet/artemis/buildagent/service/SharedQueueProcessingService.java @@ -463,8 +463,7 @@ private void processBuild(BuildJobQueueItem buildJob) { buildLogsMap.removeBuildLogs(buildJob.id()); BuildResult failedResult = new BuildResult(buildJob.buildConfig().branch(), buildJob.buildConfig().assignmentCommitHash(), buildJob.buildConfig().testCommitHash(), - false); - failedResult.setBuildLogEntries(buildLogs); + buildLogs, false); ResultQueueItem resultQueueItem = new ResultQueueItem(failedResult, job, buildLogs, ex); if (processResults.get()) { diff --git a/src/main/java/de/tum/cit/aet/artemis/buildagent/service/TestResultXmlParser.java b/src/main/java/de/tum/cit/aet/artemis/buildagent/service/TestResultXmlParser.java index ef9190911239..51603dd8f20f 100644 --- a/src/main/java/de/tum/cit/aet/artemis/buildagent/service/TestResultXmlParser.java +++ b/src/main/java/de/tum/cit/aet/artemis/buildagent/service/TestResultXmlParser.java @@ -12,7 +12,7 @@ import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlProperty; import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlText; -import de.tum.cit.aet.artemis.buildagent.dto.BuildResult; +import de.tum.cit.aet.artemis.buildagent.dto.LocalCITestJobDTO; public class TestResultXmlParser { @@ -75,8 +75,7 @@ public class TestResultXmlParser { * @param successfulTests A list of successful tests. This list will be populated by the method. * @throws IOException If an I/O error occurs while reading the test result file. */ - public static void processTestResultFile(String testResultFileString, List failedTests, List successfulTests) - throws IOException { + public static void processTestResultFile(String testResultFileString, List failedTests, List successfulTests) throws IOException { testResultFileString = testResultFileString.replaceAll(INVALID_XML_CHARS, ""); // The root element can be or @@ -105,7 +104,7 @@ public static void processTestResultFile(String testResultFileString, List failedTests, List successfulTests, TestSuite suite) { + private static void processTopLevelTestSuite(List failedTests, List successfulTests, TestSuite suite) { processTestSuiteWithNamePrefix(suite, failedTests, successfulTests, ""); } @@ -117,8 +116,7 @@ private static void processTopLevelTestSuite(List * @param successfulTests A list of successful tests. This list will be populated by the method. * @param outerNamePrefix The name prefix for the test suite, derived from its parent suites. */ - private static void processInnerTestSuite(TestSuite testSuite, List failedTests, List successfulTests, - String outerNamePrefix) { + private static void processInnerTestSuite(TestSuite testSuite, List failedTests, List successfulTests, String outerNamePrefix) { // namePrefix recursively accumulates all parent testsuite names seperated with dots String namePrefix; if (testSuite.name() != null) { @@ -140,18 +138,17 @@ private static void processInnerTestSuite(TestSuite testSuite, List failedTests, List successfulTests, - String namePrefix) { + private static void processTestSuiteWithNamePrefix(TestSuite testSuite, List failedTests, List successfulTests, String namePrefix) { for (TestCase testCase : testSuite.testCases()) { if (testCase.isSkipped()) { continue; } Failure failure = testCase.extractFailure(); if (failure != null) { - failedTests.add(new BuildResult.LocalCITestJobDTO(namePrefix + testCase.name(), List.of(failure.extractMessage()))); + failedTests.add(new LocalCITestJobDTO(namePrefix + testCase.name(), List.of(failure.extractMessage()))); } else { - successfulTests.add(new BuildResult.LocalCITestJobDTO(namePrefix + testCase.name(), List.of())); + successfulTests.add(new LocalCITestJobDTO(namePrefix + testCase.name(), List.of())); } } @@ -190,11 +187,11 @@ private boolean isSkipped() { private Failure extractFailure() { return failure != null ? failure : error; } - } - // Intentionally empty record to represent the skipped tag () - @JsonIgnoreProperties(ignoreUnknown = true) - record Skip() { + // Intentionally empty record to represent the skipped tag () + @JsonIgnoreProperties(ignoreUnknown = true) + record Skip() { + } } // Due to issues with Jackson this currently cannot be a record. diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/service/conversation/ConversationService.java b/src/main/java/de/tum/cit/aet/artemis/communication/service/conversation/ConversationService.java index 6dd36e5e9bac..b9fbbf97715c 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/service/conversation/ConversationService.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/service/conversation/ConversationService.java @@ -164,11 +164,11 @@ public Conversation loadConversationWithParticipantsIfGroupChat(Long conversatio } /** - * Gets the conversation in a course for which the user is a member + * Gets the conversations in a course for which the user is a member * * @param course the course * @param requestingUser the user for which the conversations are requested - * @return the conversation in the course for which the user is a member + * @return list of conversations in the course for which the user is a member */ public List getConversationsOfUser(Course course, User requestingUser) { var conversationsOfUser = new ArrayList(); diff --git a/src/main/java/de/tum/cit/aet/artemis/core/repository/UserRepository.java b/src/main/java/de/tum/cit/aet/artemis/core/repository/UserRepository.java index 0d3280cf5d96..6eb94330acff 100644 --- a/src/main/java/de/tum/cit/aet/artemis/core/repository/UserRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/core/repository/UserRepository.java @@ -920,7 +920,7 @@ private String getCurrentUserLogin() { } /** - * Get user with user groups and authorities with the username (i.e. user.getLogin() or principal.getName()) + * Get user with user groups and authorities with the username (i.e. user.getLogin() or principal.name()) * * @param username the username of the user who should be retrieved from the database * @return the user that belongs to the given principal with eagerly loaded groups and authorities diff --git a/src/main/java/de/tum/cit/aet/artemis/core/service/TitleCacheEvictionService.java b/src/main/java/de/tum/cit/aet/artemis/core/service/TitleCacheEvictionService.java index 0ee7cceac695..67c43ec410c9 100644 --- a/src/main/java/de/tum/cit/aet/artemis/core/service/TitleCacheEvictionService.java +++ b/src/main/java/de/tum/cit/aet/artemis/core/service/TitleCacheEvictionService.java @@ -29,7 +29,6 @@ import de.tum.cit.aet.artemis.exercise.domain.Exercise; import de.tum.cit.aet.artemis.lecture.domain.Lecture; import de.tum.cit.aet.artemis.modeling.domain.ApollonDiagram; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; /** * Listens to Hibernate events and invalidates the cached titles of an entity if the title changed. @@ -121,15 +120,6 @@ else if (entity instanceof ApollonDiagram diagram) { else if (entity instanceof Exam exam) { evictIdFromCache("examTitle", exam.getId()); } - else if (entity instanceof ExerciseHint hint) { - if (hint.getExercise() == null) { - log.warn("Unable to clear title of exercise hint {}: Exercise not present", hint.getId()); - return; - } - - var combinedId = hint.getExercise().getId() + "-" + hint.getId(); - evictIdFromCache("exerciseHintTitle", combinedId); - } else if (entity instanceof ExerciseGroup exerciseGroup) { if (!Hibernate.isInitialized(exerciseGroup.getExercises())) { log.warn("Unable to clear title of exercises from exercise group {}: Exercises not initialized", exerciseGroup.getId()); diff --git a/src/main/java/de/tum/cit/aet/artemis/exam/service/ExamImportService.java b/src/main/java/de/tum/cit/aet/artemis/exam/service/ExamImportService.java index 58fcf03cc97a..507a5c706975 100644 --- a/src/main/java/de/tum/cit/aet/artemis/exam/service/ExamImportService.java +++ b/src/main/java/de/tum/cit/aet/artemis/exam/service/ExamImportService.java @@ -33,7 +33,7 @@ import de.tum.cit.aet.artemis.modeling.service.ModelingExerciseImportService; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseTaskRepository; +import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTaskRepository; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseImportService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseService; import de.tum.cit.aet.artemis.quiz.domain.QuizExercise; diff --git a/src/main/java/de/tum/cit/aet/artemis/exercise/dto/ExerciseDetailsDTO.java b/src/main/java/de/tum/cit/aet/artemis/exercise/dto/ExerciseDetailsDTO.java index eaa46988ebda..c577b573ee6f 100644 --- a/src/main/java/de/tum/cit/aet/artemis/exercise/dto/ExerciseDetailsDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/exercise/dto/ExerciseDetailsDTO.java @@ -1,15 +1,11 @@ package de.tum.cit.aet.artemis.exercise.dto; -import java.util.Set; - import com.fasterxml.jackson.annotation.JsonInclude; import de.tum.cit.aet.artemis.exercise.domain.Exercise; import de.tum.cit.aet.artemis.iris.dto.IrisCombinedSettingsDTO; import de.tum.cit.aet.artemis.plagiarism.dto.PlagiarismCaseInfoDTO; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; @JsonInclude(JsonInclude.Include.NON_EMPTY) -public record ExerciseDetailsDTO(Exercise exercise, IrisCombinedSettingsDTO irisSettings, PlagiarismCaseInfoDTO plagiarismCaseInfo, Set availableExerciseHints, - Set activatedExerciseHints) { +public record ExerciseDetailsDTO(Exercise exercise, IrisCombinedSettingsDTO irisSettings, PlagiarismCaseInfoDTO plagiarismCaseInfo) { } diff --git a/src/main/java/de/tum/cit/aet/artemis/exercise/service/ParticipationService.java b/src/main/java/de/tum/cit/aet/artemis/exercise/service/ParticipationService.java index a8c4d3507813..3f3d8d43257e 100644 --- a/src/main/java/de/tum/cit/aet/artemis/exercise/service/ParticipationService.java +++ b/src/main/java/de/tum/cit/aet/artemis/exercise/service/ParticipationService.java @@ -50,7 +50,6 @@ import de.tum.cit.aet.artemis.programming.repository.BuildLogStatisticsEntryRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseStudentParticipationRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CoverageReportRepository; import de.tum.cit.aet.artemis.programming.service.BuildLogEntryService; import de.tum.cit.aet.artemis.programming.service.GitService; import de.tum.cit.aet.artemis.programming.service.ParticipationVcsAccessTokenService; @@ -93,8 +92,6 @@ public class ParticipationService { private final ResultService resultService; - private final CoverageReportRepository coverageReportRepository; - private final BuildLogStatisticsEntryRepository buildLogStatisticsEntryRepository; private final ParticipantScoreRepository participantScoreRepository; @@ -115,10 +112,9 @@ public ParticipationService(GitService gitService, Optional localCISharedBuildJobQueueService, ProfileService profileService, - ParticipationVcsAccessTokenService participationVCSAccessTokenService, CompetencyProgressApi competencyProgressApi) { + BuildLogStatisticsEntryRepository buildLogStatisticsEntryRepository, ParticipantScoreRepository participantScoreRepository, + StudentScoreRepository studentScoreRepository, TeamScoreRepository teamScoreRepository, Optional localCISharedBuildJobQueueService, + ProfileService profileService, ParticipationVcsAccessTokenService participationVCSAccessTokenService, CompetencyProgressApi competencyProgressApi) { this.gitService = gitService; this.continuousIntegrationService = continuousIntegrationService; this.versionControlService = versionControlService; @@ -131,7 +127,6 @@ public ParticipationService(GitService gitService, Optional irisSettingsService, - PlagiarismCaseService plagiarismCaseService, ExerciseHintService exerciseHintService) { + PlagiarismCaseService plagiarismCaseService) { this.exerciseService = exerciseService; this.exerciseDeletionService = exerciseDeletionService; this.participationService = participationService; @@ -128,7 +124,6 @@ public ExerciseResource(ExerciseService exerciseService, ExerciseDeletionService this.examAccessService = examAccessService; this.irisSettingsService = irisSettingsService; this.plagiarismCaseService = plagiarismCaseService; - this.exerciseHintService = exerciseHintService; } /** @@ -349,13 +344,7 @@ public ResponseEntity getExerciseDetails(@PathVariable Long .orElse(null); PlagiarismCaseInfoDTO plagiarismCaseInfo = plagiarismCaseService.getPlagiarismCaseInfoForExerciseAndUser(exercise.getId(), user.getId()).orElse(null); - if (exercise instanceof ProgrammingExercise programmingExercise) { - Set activatedExerciseHints = exerciseHintService.getActivatedExerciseHints(programmingExercise, user); - Set availableExerciseHints = exerciseHintService.getAvailableExerciseHints(programmingExercise, user); - return ResponseEntity.ok(new ExerciseDetailsDTO(exercise, irisSettings, plagiarismCaseInfo, availableExerciseHints, activatedExerciseHints)); - } - - return ResponseEntity.ok(new ExerciseDetailsDTO(exercise, irisSettings, plagiarismCaseInfo, null, null)); + return ResponseEntity.ok(new ExerciseDetailsDTO(exercise, irisSettings, plagiarismCaseInfo)); } /** diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Scopes.java b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Scopes.java index ff34ce2fc460..735bbf7d15d3 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/dto/Scopes.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/dto/Scopes.java @@ -1,6 +1,6 @@ package de.tum.cit.aet.artemis.lti.dto; -public class Scopes { +public final class Scopes { public static final String AGS_SCORE = "https://purl.imsglobal.org/spec/lti-ags/scope/score"; diff --git a/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ProgrammingPlagiarismDetectionService.java b/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ProgrammingPlagiarismDetectionService.java index a1b16c64ade3..cec8919e81dd 100644 --- a/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ProgrammingPlagiarismDetectionService.java +++ b/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ProgrammingPlagiarismDetectionService.java @@ -59,8 +59,8 @@ import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; import de.tum.cit.aet.artemis.programming.service.GitService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseExportService; +import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseGitDiffReportService; import de.tum.cit.aet.artemis.programming.service.UriService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseGitDiffReportService; @Profile(PROFILE_CORE) @Service diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExercise.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExercise.java index c2a4666c7c1b..63d247ffdfc8 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExercise.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExercise.java @@ -49,8 +49,6 @@ import de.tum.cit.aet.artemis.exercise.domain.participation.StudentParticipation; import de.tum.cit.aet.artemis.exercise.service.ExerciseDateService; import de.tum.cit.aet.artemis.programming.domain.build.BuildPlanType; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.domain.submissionpolicy.SubmissionPolicy; import de.tum.cit.aet.artemis.programming.service.ProgrammingLanguageFeature; @@ -149,9 +147,6 @@ public String getType() { @Column(name = "project_type", table = "programming_exercise_details") private ProjectType projectType; - @OneToMany(mappedBy = "exercise", cascade = CascadeType.REMOVE, orphanRemoval = true, fetch = FetchType.LAZY) - private Set exerciseHints = new HashSet<>(); - @Column(name = "release_tests_with_example_solution", table = "programming_exercise_details", nullable = false) private boolean releaseTestsWithExampleSolution = false; @@ -815,21 +810,12 @@ public void validateSettingsForFeedbackRequest() { } } - public Set getExerciseHints() { - return exerciseHints; - } - - public void setExerciseHints(Set exerciseHints) { - this.exerciseHints = exerciseHints; - } - /** * {@inheritDoc} */ @Override public void disconnectRelatedEntities() { - Stream.of(exerciseHints, testCases, staticCodeAnalysisCategories).filter(Objects::nonNull).forEach(Collection::clear); - + Stream.of(testCases, staticCodeAnalysisCategories).filter(Objects::nonNull).forEach(Collection::clear); super.disconnectRelatedEntities(); } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseBuildConfig.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseBuildConfig.java index d28e21bb3ad1..1a784bdf060d 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseBuildConfig.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseBuildConfig.java @@ -19,7 +19,7 @@ import com.fasterxml.jackson.core.JsonProcessingException; import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; import de.tum.cit.aet.artemis.programming.service.vcs.AbstractVersionControlService; @Entity @@ -67,9 +67,6 @@ public class ProgrammingExerciseBuildConfig extends DomainObject { @JsonIgnoreProperties("buildConfig") private ProgrammingExercise programmingExercise; - @Column(name = "testwise_coverage_enabled") - private boolean testwiseCoverageEnabled; - @Nullable @Column(name = "theia_image") private String theiaImage; @@ -98,7 +95,6 @@ public ProgrammingExerciseBuildConfig(ProgrammingExerciseBuildConfig originalBui this.setDockerFlags(originalBuildConfig.getDockerFlags()); this.setSequentialTestRuns(originalBuildConfig.hasSequentialTestRuns()); this.setBuildScript(originalBuildConfig.getBuildScript()); - this.setTestwiseCoverageEnabled(originalBuildConfig.isTestwiseCoverageEnabled()); this.setTimeoutSeconds(originalBuildConfig.getTimeoutSeconds()); this.setTheiaImage(originalBuildConfig.getTheiaImage()); this.setAllowBranching(originalBuildConfig.isAllowBranching()); @@ -247,14 +243,6 @@ public void filterSensitiveInformation() { setBuildScript(null); } - public boolean isTestwiseCoverageEnabled() { - return testwiseCoverageEnabled; - } - - public void setTestwiseCoverageEnabled(boolean testwiseCoverageEnabled) { - this.testwiseCoverageEnabled = testwiseCoverageEnabled; - } - public ProgrammingExercise getProgrammingExercise() { return programmingExercise; } @@ -296,7 +284,7 @@ public void setSolutionCheckoutPath(String solutionCheckoutPath) { public String toString() { return "BuildJobConfig{" + "id=" + getId() + ", sequentialTestRuns=" + sequentialTestRuns + ", branch='" + branch + '\'' + ", buildPlanConfiguration='" + buildPlanConfiguration + '\'' + ", buildScript='" + buildScript + '\'' + ", checkoutSolutionRepository=" + checkoutSolutionRepository + ", checkoutPath='" - + testCheckoutPath + '\'' + ", timeoutSeconds=" + timeoutSeconds + ", dockerFlags='" + dockerFlags + '\'' + ", testwiseCoverageEnabled=" + testwiseCoverageEnabled - + ", theiaImage='" + theiaImage + '\'' + ", allowBranching=" + allowBranching + ", branchRegex='" + branchRegex + '\'' + '}'; + + testCheckoutPath + '\'' + ", timeoutSeconds=" + timeoutSeconds + ", dockerFlags='" + dockerFlags + '\'' + ", theiaImage='" + theiaImage + '\'' + + ", allowBranching=" + allowBranching + ", branchRegex='" + branchRegex + '\'' + '}'; } } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseGitDiffEntry.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseGitDiffEntry.java similarity index 98% rename from src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseGitDiffEntry.java rename to src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseGitDiffEntry.java index 23095ebaea55..f1ff24c6c144 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseGitDiffEntry.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseGitDiffEntry.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; +package de.tum.cit.aet.artemis.programming.domain; import jakarta.persistence.Column; import jakarta.persistence.Entity; diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseGitDiffReport.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseGitDiffReport.java similarity index 95% rename from src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseGitDiffReport.java rename to src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseGitDiffReport.java index b63d3fd713b1..d69f509ecf9e 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseGitDiffReport.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseGitDiffReport.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; +package de.tum.cit.aet.artemis.programming.domain; import java.util.Set; @@ -17,7 +17,6 @@ import com.fasterxml.jackson.annotation.JsonInclude; import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; /** * A git-diff report representing a git-diff between the template and solution repositories of a ProgrammingExercise. diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseTask.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseTask.java similarity index 73% rename from src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseTask.java rename to src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseTask.java index 4a7a26ddee49..9b5c0d95b29d 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseTask.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseTask.java @@ -1,9 +1,8 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; +package de.tum.cit.aet.artemis.programming.domain; import java.util.HashSet; import java.util.Set; -import jakarta.persistence.CascadeType; import jakarta.persistence.Column; import jakarta.persistence.Entity; import jakarta.persistence.FetchType; @@ -11,7 +10,6 @@ import jakarta.persistence.JoinTable; import jakarta.persistence.ManyToMany; import jakarta.persistence.ManyToOne; -import jakarta.persistence.OneToMany; import jakarta.persistence.Table; import org.hibernate.annotations.Cache; @@ -22,8 +20,6 @@ import com.fasterxml.jackson.annotation.JsonInclude; import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; /** * A ProgrammingExerciseTask @@ -37,11 +33,6 @@ public class ProgrammingExerciseTask extends DomainObject { @Column(name = "task_name") private String taskName; - // No orphanRemoval here, as there should only be one parent-child relationship (which is ProgrammingExercise -> ExerciseHint) - @OneToMany(mappedBy = "task", cascade = CascadeType.REMOVE, fetch = FetchType.LAZY) - @JsonIgnoreProperties("task") - private Set exerciseHints = new HashSet<>(); - @ManyToMany(fetch = FetchType.LAZY) @JoinTable(name = "programming_exercise_task_test_case", joinColumns = @JoinColumn(name = "task_id", referencedColumnName = "id"), inverseJoinColumns = @JoinColumn(name = "test_case_id", referencedColumnName = "id")) @JsonIgnoreProperties(value = { "tasks", "exercise" }, allowSetters = true) @@ -59,14 +50,6 @@ public void setTaskName(String taskName) { this.taskName = taskName; } - public Set getExerciseHints() { - return exerciseHints; - } - - public void setExerciseHints(Set exerciseHints) { - this.exerciseHints = exerciseHints; - } - public Set getTestCases() { return this.testCases; } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseTestCase.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseTestCase.java index dae43fce9dd9..3da8bde413d7 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseTestCase.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseTestCase.java @@ -1,11 +1,10 @@ package de.tum.cit.aet.artemis.programming.domain; -import static de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTestCaseType.DEFAULT; +import static de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCaseType.DEFAULT; import java.util.HashSet; import java.util.Set; -import jakarta.persistence.CascadeType; import jakarta.persistence.Column; import jakarta.persistence.Entity; import jakarta.persistence.EnumType; @@ -15,7 +14,6 @@ import jakarta.persistence.JoinTable; import jakarta.persistence.ManyToMany; import jakarta.persistence.ManyToOne; -import jakarta.persistence.OneToMany; import jakarta.persistence.Table; import jakarta.validation.constraints.NotNull; @@ -31,10 +29,6 @@ import de.tum.cit.aet.artemis.assessment.domain.Visibility; import de.tum.cit.aet.artemis.core.domain.DomainObject; import de.tum.cit.aet.artemis.exercise.domain.Exercise; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTestCaseType; -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; /** * A ProgrammingExerciseTestCase. @@ -69,10 +63,6 @@ public class ProgrammingExerciseTestCase extends DomainObject { @JsonIgnoreProperties({ "testCases", "exercise" }) private Set tasks = new HashSet<>(); - @OneToMany(mappedBy = "testCase", cascade = CascadeType.REMOVE, orphanRemoval = true, fetch = FetchType.LAZY) - @JsonIgnoreProperties(value = "testCase", allowSetters = true) - private Set solutionEntries = new HashSet<>(); - @ManyToOne(fetch = FetchType.LAZY) @JsonIgnoreProperties("testCases") private ProgrammingExercise exercise; @@ -81,10 +71,6 @@ public class ProgrammingExerciseTestCase extends DomainObject { @Column(name = "test_case_type", nullable = false) private ProgrammingExerciseTestCaseType type = DEFAULT; // default value - @OneToMany(mappedBy = "testCase", fetch = FetchType.LAZY) - @JsonIgnoreProperties("testCase") - private Set coverageEntries; - public ProgrammingExerciseTestCase id(Long id) { setId(id); return this; @@ -152,14 +138,6 @@ public void setTasks(Set tasks) { this.tasks = tasks; } - public Set getSolutionEntries() { - return solutionEntries; - } - - public void setSolutionEntries(Set solutionEntries) { - this.solutionEntries = solutionEntries; - } - public Boolean isActive() { return active; } @@ -217,14 +195,6 @@ public void setType(ProgrammingExerciseTestCaseType programmingExerciseTestCaseT this.type = programmingExerciseTestCaseType; } - public Set getCoverageEntries() { - return coverageEntries; - } - - public void setCoverageEntries(Set coverageEntries) { - this.coverageEntries = coverageEntries; - } - /** * Needs to be checked and updated if there is a new class attribute. Creates a clone with all attributes set to the value of the object, including the id. * diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseTestCaseType.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseTestCaseType.java similarity index 77% rename from src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseTestCaseType.java rename to src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseTestCaseType.java index 047a3ca73eda..3996ca815ec5 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseTestCaseType.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/domain/ProgrammingExerciseTestCaseType.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; +package de.tum.cit.aet.artemis.programming.domain; /** * Used to define the type of a ProgrammingExerciseTestCase. @@ -14,7 +14,7 @@ public enum ProgrammingExerciseTestCaseType { */ BEHAVIORAL, /** - * Type for all programming exercises not supported by Hestia + * All other types */ DEFAULT, } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CodeHint.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CodeHint.java deleted file mode 100644 index a43ba0ac9c3f..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CodeHint.java +++ /dev/null @@ -1,69 +0,0 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; - -import java.util.HashSet; -import java.util.Set; - -import jakarta.persistence.DiscriminatorValue; -import jakarta.persistence.Entity; -import jakarta.persistence.FetchType; -import jakarta.persistence.OneToMany; -import jakarta.persistence.PreRemove; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; - -/** - * A CodeHint. - */ -@Entity -@DiscriminatorValue("C") -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public class CodeHint extends ExerciseHint { - - // No CascadeType.REMOVE here, as we want to retain the solution entries when a code hint is deleted - @OneToMany(mappedBy = "codeHint", fetch = FetchType.LAZY) - @JsonIgnoreProperties(value = { "codeHint" }, allowSetters = true) - private Set solutionEntries = new HashSet<>(); - - public Set getSolutionEntries() { - return this.solutionEntries; - } - - public void setSolutionEntries(Set solutionEntries) { - this.solutionEntries = solutionEntries; - } - - /** - * This method ensures that all solution entry references are removed before deleting a CodeHint - */ - @PreRemove - public void preRemove() { - solutionEntries.forEach(solutionEntry -> solutionEntry.setCodeHint(null)); - } - - @Override - public void removeContent() { - super.removeContent(); - setSolutionEntries(new HashSet<>()); - } - - @Override - public String toString() { - return "CodeHint{" + "id=" + getId() + ", title='" + getTitle() + "}"; - } - - /** - * Creates a copy of this hint including basic attributes, but excluding attributes referencing other models - * - * @return The copied hint - */ - @Override - public CodeHint createCopy() { - CodeHint copiedHint = new CodeHint(); - - copiedHint.setDescription(this.getDescription()); - copiedHint.setContent(this.getContent()); - copiedHint.setTitle(this.getTitle()); - return copiedHint; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CoverageFileReport.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CoverageFileReport.java deleted file mode 100644 index 518c4bad4550..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CoverageFileReport.java +++ /dev/null @@ -1,86 +0,0 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; - -import java.util.Set; - -import jakarta.persistence.CascadeType; -import jakarta.persistence.Column; -import jakarta.persistence.Entity; -import jakarta.persistence.FetchType; -import jakarta.persistence.ManyToOne; -import jakarta.persistence.OneToMany; -import jakarta.persistence.Table; - -import org.hibernate.annotations.Cache; -import org.hibernate.annotations.CacheConcurrencyStrategy; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; - -import de.tum.cit.aet.artemis.core.domain.DomainObject; - -@Entity -@Table(name = "coverage_file_report") -@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE) -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public class CoverageFileReport extends DomainObject { - - @ManyToOne - @JsonIgnoreProperties("fileReports") - private CoverageReport fullReport; - - @Column(name = "file_path") - private String filePath; - - // The number of lines in the whole file - @Column(name = "line_count") - private Integer lineCount; - - // The number of lines that are covered by test cases. The number refers to the unique count, meaning the same line - // that is covered in multiple entries will only count as one line - @Column(name = "covered_line_count") - private Integer coveredLineCount; - - @OneToMany(mappedBy = "fileReport", cascade = CascadeType.ALL, orphanRemoval = true, fetch = FetchType.LAZY) - @JsonIgnoreProperties(value = { "fileReport" }, allowSetters = true) - private Set testwiseCoverageEntries; - - public CoverageReport getFullReport() { - return fullReport; - } - - public void setFullReport(CoverageReport fullReport) { - this.fullReport = fullReport; - } - - public String getFilePath() { - return filePath; - } - - public void setFilePath(String filePath) { - this.filePath = filePath; - } - - public Integer getLineCount() { - return lineCount; - } - - public void setLineCount(Integer lineCount) { - this.lineCount = lineCount; - } - - public Integer getCoveredLineCount() { - return coveredLineCount; - } - - public void setCoveredLineCount(Integer coveredLineCount) { - this.coveredLineCount = coveredLineCount; - } - - public Set getTestwiseCoverageEntries() { - return testwiseCoverageEntries; - } - - public void setTestwiseCoverageEntries(Set testwiseCoverageEntries) { - this.testwiseCoverageEntries = testwiseCoverageEntries; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CoverageReport.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CoverageReport.java deleted file mode 100644 index 9349483cb4e6..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/CoverageReport.java +++ /dev/null @@ -1,75 +0,0 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; - -import java.util.Set; - -import jakarta.persistence.CascadeType; -import jakarta.persistence.Column; -import jakarta.persistence.Entity; -import jakarta.persistence.FetchType; -import jakarta.persistence.JoinColumn; -import jakarta.persistence.OneToMany; -import jakarta.persistence.OneToOne; -import jakarta.persistence.Table; - -import org.hibernate.annotations.Cache; -import org.hibernate.annotations.CacheConcurrencyStrategy; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; - -import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; - -/** - * A testwise coverage report representing the executed code by file path of a single ProgrammingExerciseTestCase. - * The entries contain the information about executed code by the start line and the length (i.e. number of lines) of - * a consecutively executed block. - */ -@Entity -@Table(name = "coverage_report") -@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE) -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public class CoverageReport extends DomainObject { - - // The ProgrammingSubmission to which this CoverageReport is related. This ProgrammingSubmission is always related - // to a SolutionProgrammingExerciseParticipation because the report will only be generated for solution participations. - @OneToOne(fetch = FetchType.LAZY) - @JoinColumn(name = "submission_id", referencedColumnName = "id") - @JsonIgnore - private ProgrammingSubmission submission; - - // When retrieving only the aggregated data (such as covered line ratio), the file reports are not required - @OneToMany(mappedBy = "fullReport", cascade = CascadeType.ALL, orphanRemoval = true, fetch = FetchType.LAZY) - @JsonIgnoreProperties(value = { "fullReport" }, allowSetters = true) - private Set fileReports; - - // The ratio between the number of lines that are covered by all tests and the number of lines for all files in - // the corresponding (solution) submission. The attribute can take values within the range of [0, 1]. - @Column(name = "covered_line_ratio") - private Double coveredLineRatio; - - public ProgrammingSubmission getSubmission() { - return submission; - } - - public void setSubmission(ProgrammingSubmission submission) { - this.submission = submission; - } - - public Set getFileReports() { - return fileReports; - } - - public void setFileReports(Set fileReports) { - this.fileReports = fileReports; - } - - public Double getCoveredLineRatio() { - return coveredLineRatio; - } - - public void setCoveredLineRatio(Double coveredLineRatio) { - this.coveredLineRatio = coveredLineRatio; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ExerciseHint.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ExerciseHint.java deleted file mode 100644 index 119abdf09c53..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ExerciseHint.java +++ /dev/null @@ -1,192 +0,0 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; - -import java.util.HashSet; -import java.util.Set; - -import jakarta.persistence.CascadeType; -import jakarta.persistence.Column; -import jakarta.persistence.DiscriminatorColumn; -import jakarta.persistence.DiscriminatorType; -import jakarta.persistence.DiscriminatorValue; -import jakarta.persistence.Entity; -import jakarta.persistence.FetchType; -import jakarta.persistence.Inheritance; -import jakarta.persistence.InheritanceType; -import jakarta.persistence.ManyToOne; -import jakarta.persistence.OneToMany; -import jakarta.persistence.Table; -import jakarta.persistence.Transient; -import jakarta.validation.constraints.Max; -import jakarta.validation.constraints.Min; - -import org.hibernate.annotations.Cache; -import org.hibernate.annotations.CacheConcurrencyStrategy; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonSubTypes; -import com.fasterxml.jackson.annotation.JsonTypeInfo; - -import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.service.hestia.ExerciseHintService; - -/** - * An ExerciseHint. - */ -@Entity -@Table(name = "exercise_hint") -@Inheritance(strategy = InheritanceType.SINGLE_TABLE) -@DiscriminatorColumn(name = "discriminator", discriminatorType = DiscriminatorType.STRING) -@DiscriminatorValue("T") -@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE) -@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, property = "type") -// @formatter:off -@JsonSubTypes({ - @JsonSubTypes.Type(value = ExerciseHint.class, name = "text"), - @JsonSubTypes.Type(value = CodeHint.class, name = "code") -}) -// @formatter:on -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public class ExerciseHint extends DomainObject { - - @Column(name = "title") - private String title; - - // A short description of this hint, so the student knows what this hint is about before activating it - @Column(name = "description", length = 1000) - private String description; - - @Column(name = "content") - private String content; - - @ManyToOne - @JsonIgnoreProperties("exerciseHints") - private ProgrammingExercise exercise; - - @ManyToOne - @JsonIgnoreProperties("exerciseHints") - private ProgrammingExerciseTask task; - - @OneToMany(mappedBy = "exerciseHint", cascade = CascadeType.REMOVE, orphanRemoval = true, fetch = FetchType.LAZY) - @JsonIgnore - private Set exerciseHintActivations = new HashSet<>(); - - @Column(name = "display_threshold", columnDefinition = "TINYINT") - @Min(0) - @Max(100) - private short displayThreshold = 3; - - @Transient - private Integer currentUserRatingTransient; - - public String getTitle() { - return title; - } - - public ExerciseHint title(String title) { - this.title = title; - return this; - } - - public void setTitle(String title) { - this.title = title; - } - - public String getDescription() { - return description; - } - - public void setDescription(String description) { - this.description = description; - } - - public String getContent() { - return content; - } - - public ExerciseHint content(String content) { - this.content = content; - return this; - } - - public void setContent(String content) { - this.content = content; - } - - public ProgrammingExercise getExercise() { - return exercise; - } - - public ExerciseHint exercise(ProgrammingExercise exercise) { - this.exercise = exercise; - return this; - } - - public ProgrammingExerciseTask getProgrammingExerciseTask() { - return task; - } - - public void setProgrammingExerciseTask(ProgrammingExerciseTask programmingExerciseTask) { - this.task = programmingExerciseTask; - } - - public void setExercise(ProgrammingExercise exercise) { - this.exercise = exercise; - } - - public void removeContent() { - this.content = null; - this.title = null; - } - - @Override - public String toString() { - return "ExerciseHint{" + "id=" + getId() + ", title='" + getTitle() + "'" + ", content='" + getContent() + "'" + "}"; - } - - public Integer getCurrentUserRating() { - return currentUserRatingTransient; - } - - public void setCurrentUserRating(Integer currentUserRating) { - this.currentUserRatingTransient = currentUserRating; - } - - public Set getExerciseHintActivations() { - return exerciseHintActivations; - } - - public void setExerciseHintActivations(Set exerciseHintActivations) { - this.exerciseHintActivations = exerciseHintActivations; - } - - /** - * Returns a threshold value that defines when this exercise hint is displayed to student participating in a programming exercise. - * The algorithm defining if the hint is display is described in {@link ExerciseHintService#getAvailableExerciseHints} - * - * @return the display threshold value - */ - public short getDisplayThreshold() { - return displayThreshold; - } - - public void setDisplayThreshold(short displayThreshold) { - this.displayThreshold = displayThreshold; - } - - /** - * Creates a copy of this hint including basic attributes, but excluding attributes referencing other models - * - * @return The copied hint - */ - public ExerciseHint createCopy() { - ExerciseHint copiedHint = new ExerciseHint(); - - copiedHint.setDescription(this.getDescription()); - copiedHint.setContent(this.getContent()); - copiedHint.setTitle(this.getTitle()); - return copiedHint; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ExerciseHintActivation.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ExerciseHintActivation.java deleted file mode 100644 index 595c90d61abb..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ExerciseHintActivation.java +++ /dev/null @@ -1,67 +0,0 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; - -import java.time.ZonedDateTime; - -import jakarta.persistence.Column; -import jakarta.persistence.Entity; -import jakarta.persistence.ManyToOne; -import jakarta.persistence.Table; - -import org.hibernate.annotations.Cache; -import org.hibernate.annotations.CacheConcurrencyStrategy; - -import com.fasterxml.jackson.annotation.JsonInclude; - -import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.core.domain.User; - -@Entity -@Table(name = "exercise_hint_activation") -@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE) -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public class ExerciseHintActivation extends DomainObject { - - @ManyToOne(optional = false) - private User user; - - @ManyToOne(optional = false) - private ExerciseHint exerciseHint; - - @Column(name = "activation_date", nullable = false) - private ZonedDateTime activationDate; - - @Column(name = "rating") - private Integer rating; - - public User getUser() { - return user; - } - - public void setUser(User user) { - this.user = user; - } - - public ExerciseHint getExerciseHint() { - return exerciseHint; - } - - public void setExerciseHint(ExerciseHint exerciseHint) { - this.exerciseHint = exerciseHint; - } - - public ZonedDateTime getActivationDate() { - return activationDate; - } - - public void setActivationDate(ZonedDateTime activationDate) { - this.activationDate = activationDate; - } - - public Integer getRating() { - return rating; - } - - public void setRating(Integer rating) { - this.rating = rating; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseSolutionEntry.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseSolutionEntry.java deleted file mode 100644 index 01cf4ce871b4..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/ProgrammingExerciseSolutionEntry.java +++ /dev/null @@ -1,135 +0,0 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; - -import jakarta.persistence.Column; -import jakarta.persistence.Entity; -import jakarta.persistence.FetchType; -import jakarta.persistence.ManyToOne; -import jakarta.persistence.Table; - -import org.hibernate.annotations.Cache; -import org.hibernate.annotations.CacheConcurrencyStrategy; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonProperty; - -import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; - -/** - * A ProgrammingExerciseSolutionEntry represents a single change in a file that a students has to make in order to pass the related test. - * It is structured similarly to a git diff entry. - * If it replaces existing code it will contain the previous code that it replaces otherwise previousCode will be null. - * If it is only removing existing code the code attribute will be null. - * If it encompasses the addition of an entire file, previousLine will be null. - * If it deletes an entire file, line will be null. - * previousLine and line will be different when there are other changes higher up in the file. - *

- * Example: - * A print statement gets changed: - * - * SolutionEntry { - * filePath = "<...>" - * previousLine = 12 - * previousCode = System.out.println("Tset"); - * line = 12 - * code = System.out.println("Test"); - * } - * - */ -@Entity -@Table(name = "programming_exercise_solution_entry") -@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE) -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public class ProgrammingExerciseSolutionEntry extends DomainObject { - - @Column(name = "file_path") - private String filePath; - - // The line at which the previous code segment is in the template - @Column(name = "previous_line") - private Integer previousLine; - - // The line at which the new code segment is in the solution - @Column(name = "line") - private Integer line; - - // The previous code segment to be replaced by the new code segment - @Column(name = "previous_code") - private String previousCode; - - // The new code segment that replaces the old code segment - @Column(name = "code") - private String code; - - // Fetched lazily, as we never need the code hint when fetching solution entries - @ManyToOne(fetch = FetchType.LAZY) - @JsonProperty(access = JsonProperty.Access.WRITE_ONLY) - private CodeHint codeHint; - - @ManyToOne - @JsonIgnoreProperties(value = "solutionEntries", allowSetters = true) - private ProgrammingExerciseTestCase testCase; - - public String getFilePath() { - return filePath; - } - - public void setFilePath(String file) { - this.filePath = file; - } - - public Integer getPreviousLine() { - return previousLine; - } - - public void setPreviousLine(Integer previousLine) { - this.previousLine = previousLine; - } - - public Integer getLine() { - return line; - } - - public void setLine(Integer line) { - this.line = line; - } - - public String getPreviousCode() { - return previousCode; - } - - public void setPreviousCode(String previousCode) { - this.previousCode = previousCode; - } - - public String getCode() { - return code; - } - - public void setCode(String code) { - this.code = code; - } - - public CodeHint getCodeHint() { - return codeHint; - } - - public void setCodeHint(CodeHint codeHint) { - this.codeHint = codeHint; - } - - public ProgrammingExerciseTestCase getTestCase() { - return this.testCase; - } - - public void setTestCase(ProgrammingExerciseTestCase testCase) { - this.testCase = testCase; - } - - @Override - public String toString() { - return "ProgrammingExerciseSolutionEntry{" + "id=" + getId() + '\'' + ", filePath='" + filePath + '\'' + ", previousLine=" + previousLine + ", line=" + line - + ", previousCode='" + previousCode + '\'' + ", code='" + code + '\'' + '}'; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/TestwiseCoverageReportEntry.java b/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/TestwiseCoverageReportEntry.java deleted file mode 100644 index 60383357c9dc..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/domain/hestia/TestwiseCoverageReportEntry.java +++ /dev/null @@ -1,72 +0,0 @@ -package de.tum.cit.aet.artemis.programming.domain.hestia; - -import jakarta.persistence.Column; -import jakarta.persistence.Entity; -import jakarta.persistence.ManyToOne; -import jakarta.persistence.Table; - -import org.hibernate.annotations.Cache; -import org.hibernate.annotations.CacheConcurrencyStrategy; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; - -import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; - -/** - * A single entry from testwise coverage report by file path and consecutive executed code block. - * A block is represented by the start line and the length (i.e. number of lines) of the block. - */ -@Entity -@Table(name = "testwise_coverage_report_entry") -@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE) -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public class TestwiseCoverageReportEntry extends DomainObject { - - @ManyToOne - @JsonIgnoreProperties("testwiseCoverageEntries") - private CoverageFileReport fileReport; - - @ManyToOne - @JsonIgnoreProperties("coverageEntries") - private ProgrammingExerciseTestCase testCase; - - @Column(name = "start_line") - private Integer startLine; - - @Column(name = "line_count") - private Integer lineCount; - - public CoverageFileReport getFileReport() { - return fileReport; - } - - public void setFileReport(CoverageFileReport fileReport) { - this.fileReport = fileReport; - } - - public ProgrammingExerciseTestCase getTestCase() { - return testCase; - } - - public void setTestCase(ProgrammingExerciseTestCase testCase) { - this.testCase = testCase; - } - - public Integer getStartLine() { - return startLine; - } - - public void setStartLine(Integer startLine) { - this.startLine = startLine; - } - - public Integer getLineCount() { - return lineCount; - } - - public void setLineCount(Integer lineCount) { - this.lineCount = lineCount; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/dto/BuildJobDTOInterface.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/BuildJobInterface.java similarity index 74% rename from src/main/java/de/tum/cit/aet/artemis/programming/dto/BuildJobDTOInterface.java rename to src/main/java/de/tum/cit/aet/artemis/programming/dto/BuildJobInterface.java index 3d838ca6233b..0a8a7362deef 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/dto/BuildJobDTOInterface.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/BuildJobInterface.java @@ -8,19 +8,19 @@ * Interface for DTOs that represent a build job. */ @JsonInclude(JsonInclude.Include.NON_EMPTY) -public interface BuildJobDTOInterface { +public interface BuildJobInterface { /** * Gets the failed tests of the build job. * * @return list of failed tests. */ - List getFailedTests(); + List failedTests(); /** * Gets the successful tests of the build job. * * @return list of successful tests. */ - List getSuccessfulTests(); + List successfulTests(); } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/dto/AbstractBuildResultNotificationDTO.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/BuildResultNotification.java similarity index 54% rename from src/main/java/de/tum/cit/aet/artemis/programming/dto/AbstractBuildResultNotificationDTO.java rename to src/main/java/de/tum/cit/aet/artemis/programming/dto/BuildResultNotification.java index 9435fdf8e463..bfad4eba2de8 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/dto/AbstractBuildResultNotificationDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/BuildResultNotification.java @@ -11,27 +11,25 @@ import de.tum.cit.aet.artemis.exercise.domain.SubmissionType; import de.tum.cit.aet.artemis.programming.domain.build.BuildLogEntry; -import de.tum.cit.aet.artemis.programming.service.ci.notification.dto.TestwiseCoverageReportDTO; @JsonIgnoreProperties(ignoreUnknown = true) @JsonInclude(JsonInclude.Include.NON_EMPTY) -// TODO: convert subclasses to records -public abstract class AbstractBuildResultNotificationDTO { +public interface BuildResultNotification { - public abstract ZonedDateTime getBuildRunDate(); + ZonedDateTime buildRunDate(); @Nullable - protected abstract String getCommitHashFromAssignmentRepo(); + String assignmentRepoCommitHash(); @Nullable - protected abstract String getCommitHashFromTestsRepo(); + String testsRepoCommitHash(); @Nullable - public abstract String getBranchNameFromAssignmentRepo(); + String assignmentRepoBranchName(); - public abstract boolean isBuildSuccessful(); + boolean isBuildSuccessful(); - public abstract Double getBuildScore(); + Double buildScore(); /** * Get the commit hash from the build result, the commit hash will be different for submission types or null. @@ -40,22 +38,22 @@ public abstract class AbstractBuildResultNotificationDTO { * @return if the commit hash for the given submission type was found, otherwise empty. */ @Nullable - public String getCommitHash(SubmissionType submissionType) { + default String commitHash(SubmissionType submissionType) { final var isAssignmentSubmission = List.of(SubmissionType.MANUAL, SubmissionType.INSTRUCTOR, SubmissionType.ILLEGAL).contains(submissionType); if (isAssignmentSubmission) { - return getCommitHashFromAssignmentRepo(); + return assignmentRepoCommitHash(); } else if (submissionType.equals(SubmissionType.TEST)) { - return getCommitHashFromTestsRepo(); + return testsRepoCommitHash(); } return null; } - public abstract boolean hasArtifact(); + boolean hasArtifact(); - public abstract boolean hasLogs(); + boolean hasLogs(); - public abstract List extractBuildLogs(); + List extractBuildLogs(); /** * Gets the build jobs that are part of the build result. @@ -63,19 +61,12 @@ else if (submissionType.equals(SubmissionType.TEST)) { * @return list of build jobs. */ @JsonIgnore - public abstract List getBuildJobs(); + List jobs(); /** * Gets the static code analysis reports that are part of the build result. * * @return list of static code analysis reports. */ - public abstract List getStaticCodeAnalysisReports(); - - /** - * Gets the test-wise coverage reports that are part of the build result. - * - * @return list of test-wise coverage reports. - */ - public abstract List getTestwiseCoverageReports(); + List staticCodeAnalysisReports(); } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/dto/ConsistencyErrorDTO.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/ConsistencyErrorDTO.java index d200a62e3422..4e4e582cbebe 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/dto/ConsistencyErrorDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/ConsistencyErrorDTO.java @@ -1,7 +1,5 @@ package de.tum.cit.aet.artemis.programming.dto; -import java.util.Objects; - import com.fasterxml.jackson.annotation.JsonInclude; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; @@ -9,63 +7,11 @@ /** * A DTO representing a consistency error */ +// TODO: use a ProgrammingExerciseDTO instead of the whole ProgrammingExercise entity @JsonInclude(JsonInclude.Include.NON_EMPTY) -public class ConsistencyErrorDTO { - - private ProgrammingExercise programmingExercise; - - private ErrorType type; - - public ConsistencyErrorDTO(ProgrammingExercise programmingExercise, ErrorType type) { - this.programmingExercise = programmingExercise; - this.type = type; - } - - public ProgrammingExercise getProgrammingExercise() { - return programmingExercise; - } - - public void setProgrammingExercise(ProgrammingExercise programmingExercise) { - this.programmingExercise = programmingExercise; - } - - public ErrorType getType() { - return type; - } - - public void setType(ErrorType type) { - this.type = type; - } +public record ConsistencyErrorDTO(ProgrammingExercise programmingExercise, ErrorType type) { public enum ErrorType { - VCS_PROJECT_MISSING, TEMPLATE_REPO_MISSING, SOLUTION_REPO_MISSING, AUXILIARY_REPO_MISSING, TEST_REPO_MISSING, TEMPLATE_BUILD_PLAN_MISSING, SOLUTION_BUILD_PLAN_MISSING; - - @Override - public String toString() { - return name(); - } - } - - @Override - public boolean equals(Object obj) { - if (this == obj) { - return true; - } - if (obj == null || getClass() != obj.getClass()) { - return false; - } - ConsistencyErrorDTO that = (ConsistencyErrorDTO) obj; - return Objects.equals(getProgrammingExercise(), that.getProgrammingExercise()) && getType() == that.getType(); - } - - @Override - public int hashCode() { - return Objects.hash(getProgrammingExercise(), getType()); - } - - @Override - public String toString() { - return "ConsistencyErrorDTO{" + "programmingExercise='" + programmingExercise.getTitle() + "', type='" + type.name() + "'}"; } } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/dto/CoverageReportAndSubmissionDateDTO.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/CoverageReportAndSubmissionDateDTO.java deleted file mode 100644 index a5ddfd8550d6..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/dto/CoverageReportAndSubmissionDateDTO.java +++ /dev/null @@ -1,11 +0,0 @@ -package de.tum.cit.aet.artemis.programming.dto; - -import java.time.ZonedDateTime; - -import com.fasterxml.jackson.annotation.JsonInclude; - -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageReport; - -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public record CoverageReportAndSubmissionDateDTO(CoverageReport coverageReport, ZonedDateTime submissionDate) { -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/dto/ProgrammingExerciseGitDiffEntryDTO.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/ProgrammingExerciseGitDiffEntryDTO.java index 6b5f37149cd6..67707c175ab0 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/dto/ProgrammingExerciseGitDiffEntryDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/ProgrammingExerciseGitDiffEntryDTO.java @@ -2,7 +2,7 @@ import com.fasterxml.jackson.annotation.JsonInclude; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseGitDiffEntry; /** * DTO for a git diff report entry. diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/dto/ProgrammingExerciseGitDiffReportDTO.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/ProgrammingExerciseGitDiffReportDTO.java index 6f027a90326b..04e32ff039d1 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/dto/ProgrammingExerciseGitDiffReportDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/ProgrammingExerciseGitDiffReportDTO.java @@ -5,7 +5,7 @@ import com.fasterxml.jackson.annotation.JsonInclude; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseGitDiffReport; /** * DTO for a git diff report. diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/dto/TestCaseBaseDTO.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/TestCaseBase.java similarity index 85% rename from src/main/java/de/tum/cit/aet/artemis/programming/dto/TestCaseBaseDTO.java rename to src/main/java/de/tum/cit/aet/artemis/programming/dto/TestCaseBase.java index 267caa9256db..fef892e8fb16 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/dto/TestCaseBaseDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/TestCaseBase.java @@ -9,14 +9,14 @@ * Interface for DTOs that represent a test case. */ @JsonInclude(JsonInclude.Include.NON_EMPTY) -public interface TestCaseBaseDTO { +public interface TestCaseBase { /** * Gets the name of the test case * * @return the name of the test case */ - String getName(); + String name(); /** * Gets the messages of the test case (typically error messages) @@ -24,5 +24,5 @@ public interface TestCaseBaseDTO { * @return the messages of the test case */ @JsonIgnore - List getTestMessages(); + List testMessages(); } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/Action.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/Action.java new file mode 100644 index 000000000000..34f2cfb9537f --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/Action.java @@ -0,0 +1,30 @@ +package de.tum.cit.aet.artemis.programming.dto.aeolus; + +import java.util.List; +import java.util.Map; + +import com.fasterxml.jackson.annotation.JsonIgnoreProperties; +import com.fasterxml.jackson.annotation.JsonInclude; + +/** + * Base for the actions that can be defined in a {@link Windfile} + * NOTE: you must create a record that implements this interface to specify actions + */ +@JsonIgnoreProperties(ignoreUnknown = true) +@JsonInclude(JsonInclude.Include.NON_EMPTY) +public interface Action { + + Map parameters(); + + Map environment(); + + boolean runAlways(); + + String name(); + + List results(); + + String workdir(); + + String platform(); +} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/AeolusRepository.java similarity index 85% rename from src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusRepository.java rename to src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/AeolusRepository.java index 6793fb84db1d..84dc2e893d88 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/AeolusRepository.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.service.aeolus; +package de.tum.cit.aet.artemis.programming.dto.aeolus; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonInclude; diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusResult.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/AeolusResult.java similarity index 83% rename from src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusResult.java rename to src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/AeolusResult.java index 12f319121957..096e98701702 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusResult.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/AeolusResult.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.service.aeolus; +package de.tum.cit.aet.artemis.programming.dto.aeolus; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonInclude; diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/DockerConfig.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/DockerConfig.java similarity index 95% rename from src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/DockerConfig.java rename to src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/DockerConfig.java index 5f5161737a6a..ed8cc72283c1 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/DockerConfig.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/DockerConfig.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.service.aeolus; +package de.tum.cit.aet.artemis.programming.dto.aeolus; import java.util.List; diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/PlatformAction.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/PlatformAction.java new file mode 100644 index 000000000000..d2bde6916175 --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/PlatformAction.java @@ -0,0 +1,17 @@ +package de.tum.cit.aet.artemis.programming.dto.aeolus; + +import java.util.List; +import java.util.Map; + +import com.fasterxml.jackson.annotation.JsonIgnoreProperties; +import com.fasterxml.jackson.annotation.JsonInclude; + +/** + * Represents a CI action that is intended to run only on a specific target, can be used in a {@link Windfile}. + */ +@JsonIgnoreProperties(ignoreUnknown = true) +@JsonInclude(JsonInclude.Include.NON_EMPTY) +public record PlatformAction(String name, Map parameters, Map environment, List results, String workdir, boolean runAlways, + String platform, String kind, String type) implements Action { + +} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/ScriptAction.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/ScriptAction.java similarity index 60% rename from src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/ScriptAction.java rename to src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/ScriptAction.java index 5006e8a850a6..9716b97deac0 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/ScriptAction.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/ScriptAction.java @@ -1,4 +1,7 @@ -package de.tum.cit.aet.artemis.programming.service.aeolus; +package de.tum.cit.aet.artemis.programming.dto.aeolus; + +import java.util.List; +import java.util.Map; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonInclude; @@ -8,18 +11,9 @@ * independent actions but also to run actions on a single target. (e.q. the parsing of the test results that needs to * run on Jenkins but not in LocalCI) */ -// TODO: convert into Record @JsonIgnoreProperties(ignoreUnknown = true) @JsonInclude(JsonInclude.Include.NON_EMPTY) -public class ScriptAction extends Action { - - private String script; - - public String getScript() { - return script; - } +public record ScriptAction(String name, Map parameters, Map environment, List results, String workdir, boolean runAlways, + String platform, String script) implements Action { - public void setScript(String script) { - this.script = script; - } } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/Windfile.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/Windfile.java new file mode 100644 index 000000000000..f975a45acc60 --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/Windfile.java @@ -0,0 +1,86 @@ +package de.tum.cit.aet.artemis.programming.dto.aeolus; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import com.fasterxml.jackson.annotation.JsonIgnoreProperties; +import com.fasterxml.jackson.annotation.JsonInclude; +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.module.SimpleModule; + +import de.tum.cit.aet.artemis.programming.service.aeolus.ActionDeserializer; + +/** + * Represents a windfile, the definition file for an aeolus build plan that can then be used to generate a Jenkinsfile. + */ +@JsonIgnoreProperties(ignoreUnknown = true) +@JsonInclude(JsonInclude.Include.NON_EMPTY) +public record Windfile(String api, WindfileMetadata metadata, List actions, Map repositories) { + + public Windfile { + if (actions == null) { + actions = new ArrayList<>(); + } + if (repositories == null) { + repositories = new HashMap<>(); + } + } + + private static final ObjectMapper mapper = new ObjectMapper(); + + /** + * Creates a new windfile based on an existing one with updated metadata. + * + * @param existingWindfile the existing windfile to base the new one on. + * @param metadata the metadata of the newly created windfile. + */ + public Windfile(Windfile existingWindfile, WindfileMetadata metadata) { + this(existingWindfile.api(), metadata, existingWindfile.actions(), existingWindfile.repositories()); + } + + /** + * Creates a new windfile based on an existing one with updated metadata. + * + * @param existingWindfile the existing windfile to base the new one on. + * @param metadata the metadata of the newly created windfile. + * @param repositories the repositories of the newly created windfile. + */ + public Windfile(Windfile existingWindfile, WindfileMetadata metadata, Map repositories) { + this(existingWindfile.api(), metadata, existingWindfile.actions(), repositories); + } + + /** + * Deserializes a windfile from a json string. + * + * @param json the json string to deserialize. + * @return the deserialized windfile. + * @throws JsonProcessingException if the json string is not valid. + */ + public static Windfile deserialize(String json) throws JsonProcessingException { + SimpleModule module = new SimpleModule(); + module.addDeserializer(Action.class, new ActionDeserializer()); + mapper.registerModule(module); + return mapper.readValue(json, Windfile.class); + } + + /** + * Gets the script actions of a windfile. + * + * @return the script actions of a windfile. + */ + public List scriptActions() { + return actions.stream().filter(ScriptAction.class::isInstance).map(ScriptAction.class::cast).toList(); + } + + /** + * Collects the results of all actions of a windfile. + * + * @return the results of all actions of this windfile + */ + public List results() { + return actions.stream().filter(action -> action.results() != null && !action.results().isEmpty()).flatMap(action -> action.results().stream()).toList(); + } +} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/WindfileMetadata.java b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/WindfileMetadata.java similarity index 90% rename from src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/WindfileMetadata.java rename to src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/WindfileMetadata.java index 5e9d1547ddfa..0c5def63abd7 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/WindfileMetadata.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/dto/aeolus/WindfileMetadata.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.service.aeolus; +package de.tum.cit.aet.artemis.programming.dto.aeolus; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonInclude; diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ProgrammingExerciseGitDiffReportRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseGitDiffReportRepository.java similarity index 88% rename from src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ProgrammingExerciseGitDiffReportRepository.java rename to src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseGitDiffReportRepository.java index d1c5b9acf8d0..37b6b9fb3522 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ProgrammingExerciseGitDiffReportRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseGitDiffReportRepository.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.repository.hestia; +package de.tum.cit.aet.artemis.programming.repository; import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; @@ -10,7 +10,7 @@ import org.springframework.transaction.annotation.Transactional; import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseGitDiffReport; /** * Spring Data JPA repository for the ProgrammingExerciseGitDiffReport entity. diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseRepository.java index 73ea3f0cddc3..8641ccd9e48c 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseRepository.java @@ -314,11 +314,9 @@ default ProgrammingExercise findOneByProjectKeyOrThrow(String projectKey, boolea FROM ProgrammingExercise p LEFT JOIN FETCH p.testCases tc LEFT JOIN FETCH p.staticCodeAnalysisCategories - LEFT JOIN FETCH p.exerciseHints LEFT JOIN FETCH p.templateParticipation LEFT JOIN FETCH p.solutionParticipation LEFT JOIN FETCH p.auxiliaryRepositories - LEFT JOIN FETCH tc.solutionEntries LEFT JOIN FETCH p.buildConfig WHERE p.id = :exerciseId """) @@ -330,15 +328,13 @@ Optional findByIdWithEagerTestCasesStaticCodeAnalysisCatego FROM ProgrammingExercise p LEFT JOIN FETCH p.testCases tc LEFT JOIN FETCH p.staticCodeAnalysisCategories - LEFT JOIN FETCH p.exerciseHints LEFT JOIN FETCH p.templateParticipation LEFT JOIN FETCH p.solutionParticipation LEFT JOIN FETCH p.auxiliaryRepositories - LEFT JOIN FETCH tc.solutionEntries LEFT JOIN FETCH p.buildConfig WHERE p.id = :exerciseId """) - Optional findByIdWithEagerBuildConfigTestCasesStaticCodeAnalysisCategoriesHintsAndTemplateAndSolutionParticipationsAndAuxReposAndSolutionEntriesAndBuildConfig( + Optional findByIdWithEagerBuildConfigTestCasesStaticCodeAnalysisCategoriesAndTemplateAndSolutionParticipationsAndAuxReposAndAndBuildConfig( @Param("exerciseId") long exerciseId); @Query(""" @@ -346,11 +342,9 @@ Optional findByIdWithEagerBuildConfigTestCasesStaticCodeAna FROM ProgrammingExercise p LEFT JOIN FETCH p.testCases tc LEFT JOIN FETCH p.staticCodeAnalysisCategories - LEFT JOIN FETCH p.exerciseHints LEFT JOIN FETCH p.templateParticipation LEFT JOIN FETCH p.solutionParticipation LEFT JOIN FETCH p.auxiliaryRepositories - LEFT JOIN FETCH tc.solutionEntries LEFT JOIN FETCH p.buildConfig LEFT JOIN FETCH p.plagiarismDetectionConfig WHERE p.id = :exerciseId @@ -976,7 +970,6 @@ enum ProgrammingExerciseFetchOptions implements FetchOptions { Tasks(ProgrammingExercise_.TASKS), StaticCodeAnalysisCategories(ProgrammingExercise_.STATIC_CODE_ANALYSIS_CATEGORIES), SubmissionPolicy(ProgrammingExercise_.SUBMISSION_POLICY), - ExerciseHints(ProgrammingExercise_.EXERCISE_HINTS), CompetencyLinks(ProgrammingExercise_.COMPETENCY_LINKS), Teams(ProgrammingExercise_.TEAMS), TutorParticipations(ProgrammingExercise_.TUTOR_PARTICIPATIONS), diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ProgrammingExerciseTaskRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseTaskRepository.java similarity index 74% rename from src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ProgrammingExerciseTaskRepository.java rename to src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseTaskRepository.java index 778c0c811374..fb2f47556daf 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ProgrammingExerciseTaskRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseTaskRepository.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.repository.hestia; +package de.tum.cit.aet.artemis.programming.repository; import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; @@ -15,7 +15,7 @@ import de.tum.cit.aet.artemis.core.exception.EntityNotFoundException; import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTask; /** * Spring Data repository for the ProgrammingExerciseTask entity. @@ -32,8 +32,8 @@ public interface ProgrammingExerciseTaskRepository extends ArtemisJpaRepository< * @throws EntityNotFoundException If the exercise with exerciseId does not exist */ @NotNull - default List findByExerciseIdWithTestCaseAndSolutionEntriesElseThrow(long exerciseId) throws EntityNotFoundException { - return getArbitraryValueElseThrow(findByExerciseIdWithTestCaseAndSolutionEntries(exerciseId), Long.toString(exerciseId)); + default List findByExerciseIdWithTestCaseElseThrow(long exerciseId) throws EntityNotFoundException { + return getArbitraryValueElseThrow(findByExerciseIdWithTestCase(exerciseId), Long.toString(exerciseId)); } /** @@ -46,11 +46,10 @@ default List findByExerciseIdWithTestCaseAndSolutionEnt SELECT t FROM ProgrammingExerciseTask t LEFT JOIN FETCH t.testCases tc - LEFT JOIN FETCH tc.solutionEntries WHERE t.exercise.id = :exerciseId AND tc.exercise.id = :exerciseId """) - Optional> findByExerciseIdWithTestCaseAndSolutionEntries(@Param("exerciseId") long exerciseId); + Optional> findByExerciseIdWithTestCase(@Param("exerciseId") long exerciseId); /** * Gets all tasks with its test cases for a programming exercise @@ -62,17 +61,7 @@ default List findByExerciseIdWithTestCaseAndSolutionEnt SELECT t FROM ProgrammingExerciseTask t LEFT JOIN FETCH t.testCases tc - LEFT JOIN FETCH tc.solutionEntries WHERE t.exercise.id = :exerciseId """) Set findByExerciseIdWithTestCases(@Param("exerciseId") Long exerciseId); - - @Query(""" - SELECT pt - FROM ProgrammingExerciseTask pt - LEFT JOIN FETCH pt.exerciseHints h - LEFT JOIN FETCH pt.testCases tc - WHERE h.id = :codeHintId - """) - Optional findByCodeHintIdWithTestCases(@Param("codeHintId") Long codeHintId); } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseTestCaseRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseTestCaseRepository.java index 0ec309e45e8b..3924a5de856a 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseTestCaseRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/repository/ProgrammingExerciseTestCaseRepository.java @@ -2,7 +2,6 @@ import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; -import java.util.Optional; import java.util.Set; import org.springframework.context.annotation.Profile; @@ -23,40 +22,6 @@ public interface ProgrammingExerciseTestCaseRepository extends ArtemisJpaReposit Set findByExerciseId(long exerciseId); - default ProgrammingExerciseTestCase findByIdWithExerciseElseThrow(long testCaseId) { - return getValueElseThrow(findByIdWithExercise(testCaseId), testCaseId); - } - - /** - * Returns the test case with the programming exercise - * - * @param testCaseId of the test case - * @return the test case with the programming exercise - */ - @Query(""" - SELECT tc - FROM ProgrammingExerciseTestCase tc - LEFT JOIN FETCH tc.exercise ex - WHERE tc.id = :testCaseId - """) - Optional findByIdWithExercise(@Param("testCaseId") long testCaseId); - - /** - * Returns all test cases with the associated solution entries for a programming exercise - * - * @param exerciseId of the exercise - * @param active status of the test case - * @return all test cases with the associated solution entries - */ - @Query(""" - SELECT DISTINCT tc - FROM ProgrammingExerciseTestCase tc - LEFT JOIN FETCH tc.solutionEntries se - WHERE tc.exercise.id = :exerciseId - AND tc.active = :active - """) - Set findByExerciseIdWithSolutionEntriesAndActive(@Param("exerciseId") long exerciseId, @Param("active") Boolean active); - Set findByExerciseIdAndActive(long exerciseId, Boolean active); /** diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CodeHintRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CodeHintRepository.java deleted file mode 100644 index 9c4a03766832..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CodeHintRepository.java +++ /dev/null @@ -1,50 +0,0 @@ -package de.tum.cit.aet.artemis.programming.repository.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.util.Optional; -import java.util.Set; - -import jakarta.validation.constraints.NotNull; - -import org.springframework.context.annotation.Profile; -import org.springframework.data.jpa.repository.Query; -import org.springframework.data.repository.query.Param; -import org.springframework.stereotype.Repository; - -import de.tum.cit.aet.artemis.core.exception.EntityNotFoundException; -import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; - -/** - * Spring Data repository for the CodeHint entity. - */ -@Profile(PROFILE_CORE) -@Repository -public interface CodeHintRepository extends ArtemisJpaRepository { - - Set findByExerciseId(Long exerciseId); - - @NotNull - default CodeHint findByIdWithSolutionEntriesElseThrow(long exerciseHintId) throws EntityNotFoundException { - return getValueElseThrow(findByIdWithSolutionEntries(exerciseHintId), exerciseHintId); - } - - @Query(""" - SELECT h - FROM CodeHint h - LEFT JOIN FETCH h.task t - LEFT JOIN FETCH h.solutionEntries tc - WHERE h.id = :codeHintId - """) - Optional findByIdWithSolutionEntries(@Param("codeHintId") Long codeHintId); - - @Query(""" - SELECT h - FROM CodeHint h - LEFT JOIN FETCH h.task t - LEFT JOIN FETCH h.solutionEntries tc - WHERE t.id = :taskId - """) - Set findByTaskIdWithSolutionEntries(@Param("taskId") Long taskId); -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CoverageFileReportRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CoverageFileReportRepository.java deleted file mode 100644 index 69bcac01e5ac..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CoverageFileReportRepository.java +++ /dev/null @@ -1,15 +0,0 @@ -package de.tum.cit.aet.artemis.programming.repository.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import org.springframework.context.annotation.Profile; -import org.springframework.stereotype.Repository; - -import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageFileReport; - -@Profile(PROFILE_CORE) -@Repository -public interface CoverageFileReportRepository extends ArtemisJpaRepository { - -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CoverageReportRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CoverageReportRepository.java deleted file mode 100644 index eefc9aeb5848..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/CoverageReportRepository.java +++ /dev/null @@ -1,101 +0,0 @@ -package de.tum.cit.aet.artemis.programming.repository.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; -import static org.springframework.data.jpa.repository.EntityGraph.EntityGraphType.LOAD; - -import java.util.Collections; -import java.util.List; -import java.util.Optional; - -import org.springframework.context.annotation.Profile; -import org.springframework.data.domain.Pageable; -import org.springframework.data.jpa.repository.EntityGraph; -import org.springframework.data.jpa.repository.Modifying; -import org.springframework.data.jpa.repository.Query; -import org.springframework.data.repository.query.Param; -import org.springframework.stereotype.Repository; -import org.springframework.transaction.annotation.Transactional; - -import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageReport; -import de.tum.cit.aet.artemis.programming.dto.CoverageReportAndSubmissionDateDTO; - -/** - * Spring Data JPA repository for the CoverageReport entity. - */ -@Profile(PROFILE_CORE) -@Repository -public interface CoverageReportRepository extends ArtemisJpaRepository { - - Boolean existsBySubmissionId(Long submissionId); - - @Transactional // ok because of delete - @Modifying - void deleteBySubmissionId(Long submissionId); - - @Query(""" - SELECT new de.tum.cit.aet.artemis.programming.dto.CoverageReportAndSubmissionDateDTO(r, s.submissionDate) - FROM CoverageReport r - JOIN r.submission s - JOIN ProgrammingExercise pe ON s.participation = pe.solutionParticipation - WHERE pe.id = :programmingExerciseId - AND (s.type <> de.tum.cit.aet.artemis.exercise.domain.SubmissionType.ILLEGAL OR s.type IS NULL) - ORDER BY s.submissionDate DESC - """) - List findCoverageReportsByProgrammingExerciseId(@Param("programmingExerciseId") Long programmingExerciseId, Pageable pageable); - - @EntityGraph(type = LOAD, attributePaths = "submission") - List findCoverageReportsWithSubmissionByIdIn(List ids); - - /** - * Retrieves the latest coverage reports with legal submissions for a specific programming exercise, with pagination support. - * This method avoids in-memory paging by retrieving the coverage report IDs directly from the database. - * - * @param programmingExerciseId the ID of the programming exercise to retrieve the coverage reports for - * @param pageable the pagination information - * @return a list of {@code CoverageReport} with legal submissions, or an empty list if no reports are found - */ - default List getLatestCoverageReportsWithLegalSubmissionsForProgrammingExercise(Long programmingExerciseId, Pageable pageable) { - List ids = findCoverageReportsByProgrammingExerciseId(programmingExerciseId, pageable).stream().map(CoverageReportAndSubmissionDateDTO::coverageReport) - .map(DomainObject::getId).toList(); - if (ids.isEmpty()) { - return Collections.emptyList(); - } - return findCoverageReportsWithSubmissionByIdIn(ids); - } - - @EntityGraph(type = LOAD, attributePaths = { "submission", "fileReports", "fileReports.testwiseCoverageEntries" }) - List findDistinctCoverageReportsWithEagerRelationshipsByIdIn(List ids); - - /** - * Retrieves the latest coverage reports with legal submissions for a specific programming exercise, including eager loading of file reports and entries, with pagination - * support. - * This method avoids in-memory paging by retrieving the coverage report IDs directly from the database. - * - * @param programmingExerciseId the ID of the programming exercise to retrieve the coverage reports for - * @param pageable the pagination information - * @return a list of distinct {@code CoverageReport} with eager relationships, or an empty list if no reports are found - */ - default List getLatestCoverageReportsForLegalSubmissionsForProgrammingExerciseWithEagerFileReportsAndEntries(Long programmingExerciseId, Pageable pageable) { - List ids = findCoverageReportsByProgrammingExerciseId(programmingExerciseId, pageable).stream().map(CoverageReportAndSubmissionDateDTO::coverageReport) - .map(DomainObject::getId).toList(); - if (ids.isEmpty()) { - return Collections.emptyList(); - } - return findDistinctCoverageReportsWithEagerRelationshipsByIdIn(ids); - } - - @Query(""" - SELECT DISTINCT r - FROM CoverageReport r - LEFT JOIN FETCH r.fileReports f - LEFT JOIN FETCH f.testwiseCoverageEntries - WHERE r.id = :coverageReportId - """) - Optional findCoverageReportByIdWithEagerFileReportsAndEntries(@Param("coverageReportId") Long coverageReportId); - - default CoverageReport findCoverageReportByIdWithEagerFileReportsAndEntriesElseThrow(Long coverageReportId) { - return getValueElseThrow(findCoverageReportByIdWithEagerFileReportsAndEntries(coverageReportId), coverageReportId); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ExerciseHintActivationRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ExerciseHintActivationRepository.java deleted file mode 100644 index c827a9b3052b..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ExerciseHintActivationRepository.java +++ /dev/null @@ -1,41 +0,0 @@ -package de.tum.cit.aet.artemis.programming.repository.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.util.Optional; -import java.util.Set; - -import org.springframework.context.annotation.Profile; -import org.springframework.data.jpa.repository.Query; -import org.springframework.data.repository.query.Param; -import org.springframework.stereotype.Repository; - -import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHintActivation; - -@Profile(PROFILE_CORE) -@Repository -public interface ExerciseHintActivationRepository extends ArtemisJpaRepository { - - @Query(""" - SELECT hintActivation - FROM ExerciseHintActivation hintActivation - LEFT JOIN FETCH hintActivation.exerciseHint hint - LEFT JOIN FETCH hint.solutionEntries - WHERE hintActivation.exerciseHint.exercise.id = :exerciseId - AND hintActivation.user.id = :userId - """) - Set findByExerciseAndUserWithExerciseHintRelations(@Param("exerciseId") long exerciseId, @Param("userId") long userId); - - @Query(""" - SELECT hintActivation - FROM ExerciseHintActivation hintActivation - WHERE hintActivation.exerciseHint.id = :exerciseHintId - AND hintActivation.user.id = :userId - """) - Optional findByExerciseHintAndUser(@Param("exerciseHintId") long exerciseHintId, @Param("userId") long userId); - - default ExerciseHintActivation findByExerciseHintAndUserElseThrow(long exerciseHintId, long userId) { - return getValueElseThrow(findByExerciseHintAndUser(exerciseHintId, userId)); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ExerciseHintRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ExerciseHintRepository.java deleted file mode 100644 index a5f0eb4942ca..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ExerciseHintRepository.java +++ /dev/null @@ -1,50 +0,0 @@ -package de.tum.cit.aet.artemis.programming.repository.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.util.Optional; -import java.util.Set; - -import jakarta.validation.constraints.NotNull; - -import org.springframework.context.annotation.Profile; -import org.springframework.data.jpa.repository.Query; -import org.springframework.data.repository.query.Param; -import org.springframework.stereotype.Repository; - -import de.tum.cit.aet.artemis.core.exception.EntityNotFoundException; -import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; - -/** - * Spring Data repository for the ExerciseHint entity. - */ -@Profile(PROFILE_CORE) -@Repository -public interface ExerciseHintRepository extends ArtemisJpaRepository { - - @Query(""" - SELECT h - FROM ExerciseHint h - LEFT JOIN FETCH h.solutionEntries se - WHERE h.id = :hintId - """) - Optional findByIdWithRelations(@Param("hintId") Long hintId); - - @NotNull - default ExerciseHint findByIdWithRelationsElseThrow(long hintId) throws EntityNotFoundException { - return getValueElseThrow(findByIdWithRelations(hintId), hintId); - } - - Set findByExerciseId(Long exerciseId); - - @Query(""" - SELECT h - FROM ExerciseHint h - LEFT JOIN FETCH h.solutionEntries se - WHERE h.exercise.id = :exerciseId - """) - Set findByExerciseIdWithRelations(@Param("exerciseId") Long exerciseId); - - Set findByTaskId(Long taskId); -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ProgrammingExerciseSolutionEntryRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ProgrammingExerciseSolutionEntryRepository.java deleted file mode 100644 index 14a03ed49c0a..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/ProgrammingExerciseSolutionEntryRepository.java +++ /dev/null @@ -1,83 +0,0 @@ -package de.tum.cit.aet.artemis.programming.repository.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.util.Optional; -import java.util.Set; - -import jakarta.validation.constraints.NotNull; - -import org.springframework.context.annotation.Profile; -import org.springframework.data.jpa.repository.Query; -import org.springframework.data.repository.query.Param; -import org.springframework.stereotype.Repository; - -import de.tum.cit.aet.artemis.core.exception.EntityNotFoundException; -import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; - -/** - * Spring Data repository for the ProgrammingExerciseSolutionEntry entity. - */ -@Profile(PROFILE_CORE) -@Repository -public interface ProgrammingExerciseSolutionEntryRepository extends ArtemisJpaRepository { - - /** - * Gets a solution entry with its test cases and programming exercise - * - * @param entryId The id of the solution entry - * @return The solution entry with the given ID if found - * @throws EntityNotFoundException If no solution entry with the given ID was found - */ - @NotNull - default ProgrammingExerciseSolutionEntry findByIdWithTestCaseAndProgrammingExerciseElseThrow(long entryId) throws EntityNotFoundException { - return getValueElseThrow(findByIdWithTestCaseAndProgrammingExercise(entryId), entryId); - } - - @Query(""" - SELECT se - FROM ProgrammingExerciseSolutionEntry se - LEFT JOIN FETCH se.testCase tc - LEFT JOIN FETCH tc.exercise pe - WHERE pe.id = :exerciseId - """) - Set findByExerciseIdWithTestCases(@Param("exerciseId") long exerciseId); - - /** - * Gets a solution entry with its test cases and programming exercise - * - * @param entryId The id of the solution entry - * @return The solution entry with the given ID - */ - @Query(""" - SELECT se - FROM ProgrammingExerciseSolutionEntry se - LEFT JOIN FETCH se.testCase tc - LEFT JOIN FETCH tc.exercise pe - WHERE se.id = :entryId - """) - Optional findByIdWithTestCaseAndProgrammingExercise(@Param("entryId") long entryId); - - @Query(""" - SELECT h.solutionEntries - FROM CodeHint h - WHERE h.id = :codeHintId - """) - Set findByCodeHintId(@Param("codeHintId") Long codeHintId); - - @Query(""" - SELECT t.solutionEntries - FROM ProgrammingExerciseTestCase t - WHERE t.id = :testCaseId - """) - Set findByTestCaseId(@Param("testCaseId") Long testCaseId); - - @Query(""" - SELECT se - FROM ProgrammingExerciseSolutionEntry se - LEFT JOIN FETCH se.codeHint - WHERE se.testCase.id = :testCaseId - """) - Set findByTestCaseIdWithCodeHint(@Param("testCaseId") Long testCaseId); -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/TestwiseCoverageReportEntryRepository.java b/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/TestwiseCoverageReportEntryRepository.java deleted file mode 100644 index 253adf77f45a..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/repository/hestia/TestwiseCoverageReportEntryRepository.java +++ /dev/null @@ -1,18 +0,0 @@ -package de.tum.cit.aet.artemis.programming.repository.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import org.springframework.context.annotation.Profile; -import org.springframework.stereotype.Repository; - -import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; - -/** - * Spring Data JPA repository for the TestwiseCoverageReportEntry entity. - */ -@Profile(PROFILE_CORE) -@Repository -public interface TestwiseCoverageReportEntryRepository extends ArtemisJpaRepository { - -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildScriptProviderService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildScriptProviderService.java index 487c92b3c21d..2d6e9bff2786 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildScriptProviderService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildScriptProviderService.java @@ -103,13 +103,11 @@ public String getCachedScript(String key) { * @param projectType the project type for which the template file should be returned. If omitted, a default depending on the language will be used. * @param staticAnalysis whether the static analysis template should be used * @param sequentialRuns whether the sequential runs template should be used - * @param testCoverage whether the test coverage template should be used * @return the requested template as a bash script * @throws IOException if the file does not exist */ - public String getScriptFor(ProgrammingLanguage programmingLanguage, Optional projectType, Boolean staticAnalysis, Boolean sequentialRuns, Boolean testCoverage) - throws IOException { - String templateFileName = buildTemplateName(projectType, staticAnalysis, sequentialRuns, testCoverage, "sh"); + public String getScriptFor(ProgrammingLanguage programmingLanguage, Optional projectType, Boolean staticAnalysis, Boolean sequentialRuns) throws IOException { + String templateFileName = buildTemplateName(projectType, staticAnalysis, sequentialRuns, "sh"); String uniqueKey = programmingLanguage.name().toLowerCase() + "_" + templateFileName; if (scriptCache.containsKey(uniqueKey)) { log.debug("Returning cached script for {}", uniqueKey); @@ -139,7 +137,7 @@ public String getScriptFor(ProgrammingExercise exercise) { try { ProgrammingExerciseBuildConfig buildConfig = exercise.getBuildConfig(); return getScriptFor(exercise.getProgrammingLanguage(), Optional.ofNullable(exercise.getProjectType()), exercise.isStaticCodeAnalysisEnabled(), - buildConfig.hasSequentialTestRuns(), buildConfig.isTestwiseCoverageEnabled()); + buildConfig.hasSequentialTestRuns()); } catch (IOException e) { log.error("Failed to provide build script for programming exercise " + exercise.getId(), e); @@ -153,11 +151,10 @@ public String getScriptFor(ProgrammingExercise exercise) { * @param projectType The project type for which the template file should be returned. If omitted, a default depending on the language will be used. * @param staticAnalysis whether the static analysis template should be used * @param sequentialRuns whether the sequential runs template should be used - * @param testCoverage whether the test coverage template should be used * @param fileExtension the file extension of the template file * @return The filename of the requested configuration */ - public String buildTemplateName(Optional projectType, Boolean staticAnalysis, Boolean sequentialRuns, Boolean testCoverage, String fileExtension) { + public String buildTemplateName(Optional projectType, Boolean staticAnalysis, Boolean sequentialRuns, String fileExtension) { List fileNameComponents = new ArrayList<>(); if (ProjectType.MAVEN_BLACKBOX.equals(projectType.orElse(null))) { @@ -173,9 +170,6 @@ public String buildTemplateName(Optional projectType, Boolean stati if (sequentialRuns) { fileNameComponents.add("sequential"); } - if (testCoverage) { - fileNameComponents.add("coverage"); - } return String.join("_", fileNameComponents) + "." + fileExtension; } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/CommitHistoryService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/CommitHistoryService.java index 9ee663c55c96..bf4f8496cd93 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/CommitHistoryService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/CommitHistoryService.java @@ -19,11 +19,10 @@ import org.springframework.stereotype.Service; import de.tum.cit.aet.artemis.core.service.ProfileService; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseGitDiffEntry; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseGitDiffReport; import de.tum.cit.aet.artemis.programming.domain.Repository; import de.tum.cit.aet.artemis.programming.domain.VcsRepositoryUri; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; -import de.tum.cit.aet.artemis.programming.web.GitDiffReportParserService; @Profile(PROFILE_CORE) @Service diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/GenericBuildScriptGenerationService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/GenericBuildScriptGenerationService.java index adabad775f23..5d8008d0c833 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/GenericBuildScriptGenerationService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/GenericBuildScriptGenerationService.java @@ -34,7 +34,7 @@ public String getScript(ProgrammingExercise programmingExercise) { try { ProgrammingExerciseBuildConfig buildConfig = programmingExercise.getBuildConfig(); return buildScriptProviderService.getScriptFor(programmingExercise.getProgrammingLanguage(), Optional.ofNullable(programmingExercise.getProjectType()), - programmingExercise.isStaticCodeAnalysisEnabled(), buildConfig.hasSequentialTestRuns(), buildConfig.isTestwiseCoverageEnabled()); + programmingExercise.isStaticCodeAnalysisEnabled(), buildConfig.hasSequentialTestRuns()); } catch (IOException e) { log.error("Failed to generate build script for programming exercise " + programmingExercise.getId(), e); diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/GitDiffReportParserService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/GitDiffReportParserService.java similarity index 98% rename from src/main/java/de/tum/cit/aet/artemis/programming/web/GitDiffReportParserService.java rename to src/main/java/de/tum/cit/aet/artemis/programming/service/GitDiffReportParserService.java index 37042b8cdb06..f66d85d65544 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/GitDiffReportParserService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/GitDiffReportParserService.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.web; +package de.tum.cit.aet.artemis.programming.service; import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; @@ -13,7 +13,7 @@ import org.springframework.context.annotation.Profile; import org.springframework.stereotype.Service; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseGitDiffEntry; @Profile(PROFILE_CORE) @Service diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseExportService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseExportService.java index 20d98af84bc8..76a140eb2147 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseExportService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseExportService.java @@ -74,7 +74,6 @@ import de.tum.cit.aet.artemis.programming.repository.AuxiliaryRepositoryRepository; import de.tum.cit.aet.artemis.programming.repository.BuildPlanRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseTaskService; /** * Service for exporting programming exercises. diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseFeedbackCreationService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseFeedbackCreationService.java index d22c13ebe3d8..a75efa7bf1c2 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseFeedbackCreationService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseFeedbackCreationService.java @@ -37,18 +37,17 @@ import de.tum.cit.aet.artemis.core.config.StaticCodeAnalysisConfigurer; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCaseType; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.StaticCodeAnalysisCategory; import de.tum.cit.aet.artemis.programming.domain.StaticCodeAnalysisDefaultCategory; import de.tum.cit.aet.artemis.programming.domain.StaticCodeAnalysisTool; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTestCaseType; -import de.tum.cit.aet.artemis.programming.dto.AbstractBuildResultNotificationDTO; +import de.tum.cit.aet.artemis.programming.dto.BuildResultNotification; import de.tum.cit.aet.artemis.programming.dto.StaticCodeAnalysisIssue; import de.tum.cit.aet.artemis.programming.dto.StaticCodeAnalysisReportDTO; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; import de.tum.cit.aet.artemis.programming.repository.StaticCodeAnalysisCategoryRepository; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseTaskService; /** * Service for creating feedback for programming exercises. @@ -278,7 +277,7 @@ else if (!testMessages.isEmpty()) { * @param buildResult from which to extract the test cases. * @param exercise the programming exercise for which the test cases should be extracted from the new result */ - public void extractTestCasesFromResultAndBroadcastUpdates(AbstractBuildResultNotificationDTO buildResult, ProgrammingExercise exercise) { + public void extractTestCasesFromResultAndBroadcastUpdates(BuildResultNotification buildResult, ProgrammingExercise exercise) { boolean haveTestCasesChanged = generateTestCasesFromBuildResult(buildResult, exercise); if (haveTestCasesChanged) { // Notify the client about the updated testCases @@ -296,7 +295,7 @@ public void extractTestCasesFromResultAndBroadcastUpdates(AbstractBuildResultNot * @param exercise programming exercise. * @return Returns true if the test cases have changed, false if they haven't. */ - public boolean generateTestCasesFromBuildResult(AbstractBuildResultNotificationDTO buildResult, ProgrammingExercise exercise) { + public boolean generateTestCasesFromBuildResult(BuildResultNotification buildResult, ProgrammingExercise exercise) { Set existingTestCases = testCaseRepository.findByExerciseId(exercise.getId()); // Do not generate test cases for static code analysis feedback Set testCasesFromFeedbacks = getTestCasesFromBuildResult(buildResult, exercise); @@ -375,12 +374,12 @@ public void setTestCaseType(Set testCases, Programm }); } - private Set getTestCasesFromBuildResult(AbstractBuildResultNotificationDTO buildResult, ProgrammingExercise exercise) { + private Set getTestCasesFromBuildResult(BuildResultNotification buildResult, ProgrammingExercise exercise) { Visibility defaultVisibility = exercise.getDefaultTestCaseVisibility(); - return buildResult.getBuildJobs().stream().flatMap(job -> Stream.concat(job.getFailedTests().stream(), job.getSuccessfulTests().stream())) + return buildResult.jobs().stream().flatMap(job -> Stream.concat(job.failedTests().stream(), job.successfulTests().stream())) // we use default values for weight, bonus multiplier and bonus points - .map(testCase -> new ProgrammingExerciseTestCase().testName(testCase.getName()).weight(1.0).bonusMultiplier(1.0).bonusPoints(0.0).exercise(exercise).active(true) + .map(testCase -> new ProgrammingExerciseTestCase().testName(testCase.name()).weight(1.0).bonusMultiplier(1.0).bonusPoints(0.0).exercise(exercise).active(true) .visibility(defaultVisibility)) .collect(Collectors.toSet()); } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/ProgrammingExerciseGitDiffReportService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseGitDiffReportService.java similarity index 97% rename from src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/ProgrammingExerciseGitDiffReportService.java rename to src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseGitDiffReportService.java index 6950e1216df5..785d200ad349 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/ProgrammingExerciseGitDiffReportService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseGitDiffReportService.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.service.hestia; +package de.tum.cit.aet.artemis.programming.service; import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; @@ -24,21 +24,19 @@ import de.tum.cit.aet.artemis.core.exception.InternalServerErrorException; import de.tum.cit.aet.artemis.core.service.FileService; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseGitDiffEntry; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseGitDiffReport; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; import de.tum.cit.aet.artemis.programming.domain.Repository; import de.tum.cit.aet.artemis.programming.domain.SolutionProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.TemplateProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.VcsRepositoryUri; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; +import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseGitDiffReportRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingSubmissionRepository; import de.tum.cit.aet.artemis.programming.repository.SolutionProgrammingExerciseParticipationRepository; import de.tum.cit.aet.artemis.programming.repository.TemplateProgrammingExerciseParticipationRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseGitDiffReportRepository; -import de.tum.cit.aet.artemis.programming.service.GitService; -import de.tum.cit.aet.artemis.programming.web.GitDiffReportParserService; /** * The service handling ProgrammingExerciseGitDiffReport and their ProgrammingExerciseGitDiffEntries. diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseGradingService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseGradingService.java index 7738b6615b2f..1aff8fca66f0 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseGradingService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseGradingService.java @@ -61,7 +61,7 @@ import de.tum.cit.aet.artemis.programming.domain.submissionpolicy.LockRepositoryPolicy; import de.tum.cit.aet.artemis.programming.domain.submissionpolicy.SubmissionPenaltyPolicy; import de.tum.cit.aet.artemis.programming.domain.submissionpolicy.SubmissionPolicy; -import de.tum.cit.aet.artemis.programming.dto.AbstractBuildResultNotificationDTO; +import de.tum.cit.aet.artemis.programming.dto.BuildResultNotification; import de.tum.cit.aet.artemis.programming.dto.ProgrammingExerciseGradingStatisticsDTO; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; @@ -218,9 +218,8 @@ public Result processNewProgrammingExerciseResult(@NotNull ProgrammingExercisePa * @param buildResult The build result received from the CI system. * @throws IllegalArgumentException Thrown if the result does not belong to the default branch of the exercise. */ - private void checkCorrectBranchElseThrow(final ProgrammingExerciseParticipation participation, final AbstractBuildResultNotificationDTO buildResult) - throws IllegalArgumentException { - var branchName = buildResult.getBranchNameFromAssignmentRepo(); + private void checkCorrectBranchElseThrow(final ProgrammingExerciseParticipation participation, final BuildResultNotification buildResult) throws IllegalArgumentException { + var branchName = buildResult.assignmentRepoBranchName(); // If the branch is not present, it might be because the assignment repo did not change because only the test repo was changed if (!ObjectUtils.isEmpty(branchName)) { String participationDefaultBranch = null; @@ -243,8 +242,8 @@ private void checkCorrectBranchElseThrow(final ProgrammingExerciseParticipation * * @param buildResult The build result received from the CI system. */ - private void checkHasCommitHashElseThrow(final AbstractBuildResultNotificationDTO buildResult) { - if (StringUtils.isEmpty(buildResult.getCommitHash(SubmissionType.MANUAL))) { + private void checkHasCommitHashElseThrow(final BuildResultNotification buildResult) { + if (StringUtils.isEmpty(buildResult.commitHash(SubmissionType.MANUAL))) { throw new IllegalArgumentException("The provided result does not specify the assignment commit hash. The result will not get processed."); } } @@ -256,28 +255,28 @@ private void checkHasCommitHashElseThrow(final AbstractBuildResultNotificationDT * @param buildResult The build result * @return The submission or empty if no submissions exist */ - protected Optional getSubmissionForBuildResult(Long participationId, AbstractBuildResultNotificationDTO buildResult) { + protected Optional getSubmissionForBuildResult(Long participationId, BuildResultNotification buildResult) { var submissions = programmingSubmissionRepository.findAllByParticipationIdWithResults(participationId); if (submissions.isEmpty()) { return Optional.empty(); } return submissions.stream().filter(theSubmission -> { - var commitHash = buildResult.getCommitHash(theSubmission.getType()); + var commitHash = buildResult.commitHash(theSubmission.getType()); return !ObjectUtils.isEmpty(commitHash) && commitHash.equals(theSubmission.getCommitHash()); }).max(Comparator.naturalOrder()); } @NotNull - protected ProgrammingSubmission createAndSaveFallbackSubmission(ProgrammingExerciseParticipation participation, AbstractBuildResultNotificationDTO buildResult) { - final var commitHash = buildResult.getCommitHash(SubmissionType.MANUAL); + protected ProgrammingSubmission createAndSaveFallbackSubmission(ProgrammingExerciseParticipation participation, BuildResultNotification buildResult) { + final var commitHash = buildResult.commitHash(SubmissionType.MANUAL); if (ObjectUtils.isEmpty(commitHash)) { log.error("Could not find commit hash for participation {}, build plan {}", participation.getId(), participation.getBuildPlanId()); } log.warn("Could not find pending ProgrammingSubmission for Commit Hash {} (Participation {}, Build Plan {}). Will create a new one subsequently...", commitHash, participation.getId(), participation.getBuildPlanId()); // We always take the build run date as the fallback solution - ZonedDateTime submissionDate = buildResult.getBuildRunDate(); + ZonedDateTime submissionDate = buildResult.buildRunDate(); if (!ObjectUtils.isEmpty(commitHash)) { try { // Try to get the actual date, the push might be 10s - 3min earlier, depending on how long the build takes. diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseImportBasicService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseImportBasicService.java index 8f5c1b427390..92bddf8f75d4 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseImportBasicService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseImportBasicService.java @@ -9,6 +9,7 @@ import java.util.Map; import java.util.Objects; import java.util.Optional; +import java.util.Set; import java.util.stream.Collectors; import org.springframework.context.annotation.Profile; @@ -22,34 +23,24 @@ import de.tum.cit.aet.artemis.programming.domain.AuxiliaryRepository; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseBuildConfig; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; import de.tum.cit.aet.artemis.programming.domain.RepositoryType; import de.tum.cit.aet.artemis.programming.domain.StaticCodeAnalysisCategory; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.domain.submissionpolicy.SubmissionPolicy; import de.tum.cit.aet.artemis.programming.repository.AuxiliaryRepositoryRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseBuildConfigRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; +import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTaskRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; import de.tum.cit.aet.artemis.programming.repository.StaticCodeAnalysisCategoryRepository; import de.tum.cit.aet.artemis.programming.repository.SubmissionPolicyRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ExerciseHintRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseSolutionEntryRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseTaskRepository; -import de.tum.cit.aet.artemis.programming.service.hestia.ExerciseHintService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseTaskService; import de.tum.cit.aet.artemis.programming.service.vcs.VersionControlService; @Profile(PROFILE_CORE) @Service public class ProgrammingExerciseImportBasicService { - private final ExerciseHintService exerciseHintService; - - private final ExerciseHintRepository exerciseHintRepository; - private final Optional versionControlService; private final ProgrammingExerciseParticipationService programmingExerciseParticipationService; @@ -74,22 +65,17 @@ public class ProgrammingExerciseImportBasicService { private final ProgrammingExerciseTaskService programmingExerciseTaskService; - private final ProgrammingExerciseSolutionEntryRepository solutionEntryRepository; - private final ChannelService channelService; private final ExerciseService exerciseService; - public ProgrammingExerciseImportBasicService(ExerciseHintService exerciseHintService, ExerciseHintRepository exerciseHintRepository, - Optional versionControlService, ProgrammingExerciseParticipationService programmingExerciseParticipationService, - ProgrammingExerciseTestCaseRepository programmingExerciseTestCaseRepository, StaticCodeAnalysisCategoryRepository staticCodeAnalysisCategoryRepository, - ProgrammingExerciseRepository programmingExerciseRepository, ProgrammingExerciseService programmingExerciseService, StaticCodeAnalysisService staticCodeAnalysisService, - AuxiliaryRepositoryRepository auxiliaryRepositoryRepository, SubmissionPolicyRepository submissionPolicyRepository, - ProgrammingExerciseTaskRepository programmingExerciseTaskRepository, ProgrammingExerciseTaskService programmingExerciseTaskService, - ProgrammingExerciseSolutionEntryRepository solutionEntryRepository, ChannelService channelService, + public ProgrammingExerciseImportBasicService(Optional versionControlService, + ProgrammingExerciseParticipationService programmingExerciseParticipationService, ProgrammingExerciseTestCaseRepository programmingExerciseTestCaseRepository, + StaticCodeAnalysisCategoryRepository staticCodeAnalysisCategoryRepository, ProgrammingExerciseRepository programmingExerciseRepository, + ProgrammingExerciseService programmingExerciseService, StaticCodeAnalysisService staticCodeAnalysisService, AuxiliaryRepositoryRepository auxiliaryRepositoryRepository, + SubmissionPolicyRepository submissionPolicyRepository, ProgrammingExerciseTaskRepository programmingExerciseTaskRepository, + ProgrammingExerciseTaskService programmingExerciseTaskService, ChannelService channelService, ProgrammingExerciseBuildConfigRepository programmingExerciseBuildConfigRepository, ExerciseService exerciseService) { - this.exerciseHintService = exerciseHintService; - this.exerciseHintRepository = exerciseHintRepository; this.versionControlService = versionControlService; this.programmingExerciseParticipationService = programmingExerciseParticipationService; this.programmingExerciseTestCaseRepository = programmingExerciseTestCaseRepository; @@ -101,7 +87,6 @@ public ProgrammingExerciseImportBasicService(ExerciseHintService exerciseHintSer this.submissionPolicyRepository = submissionPolicyRepository; this.programmingExerciseTaskRepository = programmingExerciseTaskRepository; this.programmingExerciseTaskService = programmingExerciseTaskService; - this.solutionEntryRepository = solutionEntryRepository; this.channelService = channelService; this.programmingExerciseBuildConfigRepository = programmingExerciseBuildConfigRepository; this.exerciseService = exerciseService; @@ -143,20 +128,16 @@ public ProgrammingExercise importProgrammingExerciseBasis(final ProgrammingExerc } // Hints, tasks, test cases and static code analysis categories - final Map newHintIdByOldId = exerciseHintService.copyExerciseHints(originalProgrammingExercise, newProgrammingExercise); - newProgrammingExercise.setBuildConfig(programmingExerciseBuildConfigRepository.save(newProgrammingExercise.getBuildConfig())); final ProgrammingExercise importedExercise = exerciseService.saveWithCompetencyLinks(newProgrammingExercise, programmingExerciseRepository::save); final Map newTestCaseIdByOldId = importTestCases(originalProgrammingExercise, importedExercise); - final Map newTaskIdByOldId = importTasks(originalProgrammingExercise, importedExercise, newTestCaseIdByOldId); - updateTaskExerciseHintReferences(originalProgrammingExercise, importedExercise, newTaskIdByOldId, newHintIdByOldId); + importTasks(originalProgrammingExercise, importedExercise, newTestCaseIdByOldId); // Set up new exercise submission policy before the solution entries are imported importSubmissionPolicy(importedExercise); // Having the submission policy in place prevents errors - importSolutionEntries(originalProgrammingExercise, importedExercise, newTestCaseIdByOldId, newHintIdByOldId); // Use the template problem statement (with ids) as a new basis (You cannot edit the problem statement while importing) // Then replace the old test ids by the newly created ones. @@ -284,33 +265,62 @@ private Map importTestCases(final ProgrammingExercise templateExerci } /** - * Copies tasks from one exercise to another. Because the tasks from the template exercise references its test cases, the - * references between tasks and test cases also need to be changed. + * Imports tasks from a template exercise to a new exercise. The tasks will get new IDs, thus being saved as a new entity. + * The remaining contents stay the same, especially the test cases. * - * @param templateExercise The template exercise which tasks should be copied - * @param targetExercise The new exercise to which all tasks should get copied to - * @param newTestCaseIdByOldId A map with the old test case id as a key and the new test case id as a value - * @return A map with the old task id as a key and the new task id as value + * @param sourceExercise The template exercise which tasks should get copied + * @param targetExercise The new exercise to which all tasks should get copied to + * @param testCaseIdMapping A map with the old test case id as a key and the new test case id as a value */ - private Map importTasks(final ProgrammingExercise templateExercise, final ProgrammingExercise targetExercise, Map newTestCaseIdByOldId) { - Map newIdByOldId = new HashMap<>(); - targetExercise.setTasks(templateExercise.getTasks().stream().map(task -> { - final var copy = new ProgrammingExerciseTask(); - - // copy everything except for the referenced exercise - copy.setTaskName(task.getTaskName()); - // change reference to newly imported test cases from the target exercise - copy.setTestCases(task.getTestCases().stream().map(testCase -> { - Long oldTestCaseId = testCase.getId(); - Long newTestCaseId = newTestCaseIdByOldId.get(oldTestCaseId); - return targetExercise.getTestCases().stream().filter(newTestCase -> Objects.equals(newTestCaseId, newTestCase.getId())).findFirst().orElseThrow(); - }).collect(Collectors.toSet())); - copy.setExercise(targetExercise); - programmingExerciseTaskRepository.save(copy); - newIdByOldId.put(task.getId(), copy.getId()); - return copy; - }).collect(Collectors.toCollection(ArrayList::new))); - return newIdByOldId; + private void importTasks(final ProgrammingExercise sourceExercise, final ProgrammingExercise targetExercise, Map testCaseIdMapping) { + // Map the tasks from the template exercise to new tasks in the target exercise + List newTasks = sourceExercise.getTasks().stream().map(templateTask -> createTaskCopy(templateTask, targetExercise, testCaseIdMapping)).toList(); + + // Set the new tasks to the target exercise + targetExercise.setTasks(new ArrayList<>(newTasks)); + } + + /** + * Creates a copy of a task from a template exercise and links it to the target exercise. The test cases of the task + * are also copied and linked to the new task. + * + * @param sourceTask The template task which should be copied + * @param targetExercise The new exercise to which the task should be linked + * @param testCaseIdMapping A map with the old test case id as a key and the new test case id as a value + * @return The new task + */ + private ProgrammingExerciseTask createTaskCopy(ProgrammingExerciseTask sourceTask, ProgrammingExercise targetExercise, Map testCaseIdMapping) { + ProgrammingExerciseTask copiedTask = new ProgrammingExerciseTask(); + + // Copy task properties + copiedTask.setTaskName(sourceTask.getTaskName()); + + // Map and set new test cases + Set mappedTestCases = sourceTask.getTestCases().stream().map(testCase -> findMappedTestCase(testCase, targetExercise, testCaseIdMapping)) + .collect(Collectors.toSet()); + copiedTask.setTestCases(mappedTestCases); + + // Link the task to the target exercise + copiedTask.setExercise(targetExercise); + + // Persist the new task + programmingExerciseTaskRepository.save(copiedTask); + return copiedTask; + } + + /** + * Finds a test case in the target exercise that corresponds to a test case in the template exercise. + * + * @param existingTestCase The test case from the template exercise + * @param targetExercise The new exercise to which the test case should be linked + * @param testCaseIdMapping A map with the old test case id as a key and the new test case id as a value + * @return The test case in the target exercise + */ + private ProgrammingExerciseTestCase findMappedTestCase(ProgrammingExerciseTestCase existingTestCase, ProgrammingExercise targetExercise, Map testCaseIdMapping) { + Long newTestCaseId = testCaseIdMapping.get(existingTestCase.getId()); + + return targetExercise.getTestCases().stream().filter(newTestCase -> Objects.equals(newTestCaseId, newTestCase.getId())).findFirst() + .orElseThrow(() -> new IllegalArgumentException("Test case not found for ID: " + newTestCaseId)); } /** @@ -380,68 +390,4 @@ else if (newExercise.isCourseExercise() && newExercise.getPlagiarismDetectionCon newExercise.setPlagiarismDetectionConfig(null); } } - - /** - * Updates the newly imported exercise hints to reference the newly imported tasks they belong to. - * - * @param templateExercise The template exercise which tasks should be copied - * @param targetExercise The new exercise to which all tasks should get copied to - * @param newTaskIdByOldId A map with the old task id as a key and the new task id as a value - * @param newHintIdByOldId A map with the old hint id as a key and the new hint id as a value - */ - private void updateTaskExerciseHintReferences(final ProgrammingExercise templateExercise, final ProgrammingExercise targetExercise, Map newTaskIdByOldId, - Map newHintIdByOldId) { - templateExercise.getExerciseHints().forEach(templateExerciseHint -> { - var templateTask = templateExerciseHint.getProgrammingExerciseTask(); - if (templateTask == null) { - return; - } - var targetTask = targetExercise.getTasks().stream().filter(newTask -> Objects.equals(newTask.getId(), newTaskIdByOldId.get(templateTask.getId()))).findAny() - .orElseThrow(); - var targetExerciseHint = targetExercise.getExerciseHints().stream() - .filter(newHint -> Objects.equals(newHint.getId(), newHintIdByOldId.get(templateExerciseHint.getId()))).findAny().orElseThrow(); - - targetExerciseHint.setProgrammingExerciseTask(targetTask); - exerciseHintRepository.save(targetExerciseHint); - targetTask.getExerciseHints().add(targetExerciseHint); - }); - } - - /** - * Copies solution entries from one exercise to another. Because the solution entries from the template exercise - * references its test cases and code hint, the references between them also need to be changed. - * - * @param templateExercise The template exercise which tasks should be copied - * @param targetExercise The new exercise to which all tasks should get copied to - * @param newTestCaseIdByOldId A map with the old test case id as a key and the new test case id as a value - * @param newHintIdByOldId A map with the old hint id as a key and the new hint id as a value - */ - private void importSolutionEntries(final ProgrammingExercise templateExercise, final ProgrammingExercise targetExercise, Map newTestCaseIdByOldId, - Map newHintIdByOldId) { - templateExercise.getTestCases().forEach(testCase -> { - var newSolutionEntries = solutionEntryRepository.findByTestCaseIdWithCodeHint(testCase.getId()).stream().map(solutionEntry -> { - Long newTestCaseId = newTestCaseIdByOldId.get(testCase.getId()); - var targetTestCase = targetExercise.getTestCases().stream().filter(newTestCase -> Objects.equals(newTestCaseId, newTestCase.getId())).findFirst().orElseThrow(); - - CodeHint codeHint = null; - if (solutionEntry.getCodeHint() != null) { - Long newHintId = newHintIdByOldId.get(solutionEntry.getCodeHint().getId()); - codeHint = (CodeHint) targetExercise.getExerciseHints().stream().filter(newHint -> Objects.equals(newHintId, newHint.getId())).findFirst().orElseThrow(); - } - var copy = new ProgrammingExerciseSolutionEntry(); - copy.setCode(solutionEntry.getCode()); - copy.setPreviousCode(solutionEntry.getPreviousCode()); - copy.setLine(solutionEntry.getLine()); - copy.setPreviousLine(solutionEntry.getPreviousLine()); - copy.setTestCase(targetTestCase); - targetTestCase.getSolutionEntries().add(copy); - copy.setCodeHint(codeHint); - if (codeHint != null) { - codeHint.getSolutionEntries().add(copy); - } - return copy; - }).collect(Collectors.toSet()); - solutionEntryRepository.saveAll(newSolutionEntries); - }); - } } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseImportService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseImportService.java index 710837ac4fb4..0c308cc80c0a 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseImportService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseImportService.java @@ -35,7 +35,6 @@ import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationService; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationTriggerService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseTaskService; import de.tum.cit.aet.artemis.programming.service.vcs.VersionControlService; @Profile(PROFILE_CORE) diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseRepositoryService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseRepositoryService.java index 43523ec566f1..ffbf6aa37f0a 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseRepositoryService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseRepositoryService.java @@ -404,8 +404,6 @@ private void setupJVMTestTemplateAndPush(final RepositoryResources resources, fi final Map sectionsMap = new HashMap<>(); // Keep or delete static code analysis configuration in the build configuration file sectionsMap.put("static-code-analysis", Boolean.TRUE.equals(programmingExercise.isStaticCodeAnalysisEnabled())); - // Keep or delete testwise coverage configuration in the build file - sectionsMap.put("record-testwise-coverage", Boolean.TRUE.equals(programmingExercise.getBuildConfig().isTestwiseCoverageEnabled())); if (programmingExercise.getBuildConfig().hasSequentialTestRuns()) { setupTestTemplateSequentialTestRuns(resources, templatePath, projectTemplatePath, projectType, sectionsMap); diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseService.java index d85af7904c4c..d03c14a5cbc5 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseService.java @@ -17,7 +17,6 @@ import java.time.ZonedDateTime; import java.util.ArrayList; import java.util.Arrays; -import java.util.Collection; import java.util.Collections; import java.util.HashSet; import java.util.List; @@ -26,7 +25,6 @@ import java.util.Set; import java.util.regex.Matcher; import java.util.regex.Pattern; -import java.util.stream.Collectors; import jakarta.annotation.Nullable; @@ -70,7 +68,7 @@ import de.tum.cit.aet.artemis.programming.domain.AuxiliaryRepository; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseBuildConfig; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.ProjectType; import de.tum.cit.aet.artemis.programming.domain.Repository; @@ -78,23 +76,19 @@ import de.tum.cit.aet.artemis.programming.domain.SolutionProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.TemplateProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.VcsRepositoryUri; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; import de.tum.cit.aet.artemis.programming.repository.AuxiliaryRepositoryRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseBuildConfigRepository; +import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseGitDiffReportRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseStudentParticipationRepository; +import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTaskRepository; import de.tum.cit.aet.artemis.programming.repository.SolutionProgrammingExerciseParticipationRepository; import de.tum.cit.aet.artemis.programming.repository.TemplateProgrammingExerciseParticipationRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseGitDiffReportRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseSolutionEntryRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseTaskRepository; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusTemplateService; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; import de.tum.cit.aet.artemis.programming.service.ci.CIPermission; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationService; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationTriggerService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseTaskService; import de.tum.cit.aet.artemis.programming.service.structureoraclegenerator.OracleGenerator; import de.tum.cit.aet.artemis.programming.service.vcs.VersionControlService; @@ -163,8 +157,6 @@ public class ProgrammingExerciseService { private final ProgrammingExerciseTaskRepository programmingExerciseTaskRepository; - private final ProgrammingExerciseSolutionEntryRepository programmingExerciseSolutionEntryRepository; - private final ProgrammingExerciseTaskService programmingExerciseTaskService; private final ProgrammingExerciseGitDiffReportRepository programmingExerciseGitDiffReportRepository; @@ -206,11 +198,11 @@ public ProgrammingExerciseService(ProgrammingExerciseRepository programmingExerc ParticipationRepository participationRepository, ResultRepository resultRepository, UserRepository userRepository, GroupNotificationScheduleService groupNotificationScheduleService, InstanceMessageSendService instanceMessageSendService, AuxiliaryRepositoryRepository auxiliaryRepositoryRepository, ProgrammingExerciseTaskRepository programmingExerciseTaskRepository, - ProgrammingExerciseSolutionEntryRepository programmingExerciseSolutionEntryRepository, ProgrammingExerciseTaskService programmingExerciseTaskService, - ProgrammingExerciseGitDiffReportRepository programmingExerciseGitDiffReportRepository, ExerciseSpecificationService exerciseSpecificationService, - ProgrammingExerciseRepositoryService programmingExerciseRepositoryService, AuxiliaryRepositoryService auxiliaryRepositoryService, - SubmissionPolicyService submissionPolicyService, Optional programmingLanguageFeatureService, ChannelService channelService, - ProgrammingSubmissionService programmingSubmissionService, Optional irisSettingsService, Optional aeolusTemplateService, + ProgrammingExerciseTaskService programmingExerciseTaskService, ProgrammingExerciseGitDiffReportRepository programmingExerciseGitDiffReportRepository, + ExerciseSpecificationService exerciseSpecificationService, ProgrammingExerciseRepositoryService programmingExerciseRepositoryService, + AuxiliaryRepositoryService auxiliaryRepositoryService, SubmissionPolicyService submissionPolicyService, + Optional programmingLanguageFeatureService, ChannelService channelService, ProgrammingSubmissionService programmingSubmissionService, + Optional irisSettingsService, Optional aeolusTemplateService, Optional buildScriptGenerationService, ProgrammingExerciseStudentParticipationRepository programmingExerciseStudentParticipationRepository, ProfileService profileService, ExerciseService exerciseService, ProgrammingExerciseBuildConfigRepository programmingExerciseBuildConfigRepository, CompetencyProgressApi competencyProgressApi, @@ -230,7 +222,6 @@ public ProgrammingExerciseService(ProgrammingExerciseRepository programmingExerc this.instanceMessageSendService = instanceMessageSendService; this.auxiliaryRepositoryRepository = auxiliaryRepositoryRepository; this.programmingExerciseTaskRepository = programmingExerciseTaskRepository; - this.programmingExerciseSolutionEntryRepository = programmingExerciseSolutionEntryRepository; this.programmingExerciseTaskService = programmingExerciseTaskService; this.programmingExerciseGitDiffReportRepository = programmingExerciseGitDiffReportRepository; this.exerciseSpecificationService = exerciseSpecificationService; @@ -405,11 +396,6 @@ public void validateNewProgrammingExerciseSettings(ProgrammingExercise programmi throw new BadRequestAlertException("Checkout solution repository is not supported for this programming language", "Exercise", "checkoutSolutionRepositoryNotSupported"); } - // Check if testwise coverage analysis is enabled - if (Boolean.TRUE.equals(buildConfig.isTestwiseCoverageEnabled()) && !programmingLanguageFeature.testwiseCoverageAnalysisSupported()) { - throw new BadRequestAlertException("Testwise coverage analysis is not supported for this language", "Exercise", "testwiseCoverageAnalysisNotSupported"); - } - programmingExerciseRepository.validateCourseSettings(programmingExercise, course); validateStaticCodeAnalysisSettings(programmingExercise); @@ -1033,12 +1019,9 @@ public boolean preCheckProjectExistsOnVCSOrCI(ProgrammingExercise programmingExe * * @param exerciseId of the exercise */ - public void deleteTasksWithSolutionEntries(Long exerciseId) { - List tasks = programmingExerciseTaskRepository.findByExerciseIdWithTestCaseAndSolutionEntriesElseThrow(exerciseId); - Set solutionEntries = tasks.stream().map(ProgrammingExerciseTask::getTestCases).flatMap(Collection::stream) - .map(ProgrammingExerciseTestCase::getSolutionEntries).flatMap(Collection::stream).collect(Collectors.toSet()); + public void deleteTasks(Long exerciseId) { + List tasks = programmingExerciseTaskRepository.findByExerciseIdWithTestCaseElseThrow(exerciseId); programmingExerciseTaskRepository.deleteAll(tasks); - programmingExerciseSolutionEntryRepository.deleteAll(solutionEntries); } private void resetAllStudentBuildPlanIdsForExercise(ProgrammingExercise programmingExercise) { diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/ProgrammingExerciseTaskService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseTaskService.java similarity index 91% rename from src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/ProgrammingExerciseTaskService.java rename to src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseTaskService.java index 0b1a14be8646..3ec052802f83 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/ProgrammingExerciseTaskService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseTaskService.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.service.hestia; +package de.tum.cit.aet.artemis.programming.service; import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; @@ -20,11 +20,10 @@ import org.springframework.stereotype.Service; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; +import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTaskRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ExerciseHintRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseTaskRepository; @Profile(PROFILE_CORE) @Service @@ -34,8 +33,6 @@ public class ProgrammingExerciseTaskService { private final ProgrammingExerciseTestCaseRepository programmingExerciseTestCaseRepository; - private final ExerciseHintRepository exerciseHintRepository; - /** * Pattern that is used to extract the tasks (capturing group {@code name}) and test case names (capturing groups {@code tests}) from the problem statement. * Example: "[task][Implement BubbleSort](testBubbleSort,testBubbleSortHidden)". Following groups are extracted by the capturing groups: @@ -85,21 +82,18 @@ public class ProgrammingExerciseTaskService { private static final Pattern TESTID_PATTERN = Pattern.compile(TESTID_START + "(\\d+)" + TESTID_END); public ProgrammingExerciseTaskService(ProgrammingExerciseTaskRepository programmingExerciseTaskRepository, - ProgrammingExerciseTestCaseRepository programmingExerciseTestCaseRepository, ExerciseHintRepository exerciseHintRepository) { + ProgrammingExerciseTestCaseRepository programmingExerciseTestCaseRepository) { this.programmingExerciseTaskRepository = programmingExerciseTaskRepository; this.programmingExerciseTestCaseRepository = programmingExerciseTestCaseRepository; - this.exerciseHintRepository = exerciseHintRepository; } /** - * Deletes a ProgrammingExerciseTask together with its CodeHints + * Deletes a ProgrammingExerciseTask * This has to be manually done, as there is no orphanRemoval between the two entities * * @param task The task to delete */ public void delete(ProgrammingExerciseTask task) { - var exerciseHints = exerciseHintRepository.findByTaskId(task.getId()); - exerciseHintRepository.deleteAll(exerciseHints); programmingExerciseTaskRepository.delete(task); } @@ -109,14 +103,13 @@ public void delete(ProgrammingExerciseTask task) { * If there is already a task with the same test cases as a new one, but with a different name the existing one will be renamed. * * @param exercise The programming exercise to extract the tasks from - * @return The current tasks of the exercise */ - public Set updateTasksFromProblemStatement(ProgrammingExercise exercise) { + public void updateTasksFromProblemStatement(ProgrammingExercise exercise) { var previousTasks = programmingExerciseTaskRepository.findByExerciseIdWithTestCases(exercise.getId()); var extractedTasks = new HashSet<>(extractTasks(exercise)); // No changes if (previousTasks.equals(extractedTasks)) { - return previousTasks; + return; } // Add all tasks that did not change var tasksToBeSaved = new HashSet<>(previousTasks); @@ -151,24 +144,7 @@ public Set updateTasksFromProblemStatement(ProgrammingE for (ProgrammingExerciseTask task : tasksToBeSaved) { task.setExercise(exercise); } - return new HashSet<>(programmingExerciseTaskRepository.saveAll(tasksToBeSaved)); - } - - /** - * Gets the tasks of a programming exercise sorted by their order in the problem statement - * TODO: Replace this with an @OrderColumn on tasks in ProgrammingExercise - * - * @param exercise The programming exercise - * @return The sorted tasks - */ - public List getSortedTasks(ProgrammingExercise exercise) { - var unsortedTasks = programmingExerciseTaskRepository.findByExerciseIdWithTestCases(exercise.getId()); - var sortedExtractedTasks = extractTasks(exercise); - return sortedExtractedTasks.stream() - .map(extractedTask -> unsortedTasks.stream() - .filter(task -> task.getTaskName().equals(extractedTask.getTaskName()) && task.getTestCases().equals(extractedTask.getTestCases())).findFirst() - .orElse(null)) - .distinct().filter(Objects::nonNull).toList(); + programmingExerciseTaskRepository.saveAll(tasksToBeSaved); } /** @@ -178,7 +154,7 @@ public List getSortedTasks(ProgrammingExercise exercise * @return Set of all tasks and its test cases */ public Set getTasksWithoutInactiveTestCases(long exerciseId) { - return programmingExerciseTaskRepository.findByExerciseIdWithTestCaseAndSolutionEntriesElseThrow(exerciseId).stream() + return programmingExerciseTaskRepository.findByExerciseIdWithTestCaseElseThrow(exerciseId).stream() .peek(task -> task.getTestCases().removeIf(Predicate.not(ProgrammingExerciseTestCase::isActive))).collect(Collectors.toSet()); } @@ -187,10 +163,10 @@ public Set getTasksWithoutInactiveTestCases(long exerci * Additionally, adds a new task for all test cases with no manually assigned task and adds all tests to that task * * @param exerciseId of the programming exercise - * @return Set of all tasks including one for not manually assigned tests + * @return List of all tasks including one for not manually assigned tests */ public List getTasksWithUnassignedTestCases(long exerciseId) { - List tasks = programmingExerciseTaskRepository.findByExerciseIdWithTestCaseAndSolutionEntriesElseThrow(exerciseId); + List tasks = programmingExerciseTaskRepository.findByExerciseIdWithTestCaseElseThrow(exerciseId); Set testsWithTasks = tasks.stream().flatMap(task -> task.getTestCases().stream()).collect(Collectors.toSet()); diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseTestCaseService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseTestCaseService.java index f3770b417e5b..f06a901b8479 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseTestCaseService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseTestCaseService.java @@ -27,7 +27,6 @@ import de.tum.cit.aet.artemis.programming.dto.ProgrammingExerciseTestCaseDTO; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseTaskService; @Profile(PROFILE_CORE) @Service diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingLanguageFeature.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingLanguageFeature.java index 88bee465e155..df07a8f6d5b3 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingLanguageFeature.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingLanguageFeature.java @@ -13,6 +13,5 @@ */ @JsonInclude(JsonInclude.Include.NON_EMPTY) public record ProgrammingLanguageFeature(ProgrammingLanguage programmingLanguage, boolean sequentialTestRuns, boolean staticCodeAnalysis, boolean plagiarismCheckSupported, - boolean packageNameRequired, boolean checkoutSolutionRepositoryAllowed, List projectTypes, boolean testwiseCoverageAnalysisSupported, - boolean auxiliaryRepositoriesSupported) { + boolean packageNameRequired, boolean checkoutSolutionRepositoryAllowed, List projectTypes, boolean auxiliaryRepositoriesSupported) { } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingSubmissionService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingSubmissionService.java index 9de4485f16b2..e1ae83935eec 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingSubmissionService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingSubmissionService.java @@ -67,7 +67,6 @@ import de.tum.cit.aet.artemis.programming.repository.ProgrammingSubmissionRepository; import de.tum.cit.aet.artemis.programming.repository.SubmissionPolicyRepository; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationTriggerService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseGitDiffReportService; import de.tum.cit.aet.artemis.programming.service.vcs.VersionControlService; // TODO: this class has too many dependencies to other services. We should reduce this diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/Action.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/Action.java deleted file mode 100644 index be22f81ddad9..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/Action.java +++ /dev/null @@ -1,86 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.aeolus; - -import java.util.List; -import java.util.Map; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; - -/** - * Base class for the actions that can be defined in a {@link Windfile} - */ -// TODO: remove and convert subclasses into Records -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public abstract class Action { - - private String name; - - private Map parameters; - - private Map environment; - - private List results; - - private String workdir; - - private boolean runAlways; - - private String platform; - - public Map getParameters() { - return parameters; - } - - public void setParameters(Map parameters) { - this.parameters = parameters; - } - - public Map getEnvironment() { - return environment; - } - - public void setEnvironment(Map environment) { - this.environment = environment; - } - - public boolean isRunAlways() { - return runAlways; - } - - public void setRunAlways(boolean runAlways) { - this.runAlways = runAlways; - } - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - - public List getResults() { - return results; - } - - public void setResults(List results) { - this.results = results; - } - - public String getWorkdir() { - return workdir; - } - - public void setWorkdir(String workdir) { - this.workdir = workdir; - } - - public String getPlatform() { - return platform; - } - - public void setPlatform(String platform) { - this.platform = platform; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/ActionDeserializer.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/ActionDeserializer.java index 65d42487e398..32463ba0d841 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/ActionDeserializer.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/ActionDeserializer.java @@ -8,6 +8,10 @@ import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.ObjectMapper; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Action; +import de.tum.cit.aet.artemis.programming.dto.aeolus.PlatformAction; +import de.tum.cit.aet.artemis.programming.dto.aeolus.ScriptAction; + /** * Deserializer for {@link Action} that determines the type of the action based on the content of the JSON. */ diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusBuildPlanService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusBuildPlanService.java index 9cb410de37fa..9dd35b363709 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusBuildPlanService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusBuildPlanService.java @@ -35,6 +35,8 @@ import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.VcsRepositoryUri; import de.tum.cit.aet.artemis.programming.dto.AeolusGenerationResponseDTO; +import de.tum.cit.aet.artemis.programming.dto.aeolus.AeolusRepository; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; import de.tum.cit.aet.artemis.programming.service.InternalUrlService; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationService; diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusBuildScriptGenerationService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusBuildScriptGenerationService.java index 8830adb96389..57141ab1a84a 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusBuildScriptGenerationService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusBuildScriptGenerationService.java @@ -10,6 +10,8 @@ import de.tum.cit.aet.artemis.core.service.ProfileService; import de.tum.cit.aet.artemis.programming.domain.AeolusTarget; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; +import de.tum.cit.aet.artemis.programming.dto.aeolus.WindfileMetadata; import de.tum.cit.aet.artemis.programming.service.BuildScriptGenerationService; import de.tum.cit.aet.artemis.programming.service.BuildScriptProviderService; @@ -52,12 +54,12 @@ public String getScript(ProgrammingExercise programmingExercise) throws JsonProc windfile = aeolusTemplateService.getDefaultWindfileFor(programmingExercise); } if (windfile != null) { - WindfileMetadata oldMetadata = windfile.getMetadata(); + WindfileMetadata oldMetadata = windfile.metadata(); // Creating a new instance of WindfileMetadata with placeholder values for id, name, and description, // and copying the rest of the fields from oldMetadata WindfileMetadata updatedMetadata = new WindfileMetadata("not-used", "not-used", "not-used", oldMetadata.author(), oldMetadata.gitCredentials(), oldMetadata.docker(), oldMetadata.resultHook(), oldMetadata.resultHookCredentials()); - windfile.setMetadata(updatedMetadata); + windfile = new Windfile(windfile, updatedMetadata); return aeolusBuildPlanService.generateBuildScript(windfile, AeolusTarget.CLI); } return null; diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusTemplateService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusTemplateService.java index 64321ad3b61d..f2e51ab78496 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusTemplateService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/AeolusTemplateService.java @@ -26,6 +26,10 @@ import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseBuildConfig; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.ProjectType; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Action; +import de.tum.cit.aet.artemis.programming.dto.aeolus.DockerConfig; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; +import de.tum.cit.aet.artemis.programming.dto.aeolus.WindfileMetadata; import de.tum.cit.aet.artemis.programming.service.BuildScriptProviderService; import de.tum.cit.aet.artemis.programming.web.localci.AeolusTemplateResource; @@ -84,7 +88,7 @@ public void cacheOnBoot() { script = buildScriptProviderService.replacePlaceholders(script, null, null, null); } Windfile windfile = readWindfile(script); - this.addInstanceVariablesToWindfile(windfile, ProgrammingLanguage.valueOf(directory.toUpperCase()), optionalProjectType); + windfile = addInstanceVariablesToWindfile(windfile, ProgrammingLanguage.valueOf(directory.toUpperCase()), optionalProjectType); templateCache.put(uniqueKey, windfile); } catch (IOException | IllegalArgumentException e) { @@ -127,17 +131,15 @@ private static Windfile readWindfile(String yaml) throws IOException { * @param projectType the project type for which the template file should be returned. If omitted, a default depending on the language will be used. * @param staticAnalysis whether the static analysis template should be used * @param sequentialRuns whether the sequential runs template should be used - * @param testCoverage whether the test coverage template should be used * @return the requested template as a {@link Windfile} object * @throws IOException if the file does not exist */ - public Windfile getWindfileFor(ProgrammingLanguage programmingLanguage, Optional projectType, Boolean staticAnalysis, Boolean sequentialRuns, Boolean testCoverage) - throws IOException { + public Windfile getWindfileFor(ProgrammingLanguage programmingLanguage, Optional projectType, Boolean staticAnalysis, Boolean sequentialRuns) throws IOException { if (programmingLanguage.equals(ProgrammingLanguage.JAVA) && projectType.isEmpty()) { // to be backwards compatible, we assume that java exercises without project type are plain maven projects projectType = Optional.of(ProjectType.PLAIN_MAVEN); } - String templateFileName = buildScriptProviderService.buildTemplateName(projectType, staticAnalysis, sequentialRuns, testCoverage, "yaml"); + String templateFileName = buildScriptProviderService.buildTemplateName(projectType, staticAnalysis, sequentialRuns, "yaml"); String uniqueKey = programmingLanguage.name().toLowerCase() + "_" + templateFileName; if (templateCache.containsKey(uniqueKey)) { return templateCache.get(uniqueKey); @@ -151,7 +153,7 @@ public Windfile getWindfileFor(ProgrammingLanguage programmingLanguage, Optional scriptCache = buildScriptProviderService.replacePlaceholders(scriptCache, null, null, null); } Windfile windfile = readWindfile(scriptCache); - this.addInstanceVariablesToWindfile(windfile, programmingLanguage, projectType); + windfile = addInstanceVariablesToWindfile(windfile, programmingLanguage, projectType); templateCache.put(uniqueKey, windfile); return windfile; } @@ -184,7 +186,7 @@ public Windfile getDefaultWindfileFor(ProgrammingExercise exercise) { try { ProgrammingExerciseBuildConfig buildConfig = exercise.getBuildConfig(); return getWindfileFor(exercise.getProgrammingLanguage(), Optional.ofNullable(exercise.getProjectType()), exercise.isStaticCodeAnalysisEnabled(), - buildConfig.hasSequentialTestRuns(), buildConfig.isTestwiseCoverageEnabled()); + buildConfig.hasSequentialTestRuns()); } catch (IOException e) { log.info("No windfile for the settings of exercise {}", exercise.getId(), e); @@ -202,23 +204,24 @@ public Windfile getDefaultWindfileFor(ProgrammingExercise exercise) { * @param windfile the Windfile template to be updated with Docker configuration * @param language the programming language used, which determines the Docker image and flags * @param projectType an optional specifying the project type; influences the Docker configuration + * @return the updated Windfile instance with Docker configuration */ - private void addInstanceVariablesToWindfile(Windfile windfile, ProgrammingLanguage language, Optional projectType) { + private Windfile addInstanceVariablesToWindfile(Windfile windfile, ProgrammingLanguage language, Optional projectType) { - WindfileMetadata metadata = windfile.getMetadata(); + WindfileMetadata metadata = windfile.metadata(); if (metadata == null) { metadata = new WindfileMetadata(); } if (projectType.isPresent() && ProjectType.XCODE.equals(projectType.get())) { // xcode does not support docker metadata = new WindfileMetadata(); - windfile.setMetadata(metadata); - return; } - String image = programmingLanguageConfiguration.getImage(language, projectType); - DockerConfig dockerConfig = new DockerConfig(image, null, null, programmingLanguageConfiguration.getDefaultDockerFlags()); - metadata = new WindfileMetadata(metadata.name(), metadata.id(), metadata.description(), metadata.author(), metadata.gitCredentials(), dockerConfig, metadata.resultHook(), - metadata.resultHookCredentials()); - windfile.setMetadata(metadata); + else { + String image = programmingLanguageConfiguration.getImage(language, projectType); + DockerConfig dockerConfig = new DockerConfig(image, null, null, programmingLanguageConfiguration.getDefaultDockerFlags()); + metadata = new WindfileMetadata(metadata.name(), metadata.id(), metadata.description(), metadata.author(), metadata.gitCredentials(), dockerConfig, + metadata.resultHook(), metadata.resultHookCredentials()); + } + return new Windfile(windfile, metadata); } } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/PlatformAction.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/PlatformAction.java deleted file mode 100644 index 5c161ffe798b..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/PlatformAction.java +++ /dev/null @@ -1,33 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.aeolus; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; - -/** - * Represents a CI action that is intended to run only on a specific target, can be used in a {@link Windfile}. - */ -// TODO: convert into Record -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public class PlatformAction extends Action { - - private String kind; - - private String type; - - public String getKind() { - return kind; - } - - public void setKind(String kind) { - this.kind = kind; - } - - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/Windfile.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/Windfile.java deleted file mode 100644 index a9f2382591d4..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/aeolus/Windfile.java +++ /dev/null @@ -1,123 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.aeolus; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.fasterxml.jackson.databind.module.SimpleModule; - -/** - * Represents a windfile, the definition file for an aeolus build plan that - * can then be used to generate a Jenkinsfile. - */ -// TODO convert into Record -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public class Windfile { - - private static final ObjectMapper mapper = new ObjectMapper(); - - private String api; - - private WindfileMetadata metadata; - - private List actions = new ArrayList<>(); - - private Map repositories = new HashMap<>(); - - public String getApi() { - return api; - } - - public void setApi(String api) { - this.api = api; - } - - public WindfileMetadata getMetadata() { - return metadata; - } - - public void setMetadata(WindfileMetadata metadata) { - this.metadata = metadata; - } - - public List getActions() { - return actions; - } - - public void setActions(List actions) { - this.actions = actions; - } - - /** - * Gets the script actions of a windfile. - * - * @return the script actions of a windfile. - */ - public List getScriptActions() { - List scriptActions = new ArrayList<>(); - for (Action action : actions) { - if (action instanceof ScriptAction) { - scriptActions.add((ScriptAction) action); - } - } - return scriptActions; - } - - public void setRepositories(Map repositories) { - this.repositories = repositories; - } - - public Map getRepositories() { - return repositories; - } - - /** - * Sets the pre-processing metadata for the windfile. - * - * @param id the id of the windfile. - * @param name the name of the windfile. - * @param gitCredentials the git credentials of the windfile. - * @param resultHook the result hook of the windfile. - * @param description the description of the windfile. - * @param repositories the repositories of the windfile. - * @param resultHookCredentials the credentials for the result hook of the windfile. - */ - public void setPreProcessingMetadata(String id, String name, String gitCredentials, String resultHook, String description, Map repositories, - String resultHookCredentials) { - this.setMetadata(new WindfileMetadata(name, id, description, null, gitCredentials, null, resultHook, resultHookCredentials)); - this.setRepositories(repositories); - } - - /** - * Deserializes a windfile from a json string. - * - * @param json the json string to deserialize. - * @return the deserialized windfile. - * @throws JsonProcessingException if the json string is not valid. - */ - public static Windfile deserialize(String json) throws JsonProcessingException { - SimpleModule module = new SimpleModule(); - module.addDeserializer(Action.class, new ActionDeserializer()); - mapper.registerModule(module); - return mapper.readValue(json, Windfile.class); - } - - /** - * Collects the results of all actions of a windfile. - * - * @return the results of all actions of this windfile - */ - public List getResults() { - List results = new ArrayList<>(); - for (Action action : actions.stream().filter(action -> action.getResults() != null && !action.getResults().isEmpty()).toList()) { - results.addAll(action.getResults()); - } - return results; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/AbstractContinuousIntegrationResultService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/AbstractContinuousIntegrationResultService.java index 238d4ffca0fc..2f540cd869b7 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/AbstractContinuousIntegrationResultService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/AbstractContinuousIntegrationResultService.java @@ -11,13 +11,12 @@ import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.build.BuildLogEntry; -import de.tum.cit.aet.artemis.programming.dto.AbstractBuildResultNotificationDTO; -import de.tum.cit.aet.artemis.programming.dto.BuildJobDTOInterface; +import de.tum.cit.aet.artemis.programming.dto.BuildJobInterface; +import de.tum.cit.aet.artemis.programming.dto.BuildResultNotification; import de.tum.cit.aet.artemis.programming.repository.BuildLogStatisticsEntryRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseBuildConfigRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseFeedbackCreationService; -import de.tum.cit.aet.artemis.programming.service.hestia.TestwiseCoverageService; public abstract class AbstractContinuousIntegrationResultService implements ContinuousIntegrationResultService { @@ -25,32 +24,29 @@ public abstract class AbstractContinuousIntegrationResultService implements Cont protected final BuildLogStatisticsEntryRepository buildLogStatisticsEntryRepository; - protected final TestwiseCoverageService testwiseCoverageService; - protected final ProgrammingExerciseFeedbackCreationService feedbackCreationService; protected final ProgrammingExerciseBuildConfigRepository programmingExerciseBuildConfigRepository; protected AbstractContinuousIntegrationResultService(ProgrammingExerciseTestCaseRepository testCaseRepository, - BuildLogStatisticsEntryRepository buildLogStatisticsEntryRepository, TestwiseCoverageService testwiseCoverageService, - ProgrammingExerciseFeedbackCreationService feedbackCreationService, ProgrammingExerciseBuildConfigRepository programmingExerciseBuildConfigRepository) { + BuildLogStatisticsEntryRepository buildLogStatisticsEntryRepository, ProgrammingExerciseFeedbackCreationService feedbackCreationService, + ProgrammingExerciseBuildConfigRepository programmingExerciseBuildConfigRepository) { this.testCaseRepository = testCaseRepository; this.buildLogStatisticsEntryRepository = buildLogStatisticsEntryRepository; - this.testwiseCoverageService = testwiseCoverageService; this.feedbackCreationService = feedbackCreationService; this.programmingExerciseBuildConfigRepository = programmingExerciseBuildConfigRepository; } @Override - public Result createResultFromBuildResult(AbstractBuildResultNotificationDTO buildResult, ProgrammingExerciseParticipation participation) { + public Result createResultFromBuildResult(BuildResultNotification buildResult, ProgrammingExerciseParticipation participation) { ProgrammingExercise exercise = participation.getProgrammingExercise(); final var result = new Result(); result.setAssessmentType(AssessmentType.AUTOMATIC); result.setSuccessful(buildResult.isBuildSuccessful()); - result.setCompletionDate(buildResult.getBuildRunDate()); + result.setCompletionDate(buildResult.buildRunDate()); // this only sets the score to a temporary value, the real score is calculated in the grading service - result.setScore(buildResult.getBuildScore(), exercise.getCourseViaExerciseGroupOrCourseMember()); + result.setScore(buildResult.buildScore(), exercise.getCourseViaExerciseGroupOrCourseMember()); result.setParticipation((Participation) participation); addFeedbackToResult(result, buildResult); @@ -63,8 +59,8 @@ public Result createResultFromBuildResult(AbstractBuildResultNotificationDTO bui * @param result the result for which the feedback should be added * @param buildResult The build result */ - private void addFeedbackToResult(Result result, AbstractBuildResultNotificationDTO buildResult) { - final var jobs = buildResult.getBuildJobs(); + private void addFeedbackToResult(Result result, BuildResultNotification buildResult) { + final var jobs = buildResult.jobs(); final var programmingExercise = (ProgrammingExercise) result.getParticipation().getExercise(); // 1) add feedback for failed and passed test cases @@ -72,32 +68,28 @@ private void addFeedbackToResult(Result result, AbstractBuildResultNotificationD // 2) process static code analysis feedback addStaticCodeAnalysisFeedbackToResult(result, buildResult, programmingExercise); - - // 3) process testwise coverage analysis report - addTestwiseCoverageReportToResult(result, buildResult, programmingExercise); } - private void addTestCaseFeedbacksToResult(Result result, List jobs, ProgrammingExercise programmingExercise) { + private void addTestCaseFeedbacksToResult(Result result, List jobs, ProgrammingExercise programmingExercise) { var activeTestCases = testCaseRepository.findByExerciseIdAndActive(programmingExercise.getId(), true); - for (final var job : jobs) { - for (final var failedTest : job.getFailedTests()) { - result.addFeedback( - feedbackCreationService.createFeedbackFromTestCase(failedTest.getName(), failedTest.getTestMessages(), false, programmingExercise, activeTestCases)); - } - result.setTestCaseCount(result.getTestCaseCount() + job.getFailedTests().size()); + jobs.forEach(job -> { + job.failedTests().forEach(failedTest -> { + result.addFeedback(feedbackCreationService.createFeedbackFromTestCase(failedTest.name(), failedTest.testMessages(), false, programmingExercise, activeTestCases)); + }); + result.setTestCaseCount(result.getTestCaseCount() + job.failedTests().size()); - for (final var successfulTest : job.getSuccessfulTests()) { + for (final var successfulTest : job.successfulTests()) { result.addFeedback( - feedbackCreationService.createFeedbackFromTestCase(successfulTest.getName(), successfulTest.getTestMessages(), true, programmingExercise, activeTestCases)); + feedbackCreationService.createFeedbackFromTestCase(successfulTest.name(), successfulTest.testMessages(), true, programmingExercise, activeTestCases)); } - result.setTestCaseCount(result.getTestCaseCount() + job.getSuccessfulTests().size()); - result.setPassedTestCaseCount(result.getPassedTestCaseCount() + job.getSuccessfulTests().size()); - } + result.setTestCaseCount(result.getTestCaseCount() + job.successfulTests().size()); + result.setPassedTestCaseCount(result.getPassedTestCaseCount() + job.successfulTests().size()); + }); } - private void addStaticCodeAnalysisFeedbackToResult(Result result, AbstractBuildResultNotificationDTO buildResult, ProgrammingExercise programmingExercise) { - final var staticCodeAnalysisReports = buildResult.getStaticCodeAnalysisReports(); + private void addStaticCodeAnalysisFeedbackToResult(Result result, BuildResultNotification buildResult, ProgrammingExercise programmingExercise) { + final var staticCodeAnalysisReports = buildResult.staticCodeAnalysisReports(); if (Boolean.TRUE.equals(programmingExercise.isStaticCodeAnalysisEnabled()) && staticCodeAnalysisReports != null && !staticCodeAnalysisReports.isEmpty()) { List scaFeedbackList = feedbackCreationService.createFeedbackFromStaticCodeAnalysisReports(staticCodeAnalysisReports); result.addFeedbacks(scaFeedbackList); @@ -105,18 +97,6 @@ private void addStaticCodeAnalysisFeedbackToResult(Result result, AbstractBuildR } } - private void addTestwiseCoverageReportToResult(Result result, AbstractBuildResultNotificationDTO buildResult, ProgrammingExercise programmingExercise) { - programmingExercise.setBuildConfig(programmingExerciseBuildConfigRepository.getProgrammingExerciseBuildConfigElseThrow(programmingExercise)); - if (Boolean.TRUE.equals(programmingExercise.getBuildConfig().isTestwiseCoverageEnabled())) { - var report = buildResult.getTestwiseCoverageReports(); - if (report != null) { - // since the test cases are not saved to the database yet, the test case is null for the entries - var coverageFileReportsWithoutTestsByTestCaseName = testwiseCoverageService.createTestwiseCoverageFileReportsWithoutTestsByTestCaseName(report); - result.setCoverageFileReportsByTestCaseName(coverageFileReportsWithoutTestsByTestCaseName); - } - } - } - /** * Find the ZonedDateTime of the first BuildLogEntry that contains the searchString in the log message. * diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/ContinuousIntegrationResultService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/ContinuousIntegrationResultService.java index b81239c44f0e..a63fc4871d4c 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/ContinuousIntegrationResultService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/ContinuousIntegrationResultService.java @@ -8,7 +8,7 @@ import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; import de.tum.cit.aet.artemis.programming.domain.ProjectType; import de.tum.cit.aet.artemis.programming.domain.build.BuildLogEntry; -import de.tum.cit.aet.artemis.programming.dto.AbstractBuildResultNotificationDTO; +import de.tum.cit.aet.artemis.programming.dto.BuildResultNotification; /** * Abstract service for managing entities related to continuous integration. @@ -21,7 +21,7 @@ public interface ContinuousIntegrationResultService { * @param requestBody the object sent from the CI system to Artemis * @return the DTO with all information in Java Object form */ - AbstractBuildResultNotificationDTO convertBuildResult(Object requestBody); + BuildResultNotification convertBuildResult(Object requestBody); /** * Generate an Artemis result object from the CI build result. Will use the test case results and issues in static code analysis as result feedback. @@ -30,7 +30,7 @@ public interface ContinuousIntegrationResultService { * @param participation to attach result to. * @return the created Artemis result with a score, completion date, etc. */ - Result createResultFromBuildResult(AbstractBuildResultNotificationDTO buildResult, ProgrammingExerciseParticipation participation); + Result createResultFromBuildResult(BuildResultNotification buildResult, ProgrammingExerciseParticipation participation); /** * Extract the build log statistics from the BuildLogEntries and persist a BuildLogStatisticsEntry. diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestCaseDTO.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestCaseDTO.java index e4e15c9227bf..5ae570f5b2d5 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestCaseDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestCaseDTO.java @@ -14,13 +14,13 @@ import com.fasterxml.jackson.annotation.Nulls; import de.tum.cit.aet.artemis.assessment.domain.Feedback; -import de.tum.cit.aet.artemis.programming.dto.TestCaseBaseDTO; +import de.tum.cit.aet.artemis.programming.dto.TestCaseBase; @JsonIgnoreProperties(ignoreUnknown = true) @JsonInclude(JsonInclude.Include.NON_EMPTY) public record TestCaseDTO(String name, String classname, double time, @JsonProperty("failures") @JsonSetter(nulls = Nulls.AS_EMPTY) List failures, @JsonProperty("errors") @JsonSetter(nulls = Nulls.AS_EMPTY) List errors, - @JsonProperty("successInfos") @JsonSetter(nulls = Nulls.AS_EMPTY) List successInfos) implements TestCaseBaseDTO { + @JsonProperty("successInfos") @JsonSetter(nulls = Nulls.AS_EMPTY) List successInfos) implements TestCaseBase { @JsonIgnore public boolean isSuccessful() { @@ -28,12 +28,7 @@ public boolean isSuccessful() { } @Override - public String getName() { - return name; - } - - @Override - public List getTestMessages() { + public List testMessages() { return extractMessage().map(Collections::singletonList).orElse(Collections.emptyList()); } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestResultsDTO.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestResultsDTO.java index cd029f5adf5e..9bb2e13c588a 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestResultsDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestResultsDTO.java @@ -4,7 +4,6 @@ import java.util.List; import java.util.stream.Stream; -import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonInclude; import com.fasterxml.jackson.annotation.JsonProperty; @@ -15,156 +14,64 @@ import de.tum.cit.aet.artemis.programming.domain.RepositoryType; import de.tum.cit.aet.artemis.programming.domain.build.BuildLogEntry; -import de.tum.cit.aet.artemis.programming.dto.AbstractBuildResultNotificationDTO; -import de.tum.cit.aet.artemis.programming.dto.BuildJobDTOInterface; +import de.tum.cit.aet.artemis.programming.dto.BuildJobInterface; +import de.tum.cit.aet.artemis.programming.dto.BuildResultNotification; import de.tum.cit.aet.artemis.programming.dto.StaticCodeAnalysisReportDTO; import de.tum.cit.aet.artemis.programming.service.ci.notification.BuildLogParseUtils; @JsonIgnoreProperties(ignoreUnknown = true) @JsonInclude(JsonInclude.Include.NON_EMPTY) // Note: due to limitations with inheritance, we cannot declare this as record, but we can use it in a similar way with final fields -public class TestResultsDTO extends AbstractBuildResultNotificationDTO { - - private final int successful; - - private final int skipped; - - private final int errors; - - private final int failures; - - private final String fullName; - - private final List commits; - - private final List results; - - private final List staticCodeAnalysisReports; - - private final List testwiseCoverageReport; - - private final ZonedDateTime runDate; - - private final boolean isBuildSuccessful; - - private final List logs; - - @JsonCreator - public TestResultsDTO(@JsonProperty("successful") int successful, @JsonProperty("skipped") int skipped, @JsonProperty("errors") int errors, - @JsonProperty("failures") int failures, @JsonProperty("fullName") String fullName, @JsonProperty("commits") @JsonSetter(nulls = Nulls.AS_EMPTY) List commits, - @JsonProperty("results") @JsonSetter(nulls = Nulls.AS_EMPTY) List results, - @JsonProperty("staticCodeAnalysisReports") @JsonSetter(nulls = Nulls.AS_EMPTY) List staticCodeAnalysisReports, - @JsonProperty("testwiseCoverageReport") @JsonSetter(nulls = Nulls.AS_EMPTY) List testwiseCoverageReport, - @JsonProperty("runDate") ZonedDateTime runDate, @JsonProperty("isBuildSuccessful") boolean isBuildSuccessful, - @JsonProperty("logs") @JsonSetter(nulls = Nulls.AS_EMPTY) List logs) { - this.successful = successful; - this.skipped = skipped; - this.errors = errors; - this.failures = failures; - this.fullName = fullName; - this.commits = commits; - this.results = results; - this.staticCodeAnalysisReports = staticCodeAnalysisReports; - this.testwiseCoverageReport = testwiseCoverageReport; - this.runDate = runDate; - this.isBuildSuccessful = isBuildSuccessful; - this.logs = logs; - } +public record TestResultsDTO(@JsonProperty("successful") int successful, @JsonProperty("skipped") int skipped, @JsonProperty("errors") int errors, + @JsonProperty("failures") int failures, @JsonProperty("fullName") String fullName, @JsonProperty("commits") @JsonSetter(nulls = Nulls.AS_EMPTY) List commits, + @JsonProperty("results") @JsonSetter(nulls = Nulls.AS_EMPTY) List results, + @JsonProperty("staticCodeAnalysisReports") @JsonSetter(nulls = Nulls.AS_EMPTY) List staticCodeAnalysisReports, + @JsonProperty("runDate") ZonedDateTime runDate, @JsonProperty("isBuildSuccessful") boolean isBuildSuccessful, + @JsonProperty("logs") @JsonSetter(nulls = Nulls.AS_EMPTY) List logs) implements BuildResultNotification { public static TestResultsDTO convert(Object someResult) { return new ObjectMapper().registerModule(new JavaTimeModule()).convertValue(someResult, TestResultsDTO.class); } - public int getSuccessful() { - return successful; - } - - public int getSkipped() { - return skipped; - } - - public int getErrors() { - return errors; - } - - public int getFailures() { - return failures; - } - - public String getFullName() { - return fullName; - } - - public ZonedDateTime getRunDate() { - return runDate; - } - - public List getLogs() { - return this.logs; - } - @Override - public ZonedDateTime getBuildRunDate() { - return getRunDate(); + public ZonedDateTime buildRunDate() { + return runDate(); } @Override - protected String getCommitHashFromAssignmentRepo() { + public String assignmentRepoCommitHash() { final var testRepoNameSuffix = RepositoryType.TESTS.getName(); - final var firstCommit = getCommits().stream().filter(commit -> !commit.repositorySlug().endsWith(testRepoNameSuffix)).findFirst(); + final var firstCommit = commits().stream().filter(commit -> !commit.repositorySlug().endsWith(testRepoNameSuffix)).findFirst(); return firstCommit.map(CommitDTO::hash).orElse(null); } @Override - protected String getCommitHashFromTestsRepo() { + public String testsRepoCommitHash() { final var testRepoNameSuffix = RepositoryType.TESTS.getName(); - final var firstCommit = getCommits().stream().filter(commit -> commit.repositorySlug().endsWith(testRepoNameSuffix)).findFirst(); + final var firstCommit = commits().stream().filter(commit -> commit.repositorySlug().endsWith(testRepoNameSuffix)).findFirst(); return firstCommit.map(CommitDTO::hash).orElse(null); } @Override - public String getBranchNameFromAssignmentRepo() { + public String assignmentRepoBranchName() { final var testRepoNameSuffix = RepositoryType.TESTS.getName(); - final var firstCommit = getCommits().stream().filter(commit -> !commit.repositorySlug().endsWith(testRepoNameSuffix)).findFirst(); + final var firstCommit = commits().stream().filter(commit -> !commit.repositorySlug().endsWith(testRepoNameSuffix)).findFirst(); return firstCommit.map(CommitDTO::branchName).orElse(null); } private int getSum() { - return getSkipped() + getFailures() + getErrors() + getSuccessful(); - } - - @Override - public boolean isBuildSuccessful() { - return isBuildSuccessful; + return skipped() + failures() + errors() + successful(); } @Override - public Double getBuildScore() { + public Double buildScore() { final var testSum = getSum(); - return testSum == 0 ? 0D : ((double) getSuccessful() / testSum) * 100D; - } - - public List getCommits() { - return commits; - } - - public List getResults() { - return results; - } - - @Override - public List getStaticCodeAnalysisReports() { - return staticCodeAnalysisReports; - } - - @Override - public List getTestwiseCoverageReports() { - return testwiseCoverageReport; + return testSum == 0 ? 0D : ((double) successful() / testSum) * 100D; } @Override public boolean hasArtifact() { - // TODO: this is not available in Jenkins or GitLab CI yet + // NOTE: this is not available in Jenkins return false; } @@ -175,13 +82,13 @@ public boolean hasLogs() { @Override public List extractBuildLogs() { - var buildLogs = BuildLogParseUtils.parseBuildLogsFromLogs(getLogs()); + var buildLogs = BuildLogParseUtils.parseBuildLogsFromLogs(logs()); return filterBuildLogs(buildLogs); } @Override - public List getBuildJobs() { - return getResults(); + public List jobs() { + return results(); } /** diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestSuiteDTO.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestSuiteDTO.java index 9be20c1a8ad9..ac3702222c00 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestSuiteDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestSuiteDTO.java @@ -9,23 +9,23 @@ import com.fasterxml.jackson.annotation.JsonSetter; import com.fasterxml.jackson.annotation.Nulls; -import de.tum.cit.aet.artemis.programming.dto.BuildJobDTOInterface; -import de.tum.cit.aet.artemis.programming.dto.TestCaseBaseDTO; +import de.tum.cit.aet.artemis.programming.dto.BuildJobInterface; +import de.tum.cit.aet.artemis.programming.dto.TestCaseBase; @JsonIgnoreProperties(ignoreUnknown = true) @JsonInclude(JsonInclude.Include.NON_EMPTY) public record TestSuiteDTO(String name, double time, int errors, int skipped, int failures, int tests, - @JsonProperty("testCases") @JsonSetter(nulls = Nulls.AS_EMPTY) List testCases) implements BuildJobDTOInterface { + @JsonProperty("testCases") @JsonSetter(nulls = Nulls.AS_EMPTY) List testCases) implements BuildJobInterface { @Override @JsonIgnore - public List getFailedTests() { + public List failedTests() { return testCases.stream().filter(testCase -> !testCase.isSuccessful()).toList(); } @Override @JsonIgnore - public List getSuccessfulTests() { + public List successfulTests() { return testCases.stream().filter(TestCaseDTO::isSuccessful).toList(); } } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestwiseCoverageReportDTO.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestwiseCoverageReportDTO.java deleted file mode 100644 index ef94ea89542f..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ci/notification/dto/TestwiseCoverageReportDTO.java +++ /dev/null @@ -1,115 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.ci.notification.dto; - -import java.io.Serializable; -import java.util.ArrayList; -import java.util.List; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonProperty; - -// TODO: convert to record -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(JsonInclude.Include.NON_EMPTY) -public class TestwiseCoverageReportDTO implements Serializable { - - @JsonProperty("uniformPath") - private String uniformPath; - - @JsonProperty("duration") - private double duration; - - @JsonProperty("content") - private String content; - - @JsonProperty("paths") - private List coveredPathsPerTestDTOs = new ArrayList<>(); - - public String getUniformPath() { - return uniformPath; - } - - public void setUniformPath(String uniformPath) { - this.uniformPath = uniformPath; - } - - public double getDuration() { - return duration; - } - - public void setDuration(double duration) { - this.duration = duration; - } - - public String getContent() { - return content; - } - - public void setContent(String content) { - this.content = content; - } - - public List getCoveredPathsPerTestDTOs() { - return coveredPathsPerTestDTOs; - } - - public void setCoveredPathsPerTestDTOs(List coveredPathsPerTestDTOs) { - this.coveredPathsPerTestDTOs = coveredPathsPerTestDTOs; - } - - // TODO: convert to record - @JsonIgnoreProperties(ignoreUnknown = true) - @JsonInclude(JsonInclude.Include.NON_EMPTY) - public static final class CoveredPathsPerTestDTO implements Serializable { - - @JsonProperty("path") - private String path; - - @JsonProperty("files") - private List coveredFilesPerTestDTOs = new ArrayList<>(); - - public String getPath() { - return path; - } - - public void setPath(String path) { - this.path = path; - } - - public List getCoveredFilesPerTestDTOs() { - return coveredFilesPerTestDTOs; - } - - public void setCoveredFilesPerTestDTOs(List coveredFilesPerTestDTOs) { - this.coveredFilesPerTestDTOs = coveredFilesPerTestDTOs; - } - } - - // TODO: convert to record - @JsonIgnoreProperties(ignoreUnknown = true) - @JsonInclude(JsonInclude.Include.NON_EMPTY) - public static final class CoveredFilesPerTestDTO implements Serializable { - - @JsonProperty("fileName") - private String fileName; - - @JsonProperty("coveredLines") - private String coveredLinesWithRanges; - - public String getFileName() { - return fileName; - } - - public void setFileName(String fileName) { - this.fileName = fileName; - } - - public String getCoveredLinesWithRanges() { - return coveredLinesWithRanges; - } - - public void setCoveredLinesWithRanges(String coveredLinesWithRanges) { - this.coveredLinesWithRanges = coveredLinesWithRanges; - } - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIProgrammingLanguageFeatureService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIProgrammingLanguageFeatureService.java index 0c71114e13bb..357cf05d97f8 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIProgrammingLanguageFeatureService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIProgrammingLanguageFeatureService.java @@ -23,9 +23,9 @@ public class GitLabCIProgrammingLanguageFeatureService extends ProgrammingLanguageFeatureService { public GitLabCIProgrammingLanguageFeatureService() { - programmingLanguageFeatures.put(EMPTY, new ProgrammingLanguageFeature(EMPTY, false, false, false, false, false, List.of(), false, false)); - programmingLanguageFeatures.put(JAVA, new ProgrammingLanguageFeature(JAVA, false, false, false, true, false, List.of(PLAIN_MAVEN, MAVEN_MAVEN), false, false)); - programmingLanguageFeatures.put(JAVASCRIPT, new ProgrammingLanguageFeature(JAVASCRIPT, false, false, true, false, false, List.of(), false, false)); - programmingLanguageFeatures.put(RUST, new ProgrammingLanguageFeature(RUST, false, false, true, false, false, List.of(), false, false)); + programmingLanguageFeatures.put(EMPTY, new ProgrammingLanguageFeature(EMPTY, false, false, false, false, false, List.of(), false)); + programmingLanguageFeatures.put(JAVA, new ProgrammingLanguageFeature(JAVA, false, false, false, true, false, List.of(PLAIN_MAVEN, MAVEN_MAVEN), false)); + programmingLanguageFeatures.put(JAVASCRIPT, new ProgrammingLanguageFeature(JAVASCRIPT, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(RUST, new ProgrammingLanguageFeature(RUST, false, false, true, false, false, List.of(), false)); } } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIResultService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIResultService.java index fa63482e88ca..0c6475c1e236 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIResultService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIResultService.java @@ -8,22 +8,18 @@ import org.springframework.context.annotation.Profile; import org.springframework.stereotype.Service; -import de.tum.cit.aet.artemis.assessment.repository.FeedbackRepository; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; import de.tum.cit.aet.artemis.programming.domain.ProjectType; import de.tum.cit.aet.artemis.programming.domain.build.BuildLogEntry; import de.tum.cit.aet.artemis.programming.domain.build.BuildLogStatisticsEntry; -import de.tum.cit.aet.artemis.programming.dto.AbstractBuildResultNotificationDTO; +import de.tum.cit.aet.artemis.programming.dto.BuildResultNotification; import de.tum.cit.aet.artemis.programming.repository.BuildLogStatisticsEntryRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseBuildConfigRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; -import de.tum.cit.aet.artemis.programming.repository.ProgrammingSubmissionRepository; -import de.tum.cit.aet.artemis.programming.service.BuildLogEntryService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseFeedbackCreationService; import de.tum.cit.aet.artemis.programming.service.ci.AbstractContinuousIntegrationResultService; import de.tum.cit.aet.artemis.programming.service.ci.notification.dto.TestResultsDTO; -import de.tum.cit.aet.artemis.programming.service.hestia.TestwiseCoverageService; // Gitlab support will be removed in 8.0.0. Please migrate to LocalVC using e.g. the PR https://github.com/ls1intum/Artemis/pull/8972 @Deprecated(since = "7.5.0", forRemoval = true) @@ -34,15 +30,13 @@ public class GitLabCIResultService extends AbstractContinuousIntegrationResultSe private static final Logger log = LoggerFactory.getLogger(GitLabCIResultService.class); - public GitLabCIResultService(ProgrammingSubmissionRepository programmingSubmissionRepository, FeedbackRepository feedbackRepository, BuildLogEntryService buildLogService, - BuildLogStatisticsEntryRepository buildLogStatisticsEntryRepository, TestwiseCoverageService testwiseCoverageService, - ProgrammingExerciseFeedbackCreationService feedbackCreationService, ProgrammingExerciseTestCaseRepository testCaseRepository, - ProgrammingExerciseBuildConfigRepository programmingExerciseBuildConfigRepository) { - super(testCaseRepository, buildLogStatisticsEntryRepository, testwiseCoverageService, feedbackCreationService, programmingExerciseBuildConfigRepository); + public GitLabCIResultService(BuildLogStatisticsEntryRepository buildLogStatisticsEntryRepository, ProgrammingExerciseFeedbackCreationService feedbackCreationService, + ProgrammingExerciseTestCaseRepository testCaseRepository, ProgrammingExerciseBuildConfigRepository programmingExerciseBuildConfigRepository) { + super(testCaseRepository, buildLogStatisticsEntryRepository, feedbackCreationService, programmingExerciseBuildConfigRepository); } @Override - public AbstractBuildResultNotificationDTO convertBuildResult(Object requestBody) { + public BuildResultNotification convertBuildResult(Object requestBody) { return TestResultsDTO.convert(requestBody); } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIService.java index e0ec98ed52cf..7226e40b16db 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/gitlabci/GitLabCIService.java @@ -314,7 +314,7 @@ public void deleteBuildPlan(String projectKey, String buildPlanId) { @Override public String getPlanKey(Object requestBody) throws ContinuousIntegrationException { TestResultsDTO dto = TestResultsDTO.convert(requestBody); - return dto.getFullName(); + return dto.fullName(); } @Override diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/CodeHintService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/CodeHintService.java deleted file mode 100644 index f04de8a56db9..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/CodeHintService.java +++ /dev/null @@ -1,187 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.util.HashSet; -import java.util.List; -import java.util.Optional; -import java.util.stream.Collectors; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Profile; -import org.springframework.stereotype.Service; - -import de.tum.cit.aet.artemis.core.exception.BadRequestAlertException; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; -import de.tum.cit.aet.artemis.programming.repository.hestia.CodeHintRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseSolutionEntryRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseTaskRepository; - -@Profile(PROFILE_CORE) -@Service -public class CodeHintService { - - private static final Logger log = LoggerFactory.getLogger(CodeHintService.class); - - private final CodeHintRepository codeHintRepository; - - private final ProgrammingExerciseTaskRepository taskRepository; - - private final ProgrammingExerciseSolutionEntryRepository solutionEntryRepository; - - public CodeHintService(CodeHintRepository codeHintRepository, ProgrammingExerciseTaskRepository taskRepository, - ProgrammingExerciseSolutionEntryRepository solutionEntryRepository) { - this.codeHintRepository = codeHintRepository; - this.taskRepository = taskRepository; - this.solutionEntryRepository = solutionEntryRepository; - } - - /** - * Generate {@link CodeHint}s for all {@link ProgrammingExerciseTask}s of an exercise. - * If requested old code hints will be deleted otherwise the new ones will be added to the existing ones. (This might however break the old ones) - * If a task does not have any test cases with solution entries it will not get a code hint. - * - * @param exercise The programming exercise - * @param deleteOldCodeHints Whether old code hint should be deleted - * @return The list of all newly generated code hints - */ - public List generateCodeHintsForExercise(ProgrammingExercise exercise, boolean deleteOldCodeHints) { - log.info("Generating code hints for exercise {} with deleteOldCodeHints={}", exercise.getId(), deleteOldCodeHints); - - var tasks = taskRepository.findByExerciseIdWithTestCaseAndSolutionEntriesElseThrow(exercise.getId()); - - return tasks.stream().map(task -> generateCodeHintForTask(task, deleteOldCodeHints)).filter(Optional::isPresent).map(Optional::get).toList(); - } - - /** - * Generate a single {@link CodeHint} for a single {@link ProgrammingExerciseTask} - * If requested old code hints will be deleted otherwise the new one will be added to the existing ones. (This might however break the old ones) - * If the task does not have any test cases with solution entries it will not get a code hint. - * - * @param task The programming exercise task - * @param deleteOldCodeHints Whether old code hint should be deleted - * @return The newly created code hint if one was needed - */ - public Optional generateCodeHintForTask(ProgrammingExerciseTask task, boolean deleteOldCodeHints) { - log.info("Generating code hints for task {} ({}) in exercise {} with deleteOldCodeHints={}", task.getId(), task.getTaskName(), task.getExercise().getId(), - deleteOldCodeHints); - - var codeHint = new CodeHint(); - codeHint.setExercise(task.getExercise()); - codeHint.setProgrammingExerciseTask(task); - codeHint.setTitle("Code hint for task " + task.getTaskName()); - - var solutionEntries = task.getTestCases().stream().flatMap(testCase -> testCase.getSolutionEntries().stream()).peek(solutionEntry -> solutionEntry.setCodeHint(codeHint)) - .collect(Collectors.toSet()); - - if (deleteOldCodeHints) { - deleteCodeHintsForTask(task); - } - if (solutionEntries.isEmpty()) { - return Optional.empty(); - } - - codeHint.setSolutionEntries(solutionEntries); - codeHintRepository.save(codeHint); - solutionEntryRepository.saveAll(solutionEntries); - - return Optional.of(codeHint); - } - - /** - * Deletes all code hints of a {@link ProgrammingExerciseTask} - * - * @param task The programming exercise task - */ - public void deleteCodeHintsForTask(ProgrammingExerciseTask task) { - log.info("Deleting all code hints of task {} ({}) in exercise {}", task.getId(), task.getTaskName(), task.getExercise().getId()); - - var codeHints = codeHintRepository.findByTaskIdWithSolutionEntries(task.getId()); - var solutionEntries = codeHints.stream().flatMap(codeHint -> codeHint.getSolutionEntries().stream()).peek(solutionEntry -> solutionEntry.setCodeHint(null)).toList(); - solutionEntryRepository.saveAll(solutionEntries); - codeHintRepository.deleteAll(codeHints); - } - - /** - * Deletes a single code hint. - * Sets the code hint of all related solution entries to null before deleting. - * - * @param codeHint The code hint to be deleted - */ - public void deleteCodeHint(CodeHint codeHint) { - log.info("Deleting code hint {}", codeHint.getId()); - - var solutionEntries = solutionEntryRepository.findByCodeHintId(codeHint.getId()); - for (ProgrammingExerciseSolutionEntry solutionEntry : solutionEntries) { - solutionEntry.setCodeHint(null); - } - solutionEntryRepository.saveAll(solutionEntries); - codeHintRepository.delete(codeHint); - } - - /** - * Persists the updated solution entries for a code hint. The solution entries are loaded from the database and can result in three scenarios: - * 1. The code or test case for an existing entry is updated. - * 2. A new solution entry is created. The test case of the entry must be contained in hint related task. - * 3. An entry is removed from the code hint. This method does not delete the entry itself, but removes the reference to the corresponding hint. - * - * @param hint the code hint containing the solution entries to be updated - */ - public void updateSolutionEntriesForCodeHint(CodeHint hint) { - var optionalTask = taskRepository.findByCodeHintIdWithTestCases(hint.getId()); - - if (optionalTask.isEmpty()) { - throw new BadRequestAlertException("No task has been assigned to the code hint.", "CodeHint", "codeHint"); - } - - var task = optionalTask.get(); - - var savedSolutionEntries = solutionEntryRepository.findByCodeHintId(hint.getId()); - - var newEntries = hint.getSolutionEntries().stream().filter(entry -> savedSolutionEntries.stream().noneMatch(savedEntry -> savedEntry.getId().equals(entry.getId()))) - .peek(entry -> entry.setCodeHint(hint)).toList(); - - // check that the task of the hint contains all test cases of every solution entry to be saved - // we only assume changes on the test case of an entry, if the value is defined - boolean hasUnrelatedTestCaseEntries = newEntries.stream() - .anyMatch(entry -> task.getTestCases().stream().noneMatch(containedTestCase -> containedTestCase.getId().equals(entry.getTestCase().getId()))); - if (hasUnrelatedTestCaseEntries) { - throw new BadRequestAlertException("There is at least one solution entry that references a test case that does not belong to the task of the code hint.", "Code Hint", - "codeHint"); - } - - var result = new HashSet<>(newEntries); - - var updatedEntries = new HashSet<>(hint.getSolutionEntries()); - newEntries.forEach(updatedEntries::remove); - - var removedEntries = savedSolutionEntries.stream() - .filter(savedEntry -> hint.getSolutionEntries().stream().noneMatch(updatedEntry -> savedEntry.getId().equals(updatedEntry.getId()))) - .peek(entryToRemove -> entryToRemove.setCodeHint(null)).toList(); - - updatedEntries.forEach(updatedEntry -> { - var optionalMatch = savedSolutionEntries.stream().filter(savedEntry -> savedEntry.getId().equals(updatedEntry.getId())).findFirst(); - if (optionalMatch.isPresent()) { - var match = optionalMatch.get(); - match.setLine(updatedEntry.getLine()); - match.setPreviousLine(updatedEntry.getPreviousLine()); - match.setFilePath(updatedEntry.getFilePath()); - match.setCode(updatedEntry.getCode()); - match.setPreviousCode(updatedEntry.getPreviousCode()); - - // update test case if defined - match.setTestCase(updatedEntry.getTestCase() != null ? updatedEntry.getTestCase() : match.getTestCase()); - result.add(match); - } - }); - hint.setSolutionEntries(result); - solutionEntryRepository.saveAll(result); - solutionEntryRepository.saveAll(removedEntries); - - codeHintRepository.save(hint); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/ExerciseHintService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/ExerciseHintService.java deleted file mode 100644 index 477bcaf88749..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/ExerciseHintService.java +++ /dev/null @@ -1,326 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.time.ZonedDateTime; -import java.util.ArrayList; -import java.util.Comparator; -import java.util.HashMap; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Objects; -import java.util.Optional; -import java.util.Set; -import java.util.stream.Collectors; - -import org.springframework.cache.annotation.Cacheable; -import org.springframework.context.annotation.Profile; -import org.springframework.stereotype.Service; - -import de.tum.cit.aet.artemis.assessment.domain.Feedback; -import de.tum.cit.aet.artemis.core.domain.User; -import de.tum.cit.aet.artemis.core.exception.BadRequestAlertException; -import de.tum.cit.aet.artemis.core.exception.ConflictException; -import de.tum.cit.aet.artemis.core.security.Role; -import de.tum.cit.aet.artemis.core.service.AuthorizationCheckService; -import de.tum.cit.aet.artemis.exercise.domain.Submission; -import de.tum.cit.aet.artemis.exercise.repository.StudentParticipationRepository; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHintActivation; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; -import de.tum.cit.aet.artemis.programming.repository.hestia.ExerciseHintActivationRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ExerciseHintRepository; - -@Profile(PROFILE_CORE) -@Service -public class ExerciseHintService { - - private final AuthorizationCheckService authCheckService; - - private final ExerciseHintRepository exerciseHintRepository; - - private final ProgrammingExerciseTaskService programmingExerciseTaskService; - - private final StudentParticipationRepository studentParticipationRepository; - - private final ExerciseHintActivationRepository exerciseHintActivationRepository; - - public ExerciseHintService(AuthorizationCheckService authCheckService, ExerciseHintRepository exerciseHintRepository, - ProgrammingExerciseTaskService programmingExerciseTaskService, StudentParticipationRepository studentParticipationRepository, - ExerciseHintActivationRepository exerciseHintActivationRepository) { - this.authCheckService = authCheckService; - this.exerciseHintRepository = exerciseHintRepository; - this.programmingExerciseTaskService = programmingExerciseTaskService; - this.studentParticipationRepository = studentParticipationRepository; - this.exerciseHintActivationRepository = exerciseHintActivationRepository; - } - - /** - * Copies the hints of an exercise to a new target exercise by cloning the hint objects and saving them - * resulting in new IDs for the copied hints. The contents stay the same. On top of that, all hints in the - * problem statement of the target exercise get replaced by the new IDs. - * - * @param template The template exercise containing the hints that should be copied - * @param target The new target exercise, to which all hints should get copied to. - * @return A map with the old hint id as a key and the new hint id as a value - */ - public Map copyExerciseHints(final ProgrammingExercise template, final ProgrammingExercise target) { - final Map hintIdMapping = new HashMap<>(); - target.setExerciseHints(template.getExerciseHints().stream().map(hint -> { - ExerciseHint copiedHint = hint.createCopy(); - copiedHint.setExercise(target); - exerciseHintRepository.save(copiedHint); - hintIdMapping.put(hint.getId(), copiedHint.getId()); - return copiedHint; - }).collect(Collectors.toSet())); - - return hintIdMapping; - } - - /** - * Sets the rating of an exercise hint for a user - * The rating is saved in the associated ExerciseHintActivation. - * - * @param exerciseHint The exercise hint to rate - * @param user The user that submits the rating - * @param ratingValue The value of the rating - */ - public void rateExerciseHint(ExerciseHint exerciseHint, User user, Integer ratingValue) { - if (ratingValue < 1 || ratingValue > 5) { - throw new BadRequestAlertException("rating has to be between 1 and 5", "exerciseHint", "ratingValue.invalid", false); - } - var exerciseHintActivation = exerciseHintActivationRepository.findByExerciseHintAndUserElseThrow(exerciseHint.getId(), user.getId()); - exerciseHintActivation.setRating(ratingValue); - exerciseHintActivationRepository.save(exerciseHintActivation); - } - - /** - * Activates an ExerciseHint for a user. - * After activation the user can view the full content of the hint without restrictions. - * This action cannot be undone - * - * @param exerciseHint The exercise hint - * @param user The user - * @return true if the hint was activated - */ - public boolean activateHint(ExerciseHint exerciseHint, User user) { - // Check if the user has access to the exercise - // This is done here to prevent illegal activation of hints in all possible future cases - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.STUDENT, exerciseHint.getExercise(), user); - - // Check if the user can activate the hint - if (!getAvailableExerciseHints(exerciseHint.getExercise(), user).contains(exerciseHint)) { - return false; - } - - // Check if the hint was already activated - if (exerciseHintActivationRepository.findByExerciseHintAndUser(exerciseHint.getId(), user.getId()).isPresent()) { - return false; - } - - var exerciseHintActivation = new ExerciseHintActivation(); - exerciseHintActivation.setExerciseHint(exerciseHint); - exerciseHintActivation.setUser(user); - exerciseHintActivation.setActivationDate(ZonedDateTime.now()); - exerciseHintActivationRepository.save(exerciseHintActivation); - return true; - } - - /** - * Retrieves all exercise hints that a user has activated for an exercise. - * - * @param exercise The programming exercise - * @param user The user - * @return All activated exercise hints for this user and exercise - */ - public Set getActivatedExerciseHints(ProgrammingExercise exercise, User user) { - return exerciseHintActivationRepository.findByExerciseAndUserWithExerciseHintRelations(exercise.getId(), user.getId()).stream().map(exerciseHintActivation -> { - var exerciseHint = exerciseHintActivation.getExerciseHint(); - exerciseHint.setCurrentUserRating(exerciseHintActivation.getRating()); - return exerciseHint; - }).collect(Collectors.toSet()); - } - - /** - * Returns all exercise hints that the user can currently activate for a given programming exercise. - * Exercise hints will be shown for the first task that meets the following conditions: - * (1) the subsequent number of the latest submissions the previous task is successful is greater or equal to the hint's threshold - * (2) the subsequent number of the latest submissions the current task is unsuccessful is greater or equal to the hint's threshold - * If no task matches these conditions, no exercise hints will be returned - * Note: A task is successful, if the feedback within the submission is positive for all associated test cases within this task - * - * @param exercise The programming exercise - * @param user The user - * @return All available exercise hints - */ - public Set getAvailableExerciseHints(ProgrammingExercise exercise, User user) { - var exerciseHints = exerciseHintRepository.findByExerciseId(exercise.getId()); - if (exerciseHints.isEmpty()) { - return new HashSet<>(); - } - - var submissions = getSubmissionsForStudent(exercise, user); - - if (submissions.isEmpty()) { - return new HashSet<>(); - } - - var latestResult = submissions.getFirst().getLatestResult(); - - // latest submissions has no result or latest result has no feedback (most commonly due to a build error) - if (latestResult == null || latestResult.getFeedbacks().isEmpty()) { - return new HashSet<>(); - } - var tasks = programmingExerciseTaskService.getSortedTasks(exercise); - - var subsequentNumberOfUnsuccessfulSubmissionsByTask = tasks.stream() - .collect(Collectors.toMap(task -> task, task -> subsequentNumberOfSubmissionsForTaskWithStatus(submissions, task, false))); - var subsequentNumberOfSuccessfulSubmissionsByTask = tasks.stream() - .collect(Collectors.toMap(task -> task, task -> subsequentNumberOfSubmissionsForTaskWithStatus(submissions, task, true))); - - var availableHints = new HashSet(); - - for (int i = 0; i < tasks.size(); i++) { - var task = tasks.get(i); - int subsequentNumberOfUnsuccessfulSubmissionsForCurrentTask = subsequentNumberOfUnsuccessfulSubmissionsByTask.get(task); - // current task is successful - if (subsequentNumberOfUnsuccessfulSubmissionsForCurrentTask == 0) { - continue; - } - - var hintsInTask = exerciseHints.stream() - .filter(hint -> hint.getProgrammingExerciseTask() != null && Objects.equals(hint.getProgrammingExerciseTask().getId(), task.getId())) - .collect(Collectors.toSet()); - // no hints exist for the current task - if (hintsInTask.isEmpty()) { - continue; - } - - Optional subsequentNumberSuccessfulSubmissionsForPreviousTask; - if (i == 0) { - subsequentNumberSuccessfulSubmissionsForPreviousTask = Optional.empty(); - } - else { - subsequentNumberSuccessfulSubmissionsForPreviousTask = Optional.of(subsequentNumberOfSuccessfulSubmissionsByTask.get(tasks.get(i - 1))); - } - - // skip current task if the previous task was not successful - if (0 == subsequentNumberSuccessfulSubmissionsForPreviousTask.orElse(-1)) { - continue; - } - - // add the available hints for the current task - var availableHintsForCurrentTask = getAvailableExerciseHintsForTask(subsequentNumberSuccessfulSubmissionsForPreviousTask, - subsequentNumberOfUnsuccessfulSubmissionsForCurrentTask, hintsInTask); - if (!availableHintsForCurrentTask.isEmpty()) { - availableHints.addAll(availableHintsForCurrentTask); - break; - } - } - // Hints with a threshold of 0 will always be displayed - availableHints.addAll(exerciseHints.stream().filter(hint -> hint.getDisplayThreshold() == 0).toList()); - return availableHints; - } - - private boolean isTaskSuccessfulInSubmission(ProgrammingExerciseTask task, Submission submission) { - var result = submission.getLatestResult(); - if (result == null || result.getFeedbacks().isEmpty()) { - return false; - } - var testCasesInTask = task.getTestCases(); - var feedbacks = result.getFeedbacks(); - // a task is successful if feedback for all in the task linked tests exist and if these tests all passed. - var feedbackForTask = testCasesInTask.stream().map(testCase -> findFeedbackForTestCase(testCase, feedbacks)).filter(Optional::isPresent).map(Optional::get).toList(); - if (feedbackForTask.size() != testCasesInTask.size()) { - // some expected test cases were not executed in the student's result - return false; - } - return feedbackForTask.stream().allMatch(feedback -> Boolean.TRUE.equals(feedback.isPositive())); - } - - /** - * Finds the first feedback in the provided list, that was created by the test case. - * - * @param testCase the test case to search the linked feedback for - * @return the feedback related by the test case, if present - */ - private Optional findFeedbackForTestCase(ProgrammingExerciseTestCase testCase, List feedbacks) { - return feedbacks.stream().filter(feedback -> testCase.equals(feedback.getTestCase())).findAny(); - } - - private List getSubmissionsForStudent(ProgrammingExercise exercise, User student) { - List submissions = new ArrayList<>(); - - var ratedStudentParticipation = studentParticipationRepository.findByExerciseIdAndStudentIdAndTestRunWithEagerSubmissionsResultsFeedbacksTestCases(exercise.getId(), - student.getId(), false); - ratedStudentParticipation.ifPresent(participation -> submissions.addAll(participation.getSubmissions())); - - var practiceStudentParticipation = studentParticipationRepository.findByExerciseIdAndStudentIdAndTestRunWithEagerSubmissionsResultsFeedbacksTestCases(exercise.getId(), - student.getId(), true); - practiceStudentParticipation.ifPresent(participation -> submissions.addAll(participation.getSubmissions())); - - submissions.sort(Comparator.reverseOrder()); - - return submissions; - } - - private int subsequentNumberOfSubmissionsForTaskWithStatus(List submissions, ProgrammingExerciseTask task, boolean successful) { - int subsequentNumberSuccessfulSubmissionsForTask = 0; - for (Submission submission : submissions) { - if (isTaskSuccessfulInSubmission(task, submission) != successful) { - break; - } - subsequentNumberSuccessfulSubmissionsForTask++; - } - return subsequentNumberSuccessfulSubmissionsForTask; - } - - /** - * Filter hints that meet the following conditions: - * 1. the previous task (if existing) is successful for at least the hint's threshold value - * 2. the current task is unsuccessful for at least the hint's threshold value - * - * @param subsequentNumberSuccessfulSubmissionsForPreviousTask the subsequent number of the latest submissions the previous task is successful - * @param subsequentNumberOfUnsuccessfulSubmissionsForCurrentTask the subsequent number of the latest submissions the current task is unsuccessful - * @param taskHints all exercise hints in current tasks - * @return the available exercise hints - */ - private Set getAvailableExerciseHintsForTask(Optional subsequentNumberSuccessfulSubmissionsForPreviousTask, - int subsequentNumberOfUnsuccessfulSubmissionsForCurrentTask, Set taskHints) { - Set availableHintsForTask = new HashSet<>(); - for (ExerciseHint hint : taskHints) { - // condition 1 - if (subsequentNumberSuccessfulSubmissionsForPreviousTask.isPresent() && subsequentNumberSuccessfulSubmissionsForPreviousTask.get() < hint.getDisplayThreshold()) { - continue; - } - - // condition 2 - if (subsequentNumberOfUnsuccessfulSubmissionsForCurrentTask >= hint.getDisplayThreshold()) { - availableHintsForTask.add(hint); - } - } - return availableHintsForTask; - } - - /** - * Returns the title of the hint identified by the given hint id if the exercise id stored in the hint matches the - * provided exercise id. - * - * @param exerciseId the exercise id that must match the one stored in the hint - * @param exerciseHintId the id of the hint - * @return the title of the hint if it was found; null otherwise - * @throws ConflictException if the provided exercise id does not match the one stored in the hint - */ - @Cacheable(cacheNames = "exerciseHintTitle", key = "''.concat(#exerciseId).concat('-').concat(#exerciseHintId)", unless = "#result == null") - public String getExerciseHintTitle(Long exerciseId, Long exerciseHintId) { - final var hint = exerciseHintRepository.findByIdElseThrow(exerciseHintId); - if (hint.getExercise() == null || !hint.getExercise().getId().equals(exerciseId)) { - throw new ConflictException("An exercise hint can only be retrieved if the exerciseIds match.", "exerciseHint", "exerciseIdsMismatch"); - } - - return hint.getTitle(); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/TestwiseCoverageService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/TestwiseCoverageService.java deleted file mode 100644 index 9fd6b6ca59f8..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/TestwiseCoverageService.java +++ /dev/null @@ -1,314 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Optional; -import java.util.Set; -import java.util.stream.Collectors; -import java.util.stream.IntStream; - -import org.eclipse.jgit.api.errors.GitAPIException; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Profile; -import org.springframework.data.domain.Pageable; -import org.springframework.stereotype.Service; - -import de.tum.cit.aet.artemis.core.exception.InternalServerErrorException; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; -import de.tum.cit.aet.artemis.programming.domain.SolutionProgrammingExerciseParticipation; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageFileReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; -import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CoverageFileReportRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CoverageReportRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.TestwiseCoverageReportEntryRepository; -import de.tum.cit.aet.artemis.programming.service.GitService; -import de.tum.cit.aet.artemis.programming.service.RepositoryService; -import de.tum.cit.aet.artemis.programming.service.ci.notification.dto.TestwiseCoverageReportDTO; - -/** - * Service for managing testwise coverage data and interacts with both CoverageReport, CoverageFileReport - * and TestwiseCoverageReportEntry - */ -@Profile(PROFILE_CORE) -@Service -public class TestwiseCoverageService { - - private static final Logger log = LoggerFactory.getLogger(TestwiseCoverageService.class); - - private final CoverageReportRepository coverageReportRepository; - - private final CoverageFileReportRepository coverageFileReportRepository; - - private final TestwiseCoverageReportEntryRepository testwiseCoverageReportEntryRepository; - - private final ProgrammingExerciseTestCaseRepository programmingExerciseTestCaseRepository; - - private final GitService gitService; - - private final RepositoryService repositoryService; - - public TestwiseCoverageService(CoverageReportRepository coverageReportRepository, CoverageFileReportRepository coverageFileReportRepository, - TestwiseCoverageReportEntryRepository testwiseCoverageReportEntryRepository, ProgrammingExerciseTestCaseRepository programmingExerciseTestCaseRepository, - RepositoryService repositoryService, GitService gitService) { - this.coverageReportRepository = coverageReportRepository; - this.coverageFileReportRepository = coverageFileReportRepository; - this.testwiseCoverageReportEntryRepository = testwiseCoverageReportEntryRepository; - this.programmingExerciseTestCaseRepository = programmingExerciseTestCaseRepository; - this.gitService = gitService; - this.repositoryService = repositoryService; - } - - /** - * Transforms the testwise coverage report DTOs to CoverageFileReports (without the test case attribute) mapped by the test case name. - * This method maps the file reports to primitive test case names because the test case is not present in the database - * on creating the entities from the DTOs. - * - * @param coverageReports the coverage report DTOs - * @return coverage file reports mapped by the test case name - */ - public Map> createTestwiseCoverageFileReportsWithoutTestsByTestCaseName(List coverageReports) { - Map> fileReportsByTestName = new HashMap<>(); - coverageReports.forEach(coveragePerTestDTO -> { - - // file reports for the current test case - Set testCoverageReports = new HashSet<>(); - for (var pathDTO : coveragePerTestDTO.getCoveredPathsPerTestDTOs()) { - - // the file reports for the current test case and path - Set fileCoverageReports = new HashSet<>(); - for (var fileDTO : pathDTO.getCoveredFilesPerTestDTOs()) { - var coverageEntriesPerFile = Arrays.stream(fileDTO.getCoveredLinesWithRanges().split(",")).map(optionalLineRange -> { - // The test case is not set for the report because it has not been saved yet to the database - var entry = new TestwiseCoverageReportEntry(); - // retrieve consecutive blocks from the ranged covered lines number - // Example: "2,3-6,7,9-30" - if (optionalLineRange.contains("-")) { - String[] range = optionalLineRange.split("-"); - int startLineNumber = Integer.parseInt(range[0]); - entry.setStartLine(startLineNumber); - entry.setLineCount(Integer.parseInt(range[1]) - startLineNumber + 1); - } - else { - entry.setStartLine(Integer.parseInt(optionalLineRange)); - entry.setLineCount(1); - } - return entry; - }).collect(Collectors.toSet()); - - // build the file report with the entries for this specific file - var fileReport = new CoverageFileReport(); - // 'src/' needs to be prepended to match the repositories' relative file path - String filePath = "src/" + pathDTO.getPath() + "/" + fileDTO.getFileName(); - fileReport.setFilePath(filePath); - fileReport.setTestwiseCoverageEntries(coverageEntriesPerFile); - fileCoverageReports.add(fileReport); - } - testCoverageReports.addAll(fileCoverageReports); - } - - // extract the test case name from the uniformPath - String[] split = coveragePerTestDTO.getUniformPath().split("/"); - String receivedTestCaseName = split[split.length - 1]; - fileReportsByTestName.put(receivedTestCaseName, testCoverageReports); - }); - - return fileReportsByTestName; - } - - /** - * Creates a coverage report from a testwise coverage report. - * Test case names are resolved to a test case of the given programming exercise, adds this reference to the given - * entries and saves the entries with the test case reference to the database. - * In case, no test case could be found for the given name, the report for this test case will not be saved - * - * @param fileReportByTestCaseName a map containing the test case name as a key and the file coverage reports without the - * reference to a test case as a value - * @param submission the solution programming submission for which the report is updated - * @param exercise the exercise for which the report should be updated - */ - public void createTestwiseCoverageReport(Map> fileReportByTestCaseName, ProgrammingExercise exercise, ProgrammingSubmission submission) { - // If the report already exists, do not create a new report. This is the case if the build plan will be re-run - boolean reportAlreadyExists = coverageReportRepository.existsBySubmissionId(submission.getId()); - if (reportAlreadyExists) { - return; - } - - var testCases = programmingExerciseTestCaseRepository.findByExerciseId(exercise.getId()); - var solutionLineCountByFilePath = getLineCountByFilePath(submission); - - // Save the full report with the test case and submission, but without the individual file reports as they do not have an ID yet - var fullReport = new CoverageReport(); - fullReport.setSubmission(submission); - var savedFullReport = coverageReportRepository.save(fullReport); - - // The file reports unique for a file path. the set is used to aggregate multiple file reports from multiple test - // cases into one file report - var uniqueFileReports = new HashSet(); - - fileReportByTestCaseName.forEach((testCaseName, fileReports) -> { - // retrieve the test matching the extracted test case name - var optionalTestCase = testCases.stream() - .filter(testCase -> testCaseName.equals(testCase.getTestName()) || testCaseName.replace("()", "").equals(testCase.getTestName())).findFirst(); - if (optionalTestCase.isEmpty()) { - log.error("No test case with name {} could be found when matching with the testwise coverage", testCaseName); - return; - } - var testCase = optionalTestCase.get(); - - fileReports.forEach(fileReport -> { - // If the file does not exist in the solution repository, no file report will be created - // This is for example the case if the test itself invokes code in a test class - if (solutionLineCountByFilePath.get(fileReport.getFilePath()) == null) { - return; - } - - // Temporarily save the testwise entries for the current file report as the entries do not have - // an ID and therefore cause an exception when the file report is saved to the database - var testwiseEntries = fileReport.getTestwiseCoverageEntries(); - - var optionalReportWithSameName = uniqueFileReports.stream().filter(report -> report.getFilePath().equals(fileReport.getFilePath())).findFirst(); - CoverageFileReport savedFileReport; - - if (optionalReportWithSameName.isEmpty()) { - // Remove the testwise entries temporarily as they do not have an ID yet - fileReport.setTestwiseCoverageEntries(Collections.emptySet()); - fileReport.setFullReport(savedFullReport); - // Save the line count - var lineCount = solutionLineCountByFilePath.get(fileReport.getFilePath()); - fileReport.setLineCount(lineCount); - - savedFileReport = coverageFileReportRepository.save(fileReport); - } - else { - var reportWithSameName = optionalReportWithSameName.get(); - savedFileReport = coverageFileReportRepository.save(reportWithSameName); - } - uniqueFileReports.add(savedFileReport); - - // Save all entries for the current file report to the database - testwiseEntries.forEach(entry -> { - entry.setTestCase(testCase); - entry.setFileReport(savedFileReport); - testwiseCoverageReportEntryRepository.save(entry); - }); - }); - }); - - var updatedFullReport = coverageReportRepository.findCoverageReportByIdWithEagerFileReportsAndEntriesElseThrow(savedFullReport.getId()); - // Calculate the unique line count for all file reports and save this value to the database - var coveredLinesCountByFilePath = calculateAndSaveUniqueLineCountsByFilePath(updatedFullReport); - - // Calculate the aggregated covered line ratio over all files - double aggregatedCoveredLineRatio = calculateAggregatedLineCoverage(solutionLineCountByFilePath, coveredLinesCountByFilePath); - updatedFullReport.setCoveredLineRatio(aggregatedCoveredLineRatio); - coverageReportRepository.save(updatedFullReport); - } - - /** - * Returns the line count by file name for all files in the solution repository for the last submission. - * - * @return line count by file name of files in the last submission's solution repository - */ - private Map getLineCountByFilePath(ProgrammingSubmission submission) { - try { - var solutionParticipation = (SolutionProgrammingExerciseParticipation) submission.getParticipation(); - var solutionRepo = gitService.getOrCheckoutRepository(solutionParticipation.getVcsRepositoryUri(), true); - gitService.resetToOriginHead(solutionRepo); - gitService.pullIgnoreConflicts(solutionRepo); - var solutionFiles = repositoryService.getFilesContentFromWorkingCopy(solutionRepo); - var result = new HashMap(); - solutionFiles.forEach((filePath, value) -> { - // do not count lines for non-java/kotlin files - if (!(filePath.endsWith(".java") || filePath.endsWith(".kt"))) { - return; - } - var lineCount = value.split("\n").length + 1; - result.put(filePath, lineCount); - }); - return result; - } - catch (GitAPIException e) { - log.error("Exception while generating testwise coverage report", e); - throw new InternalServerErrorException("Error while generating testwise coverage report: " + e.getMessage()); - } - } - - /** - * Calculates the aggregated covered line ratio for all file reports. - * - * @param lineCountByFileName the general line count by file name - * @param coveredLineCountByFileName the covered line count by file name - * @return the covered line ratio for all files - */ - private double calculateAggregatedLineCoverage(Map lineCountByFileName, Map coveredLineCountByFileName) { - var aggregatedLineCount = lineCountByFileName.values().stream().mapToInt(Integer::intValue).sum(); - if (aggregatedLineCount == 0) { - return 0; - } - var aggregatedCoveredLineCount = coveredLineCountByFileName.values().stream().mapToInt(Integer::intValue).sum(); - - return aggregatedCoveredLineCount / (double) aggregatedLineCount; - } - - /** - * Calculate the unique covered line count for all file reports and save this value to the database for all - * individual file reports. CoverageFileReports can contain multiple TestwiseCoverageReportEntries referencing - * the same lines, but referencing a different test case. This mapping is still required, but simple summing may - * count the same covered lines multiple times. - * - * @param report the report for which the line counts of its file reports should be calculated and saved - * @return a map with the number of covered lines (value) by file path (key) - */ - private Map calculateAndSaveUniqueLineCountsByFilePath(CoverageReport report) { - var coveredLinesByFilePath = new HashMap(); - report.getFileReports().forEach(fileReport -> { - var lineSet = new HashSet(); - fileReport.getTestwiseCoverageEntries() - .forEach(entry -> lineSet.addAll(IntStream.rangeClosed(entry.getStartLine(), entry.getStartLine() + entry.getLineCount() - 1).boxed().toList())); - fileReport.setCoveredLineCount(lineSet.size()); - coverageFileReportRepository.save(fileReport); - coveredLinesByFilePath.put(fileReport.getFilePath(), lineSet.size()); - }); - return coveredLinesByFilePath; - } - - /** - * Return the test-wise coverage report for the latest solution submission for a programming exercise without the file reports. - * - * @param exerciseId the exercise id for which the latest coverage report should be retrieved - * @return an Optional of the test-wise coverage report for the latest solution submission without the file reports - * if a report exists for the latest submission, otherwise an empty Optional - */ - public Optional getCoverageReportForLatestSolutionSubmissionFromProgrammingExercise(long exerciseId) { - var reports = coverageReportRepository.getLatestCoverageReportsWithLegalSubmissionsForProgrammingExercise(exerciseId, Pageable.ofSize(1)); - if (reports.isEmpty()) { - return Optional.empty(); - } - return Optional.of(reports.getFirst()); - } - - /** - * Return the full test-wise coverage report for the latest solution submission for a programming exercise containing all file reports - * - * @param exerciseId the exercise id for which the latest coverage report should be retrieved - * @return an Optional of the full test-wise coverage report for the latest solution submission with all file reports - * if a report exists for the latest submission, otherwise an empty Optional - */ - public Optional getFullCoverageReportForLatestSolutionSubmissionFromProgrammingExercise(long exerciseId) { - var reports = coverageReportRepository.getLatestCoverageReportsForLegalSubmissionsForProgrammingExerciseWithEagerFileReportsAndEntries(exerciseId, Pageable.ofSize(1)); - if (reports.isEmpty()) { - return Optional.empty(); - } - return Optional.of(reports.getFirst()); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralBlackboard.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralBlackboard.java deleted file mode 100644 index dd1417117e6b..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralBlackboard.java +++ /dev/null @@ -1,58 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral; - -import java.util.List; -import java.util.Map; - -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; - -/** - * The blackboard for creating SolutionEntries for behavioral test cases utilizing the git-diff and test-wise coverage report. - */ -public class BehavioralBlackboard { - - private final ProgrammingExerciseGitDiffReport gitDiffReport; - - private final CoverageReport coverageReport; - - private final Map solutionRepoFiles; - - private List groupedFiles; - - private List solutionEntries; - - public BehavioralBlackboard(ProgrammingExerciseGitDiffReport gitDiffReport, CoverageReport coverageReport, Map solutionRepoFiles) { - this.gitDiffReport = gitDiffReport; - this.coverageReport = coverageReport; - this.solutionRepoFiles = solutionRepoFiles; - } - - public ProgrammingExerciseGitDiffReport getGitDiffReport() { - return gitDiffReport; - } - - public CoverageReport getCoverageReport() { - return coverageReport; - } - - public Map getSolutionRepoFiles() { - return solutionRepoFiles; - } - - public List getSolutionEntries() { - return solutionEntries; - } - - public void setSolutionEntries(List solutionEntries) { - this.solutionEntries = solutionEntries; - } - - public List getGroupedFiles() { - return groupedFiles; - } - - public void setGroupedFiles(List groupedFiles) { - this.groupedFiles = groupedFiles; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralSolutionEntryGenerationException.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralSolutionEntryGenerationException.java deleted file mode 100644 index 118662864564..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralSolutionEntryGenerationException.java +++ /dev/null @@ -1,18 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral; - -/** - * Exception used for the generation of solution entries for behavioral test cases - * It is thrown if there was an error while generating the solution entries - */ -public class BehavioralSolutionEntryGenerationException extends Exception { - - private static final String MESSAGE_PREFIX = "Unable to generate behavioral solution entries: "; - - public BehavioralSolutionEntryGenerationException(String message) { - super(MESSAGE_PREFIX + message); - } - - public BehavioralSolutionEntryGenerationException(String message, Throwable cause) { - super(MESSAGE_PREFIX + message, cause); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralTestCaseService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralTestCaseService.java deleted file mode 100644 index 71df2fdc6690..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/BehavioralTestCaseService.java +++ /dev/null @@ -1,201 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.util.Arrays; -import java.util.Collections; -import java.util.List; -import java.util.Map; -import java.util.Objects; - -import org.eclipse.jgit.api.errors.GitAPIException; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Profile; -import org.springframework.stereotype.Service; - -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; -import de.tum.cit.aet.artemis.programming.repository.SolutionProgrammingExerciseParticipationRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseSolutionEntryRepository; -import de.tum.cit.aet.artemis.programming.service.GitService; -import de.tum.cit.aet.artemis.programming.service.RepositoryService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseGitDiffReportService; -import de.tum.cit.aet.artemis.programming.service.hestia.TestwiseCoverageService; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.AddUncoveredLinesAsPotentialCodeBlocks; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.BehavioralKnowledgeSource; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.CombineChangeBlocks; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.CreateCommonChangeBlocks; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.CreateSolutionEntries; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.DropRemovedGitDiffEntries; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.ExtractChangedLines; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.ExtractCoveredLines; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.FindCommonLines; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.GroupGitDiffAndCoverageEntriesByFilePathAndTestCase; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.InsertFileContents; - -/** - * Service for handling Solution Entries of behavioral Test Cases. - */ -@Profile(PROFILE_CORE) -@Service -public class BehavioralTestCaseService { - - private static final Logger log = LoggerFactory.getLogger(BehavioralTestCaseService.class); - - private final GitService gitService; - - private final RepositoryService repositoryService; - - private final TestwiseCoverageService testwiseCoverageService; - - private final ProgrammingExerciseTestCaseRepository testCaseRepository; - - private final ProgrammingExerciseSolutionEntryRepository solutionEntryRepository; - - private final ProgrammingExerciseGitDiffReportService programmingExerciseGitDiffReportService; - - private final SolutionProgrammingExerciseParticipationRepository solutionProgrammingExerciseParticipationRepository; - - public BehavioralTestCaseService(GitService gitService, RepositoryService repositoryService, TestwiseCoverageService testwiseCoverageService, - ProgrammingExerciseTestCaseRepository testCaseRepository, ProgrammingExerciseSolutionEntryRepository solutionEntryRepository, - ProgrammingExerciseGitDiffReportService programmingExerciseGitDiffReportService, - SolutionProgrammingExerciseParticipationRepository solutionProgrammingExerciseParticipationRepository) { - this.gitService = gitService; - this.repositoryService = repositoryService; - this.testCaseRepository = testCaseRepository; - this.testwiseCoverageService = testwiseCoverageService; - this.solutionEntryRepository = solutionEntryRepository; - this.programmingExerciseGitDiffReportService = programmingExerciseGitDiffReportService; - this.solutionProgrammingExerciseParticipationRepository = solutionProgrammingExerciseParticipationRepository; - } - - /** - * Generates the solution entries for all behavioral test cases of a programming exercise. - * This uses the git-diff report, the testwise coverage, and the test cases of the programming exercise and - * requires them to exist. - * Therefore, this method also requires the exercise to have testwiseCoverageEnabled set to true. - * - * @param programmingExercise The programming exercise - * @return The new behavioral solution entries - * @throws BehavioralSolutionEntryGenerationException If there was an error while generating the solution entries - */ - public List generateBehavioralSolutionEntries(ProgrammingExercise programmingExercise) throws BehavioralSolutionEntryGenerationException { - if (!programmingExercise.getBuildConfig().isTestwiseCoverageEnabled()) { - throw new BehavioralSolutionEntryGenerationException("This feature is only supported for Java Exercises with active Testwise Coverage"); - } - var testCases = testCaseRepository.findByExerciseIdWithSolutionEntriesAndActive(programmingExercise.getId(), true); - if (testCases.isEmpty()) { - throw new BehavioralSolutionEntryGenerationException("Test cases have not been received yet"); - } - var gitDiffReport = programmingExerciseGitDiffReportService.getOrCreateReportOfExercise(programmingExercise); - if (gitDiffReport == null) { - throw new BehavioralSolutionEntryGenerationException("Git-Diff Report has not been generated"); - } - - var coverageReport = testwiseCoverageService.getFullCoverageReportForLatestSolutionSubmissionFromProgrammingExercise(programmingExercise.getId()).orElse(null); - if (coverageReport == null) { - throw new BehavioralSolutionEntryGenerationException("Testwise coverage report has not been generated"); - } - log.info("Generating the behavioral solution entries for programming exercise {}", programmingExercise.getId()); - - var solutionRepoFiles = readSolutionRepo(programmingExercise); - - var blackboard = new BehavioralBlackboard(gitDiffReport, coverageReport, solutionRepoFiles); - applyKnowledgeSources(blackboard); - var newSolutionEntries = blackboard.getSolutionEntries(); - if (newSolutionEntries == null || newSolutionEntries.isEmpty()) { - throw new BehavioralSolutionEntryGenerationException("No solution entry was generated"); - } - // Remove temporary id before saving - for (ProgrammingExerciseSolutionEntry solutionEntry : newSolutionEntries) { - solutionEntry.setId(null); - } - newSolutionEntries = solutionEntryRepository.saveAll(newSolutionEntries); - - // Get all old solution entries - var oldSolutionEntries = newSolutionEntries.stream().map(ProgrammingExerciseSolutionEntry::getTestCase) - .map(testCase1 -> testCases.stream().filter(testCase2 -> Objects.equals(testCase1.getId(), testCase2.getId())).findFirst().orElse(null)).filter(Objects::nonNull) - .flatMap(testCase -> testCase.getSolutionEntries().stream()).distinct().toList(); - - // Save new solution entries - newSolutionEntries = solutionEntryRepository.saveAll(newSolutionEntries); - - // Delete old solution entries - solutionEntryRepository.deleteAll(oldSolutionEntries); - - log.info("{} behavioral solution entries for programming exercise {} have been generated", newSolutionEntries.size(), programmingExercise.getId()); - return newSolutionEntries; - } - - /** - * Utilizing the blackboard pattern this method creates the behavioral solution entries step by step. - * Look at the specific KnowledgeSources to learn more. - * - * @param blackboard The blackboard containing the base information - * @throws BehavioralSolutionEntryGenerationException If there was an error while generating the solution entries - */ - private void applyKnowledgeSources(BehavioralBlackboard blackboard) throws BehavioralSolutionEntryGenerationException { - // Create knowledge sources (Turning the formatter off to make the code more readable) - // @formatter:off - List behavioralKnowledgeSources = Arrays.asList( - new DropRemovedGitDiffEntries(blackboard), - new GroupGitDiffAndCoverageEntriesByFilePathAndTestCase(blackboard), - new ExtractCoveredLines(blackboard), - new ExtractChangedLines(blackboard), - new FindCommonLines(blackboard), - new CreateCommonChangeBlocks(blackboard), - new InsertFileContents(blackboard), - new AddUncoveredLinesAsPotentialCodeBlocks(blackboard), - new CombineChangeBlocks(blackboard), - new CreateSolutionEntries(blackboard) - ); - // @formatter:on - - boolean done = false; - int iterations = 0; - while (!done) { - boolean didChanges = false; - for (BehavioralKnowledgeSource behavioralKnowledgeSource : behavioralKnowledgeSources) { - if (behavioralKnowledgeSource.executeCondition()) { - log.debug("Executing knowledge source {}", behavioralKnowledgeSource.getClass().getSimpleName()); - didChanges = behavioralKnowledgeSource.executeAction() || didChanges; - } - } - done = !didChanges; - iterations++; - // Safeguard to prevent an infinite loop - if (iterations >= 200) { - throw new BehavioralSolutionEntryGenerationException("The creation of the solution entries got stuck and was cancelled"); - } - } - } - - /** - * Reads the contents of all files in the solution repository and returns them mapped with the file path as the key. - * - * @param programmingExercise The programming exercise - * @return The file path of each file mapped to their contents - * @throws BehavioralSolutionEntryGenerationException If there was an error while reading the solution repository - */ - private Map readSolutionRepo(ProgrammingExercise programmingExercise) throws BehavioralSolutionEntryGenerationException { - try { - log.debug("Reading the contents of the solution repository"); - var solutionParticipationOptional = solutionProgrammingExerciseParticipationRepository.findByProgrammingExerciseId(programmingExercise.getId()); - if (solutionParticipationOptional.isEmpty()) { - return Collections.emptyMap(); - } - var solutionParticipation = solutionParticipationOptional.get(); - var solutionRepo = gitService.getOrCheckoutRepository(solutionParticipation.getVcsRepositoryUri(), true); - - gitService.resetToOriginHead(solutionRepo); - gitService.pullIgnoreConflicts(solutionRepo); - - return repositoryService.getFilesContentFromWorkingCopy(solutionRepo); - } - catch (GitAPIException e) { - throw new BehavioralSolutionEntryGenerationException("Error while reading solution repository", e); - } - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/GroupedFile.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/GroupedFile.java deleted file mode 100644 index 62eab75c4b1f..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/GroupedFile.java +++ /dev/null @@ -1,150 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral; - -import java.util.Collection; -import java.util.Objects; -import java.util.Set; -import java.util.SortedSet; -import java.util.TreeSet; - -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; - -/** - * One GroupedFile groups the {@link ProgrammingExerciseGitDiffEntry}s and {@link TestwiseCoverageReportEntry}s together - * that belong to the same file. For each {@link ProgrammingExerciseTestCase} that covered the file a separate GroupedFile exists. - */ -public class GroupedFile { - - // The path of the file - private final String filePath; - - // The test case that covered something in this file - private final ProgrammingExerciseTestCase testCase; - - // All changes between the template and solution repositories in this file - private final Set gitDiffEntries; - - // All coverage entries of the test case in this file - private final Set coverageReportEntries; - - // The content of this file - private String fileContent; - - // The lines of this file that were changed - private Set changedLines; - - // The lines of this file that were covered by the test case - private Set coveredLines; - - // The lines in this file that were both covered and changed - private SortedSet commonLines; - - // The changes in this file that should be included in the solution entries - private SortedSet commonChanges; - - public GroupedFile(String filePath, ProgrammingExerciseTestCase testCase, Set gitDiffEntries, - Set coverageReportEntries) { - this.filePath = filePath; - this.testCase = testCase; - this.gitDiffEntries = gitDiffEntries; - this.coverageReportEntries = coverageReportEntries; - } - - public String getFilePath() { - return filePath; - } - - public ProgrammingExerciseTestCase getTestCase() { - return testCase; - } - - public Set getGitDiffEntries() { - return gitDiffEntries; - } - - public Set getCoverageReportEntries() { - return coverageReportEntries; - } - - public String getFileContent() { - return fileContent; - } - - public void setFileContent(String fileContent) { - this.fileContent = fileContent; - } - - public Set getChangedLines() { - return changedLines; - } - - public void setChangedLines(Set changedLines) { - this.changedLines = changedLines; - } - - public Set getCoveredLines() { - return coveredLines; - } - - public void setCoveredLines(Set coveredLines) { - this.coveredLines = coveredLines; - } - - public SortedSet getCommonLines() { - return commonLines; - } - - public void setCommonLines(Collection commonLines) { - this.commonLines = new TreeSet<>(commonLines); - } - - public SortedSet getCommonChanges() { - return commonChanges; - } - - public void setCommonChanges(Collection commonChanges) { - this.commonChanges = new TreeSet<>(commonChanges); - } - - @Override - public int hashCode() { - return Objects.hash(filePath, testCase, gitDiffEntries, coverageReportEntries, fileContent, changedLines, coveredLines, commonLines, commonChanges); - } - - @Override - public boolean equals(Object obj) { - if (this == obj) { - return true; - } - if (obj == null || getClass() != obj.getClass()) { - return false; - } - GroupedFile that = (GroupedFile) obj; - return Objects.equals(filePath, that.filePath) && Objects.equals(testCase, that.testCase) && Objects.equals(gitDiffEntries, that.gitDiffEntries) - && Objects.equals(coverageReportEntries, that.coverageReportEntries) && Objects.equals(fileContent, that.fileContent) - && Objects.equals(changedLines, that.changedLines) && Objects.equals(coveredLines, that.coveredLines) && Objects.equals(commonLines, that.commonLines) - && Objects.equals(commonChanges, that.commonChanges); - } - - public record ChangeBlock(SortedSet lines, boolean isPotential) implements Comparable { - - public ChangeBlock(Collection lines) { - this(new TreeSet<>(lines), false); - } - - public ChangeBlock(Collection lines, boolean isPotential) { - this(new TreeSet<>(lines), isPotential); - } - - @Override - public int compareTo(GroupedFile.ChangeBlock other) { - return Integer.compare(this.lines.first(), other.lines.first()); - } - - public boolean intersectsOrTouches(ChangeBlock other) { - return (this.lines.first() > other.lines.first() && this.lines.first() <= other.lines.last() + 1) - || (this.lines.first() < other.lines.first() && this.lines.last() >= other.lines.first() - 1); - } - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/AddUncoveredLinesAsPotentialCodeBlocks.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/AddUncoveredLinesAsPotentialCodeBlocks.java deleted file mode 100644 index 3bacd3966662..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/AddUncoveredLinesAsPotentialCodeBlocks.java +++ /dev/null @@ -1,110 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import java.util.TreeSet; -import java.util.regex.Pattern; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile.ChangeBlock; - -/** - * For each {@link GroupedFile}: - * There are certain lines that are not covered by Jacoco (which is used for the testwise coverage) that may still be - * relevant to the code. This includes for example the `else` expression and curly braces. - * For each {@link ChangeBlock}: - * Check if there are such lines before or after the ChangeBlock. If there are add the entire prefix/postfix as a - * potential ChangeBlock to the GroupedFile. - */ -public class AddUncoveredLinesAsPotentialCodeBlocks extends BehavioralKnowledgeSource { - - private static final Pattern CURLY_BRACES_PATTERN = Pattern.compile("\\s*[}{]\\s*"); - - private static final Pattern ELSE_PATTERN = Pattern.compile("\\s*}?\\s*else\\s*\\{?\\s*"); - - private static final Pattern EMPTY_LINE_PATTERN = Pattern.compile("\\s*"); - - public AddUncoveredLinesAsPotentialCodeBlocks(BehavioralBlackboard blackboard) { - super(blackboard); - } - - @Override - public boolean executeCondition() { - return blackboard.getGroupedFiles() != null && blackboard.getGroupedFiles().stream().noneMatch(groupedFile -> groupedFile.getFileContent() == null) - && blackboard.getGroupedFiles().stream().noneMatch(groupedFile -> groupedFile.getCommonChanges() == null); - } - - @Override - public boolean executeAction() { - boolean didChanges = false; - - for (GroupedFile groupedFile : blackboard.getGroupedFiles()) { - var newChangeBlocks = new TreeSet(); - - for (ChangeBlock commonChange : groupedFile.getCommonChanges()) { - var firstLine = commonChange.lines().first(); - var potentialPrefix = getPotentialPrefix(firstLine, groupedFile.getFileContent()); - if (potentialPrefix != null) { - newChangeBlocks.add(potentialPrefix); - } - var lastLine = commonChange.lines().last(); - var potentialPostfix = getPotentialPostfix(lastLine, groupedFile.getFileContent()); - if (potentialPostfix != null) { - newChangeBlocks.add(potentialPostfix); - } - } - - if (!newChangeBlocks.isEmpty()) { - groupedFile.getCommonChanges().addAll(newChangeBlocks); - didChanges = true; - } - } - - return didChanges; - } - - private ChangeBlock getPotentialPrefix(int firstLine, String fileContent) { - var potentialLines = new TreeSet(); - var lineContents = fileContent.split("\n"); - // Starting two lines before the first line, as first line is the line in the file which starts at 1 and not 0 - for (int i = firstLine - 2; i >= 0; i--) { - var lineContent = lineContents[i]; - if (doesLineMatch(lineContent)) { - potentialLines.add(i + 1); - } - else { - break; - } - } - if (potentialLines.isEmpty()) { - return null; - } - else { - return new ChangeBlock(potentialLines, true); - } - } - - private ChangeBlock getPotentialPostfix(int lastLine, String fileContent) { - var potentialLines = new TreeSet(); - var lineContents = fileContent.split("\n"); - // Starting at last line instead of lastLine + 1, as last line is the line in the file which starts at 1 and not 0 - for (int i = lastLine; i < lineContents.length; i++) { - var lineContent = lineContents[i]; - if (doesLineMatch(lineContent)) { - potentialLines.add(i + 1); - } - else { - break; - } - } - if (potentialLines.isEmpty()) { - return null; - } - else { - return new ChangeBlock(potentialLines, true); - } - } - - private boolean doesLineMatch(String lineContent) { - return CURLY_BRACES_PATTERN.matcher(lineContent).matches() || ELSE_PATTERN.matcher(lineContent).matches() || EMPTY_LINE_PATTERN.matcher(lineContent).matches(); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/BehavioralKnowledgeSource.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/BehavioralKnowledgeSource.java deleted file mode 100644 index a96c534a2623..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/BehavioralKnowledgeSource.java +++ /dev/null @@ -1,27 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralSolutionEntryGenerationException; - -public abstract class BehavioralKnowledgeSource { - - protected final BehavioralBlackboard blackboard; - - public BehavioralKnowledgeSource(BehavioralBlackboard blackboard) { - this.blackboard = blackboard; - } - - /** - * Checks if the knowledge source can be applied - * - * @return true if the knowledge source can be applied - */ - public abstract boolean executeCondition(); - - /** - * Applies this knowledge source to the blackboard - * - * @return true if changes were made - */ - public abstract boolean executeAction() throws BehavioralSolutionEntryGenerationException; -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CombineChangeBlocks.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CombineChangeBlocks.java deleted file mode 100644 index d7f1b0c8f764..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CombineChangeBlocks.java +++ /dev/null @@ -1,61 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import java.util.ArrayList; -import java.util.TreeSet; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; - -/** - * For each {@link GroupedFile}: - * Combines the {@link GroupedFile.ChangeBlock}s that have intersecting or directly attaching lines. - * E.g. if you have the block with the lines `1, 2, 3` and another with the lines `4, 5, 6`, these would be combined - * into one block with the lines `1, 2, 3, 4, 5, 6`. - */ -public class CombineChangeBlocks extends BehavioralKnowledgeSource { - - public CombineChangeBlocks(BehavioralBlackboard blackboard) { - super(blackboard); - } - - @Override - public boolean executeCondition() { - return blackboard.getGroupedFiles() != null && blackboard.getGroupedFiles().stream().noneMatch(groupedFile -> groupedFile.getCommonChanges() == null); - } - - @Override - public boolean executeAction() { - boolean didChanges = false; - - for (GroupedFile groupedFile : blackboard.getGroupedFiles()) { - while (true) { - var currentChangeBlocks = new ArrayList<>(groupedFile.getCommonChanges()); - var newChangeBlocks = new ArrayList(); - for (int i = 0; i < currentChangeBlocks.size(); i++) { - var currentChangeBlock = currentChangeBlocks.get(i); - // If this is not the last change block check if it can be merged with the next change block - if (i < currentChangeBlocks.size() - 1) { - var nextChangeBlock = currentChangeBlocks.get(i + 1); - if (currentChangeBlock.intersectsOrTouches(nextChangeBlock)) { - var lines = new TreeSet<>(currentChangeBlock.lines()); - lines.addAll(nextChangeBlock.lines()); - newChangeBlocks.add(new GroupedFile.ChangeBlock(lines)); - // Skip the next change block, as it has already been processed - i++; - continue; - } - } - newChangeBlocks.add(currentChangeBlock); - } - if (!newChangeBlocks.equals(currentChangeBlocks)) { - groupedFile.setCommonChanges(newChangeBlocks); - didChanges = true; - } - else { - break; - } - } - } - return didChanges; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CreateCommonChangeBlocks.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CreateCommonChangeBlocks.java deleted file mode 100644 index 5da770d0d9bb..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CreateCommonChangeBlocks.java +++ /dev/null @@ -1,64 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import java.util.TreeSet; -import java.util.stream.IntStream; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; - -/** - * For each {@link GroupedFile}: - * Takes the common lines (intersection of covered and changed lines) and creates {@link GroupedFile.ChangeBlock}s - * from them. Each ChangeBlock represents one continuous block of common lines. - */ -public class CreateCommonChangeBlocks extends BehavioralKnowledgeSource { - - public CreateCommonChangeBlocks(BehavioralBlackboard blackboard) { - super(blackboard); - } - - @Override - public boolean executeCondition() { - return blackboard.getGroupedFiles() != null && blackboard.getGroupedFiles().stream().noneMatch(groupedFile -> groupedFile.getCommonLines() == null) - && blackboard.getGroupedFiles().stream().anyMatch(groupedFile -> groupedFile.getCommonChanges() == null); - } - - @Override - public boolean executeAction() { - boolean didChanges = false; - - for (GroupedFile groupedFile : blackboard.getGroupedFiles()) { - var changeBlocks = new TreeSet(); - - Integer previousLine = null; - Integer startLine = null; - int lineCount = 0; - - for (Integer currentLine : groupedFile.getCommonLines()) { - if (startLine == null) { - startLine = currentLine; - } - // Check if this is a new change block - if (previousLine != null && currentLine - 1 > previousLine) { - changeBlocks.add(new GroupedFile.ChangeBlock(IntStream.range(startLine, startLine + lineCount).boxed().toList())); - lineCount = 0; - startLine = currentLine; - } - lineCount++; - previousLine = currentLine; - } - - // Add the last change block if any existA - if (startLine != null) { - changeBlocks.add(new GroupedFile.ChangeBlock(IntStream.range(startLine, startLine + lineCount).boxed().toList())); - } - - if (!changeBlocks.equals(groupedFile.getCommonChanges())) { - groupedFile.setCommonChanges(changeBlocks); - didChanges = true; - } - } - - return didChanges; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CreateSolutionEntries.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CreateSolutionEntries.java deleted file mode 100644 index 90219039cbb1..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/CreateSolutionEntries.java +++ /dev/null @@ -1,57 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.stream.Collectors; - -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; - -/** - * For each {@link GroupedFile}: - * Creates one {@link ProgrammingExerciseSolutionEntry} for each non-potential {@link GroupedFile.ChangeBlock} - */ -public class CreateSolutionEntries extends BehavioralKnowledgeSource { - - public CreateSolutionEntries(BehavioralBlackboard blackboard) { - super(blackboard); - } - - @Override - public boolean executeCondition() { - return blackboard.getGroupedFiles() != null && blackboard.getGroupedFiles().stream().noneMatch(groupedFile -> groupedFile.getCommonChanges() == null); - } - - @Override - public boolean executeAction() { - var solutionEntries = new ArrayList(); - for (GroupedFile groupedFile : blackboard.getGroupedFiles()) { - for (GroupedFile.ChangeBlock changeBlock : groupedFile.getCommonChanges()) { - if (!changeBlock.isPotential()) { - solutionEntries.add(createSolutionEntry(groupedFile, changeBlock)); - } - } - } - if (solutionEntries.equals(blackboard.getSolutionEntries())) { - return false; - } - blackboard.setSolutionEntries(solutionEntries); - return !blackboard.getSolutionEntries().isEmpty(); - } - - private ProgrammingExerciseSolutionEntry createSolutionEntry(GroupedFile groupedFile, GroupedFile.ChangeBlock changeBlock) { - var solutionEntry = new ProgrammingExerciseSolutionEntry(); - // Set temporary id, as equals checks won't work otherwise - solutionEntry.setId(0L); - solutionEntry.setLine(changeBlock.lines().first()); - solutionEntry.setFilePath(groupedFile.getFilePath()); - solutionEntry.setTestCase(groupedFile.getTestCase()); - var fileContent = groupedFile.getFileContent(); - if (fileContent != null) { - var code = Arrays.stream(fileContent.split("\n")).skip(changeBlock.lines().first() - 1).limit(changeBlock.lines().size()).collect(Collectors.joining("\n")); - solutionEntry.setCode(code); - } - return solutionEntry; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/DropRemovedGitDiffEntries.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/DropRemovedGitDiffEntries.java deleted file mode 100644 index 679565d2c0e6..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/DropRemovedGitDiffEntries.java +++ /dev/null @@ -1,35 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import java.util.HashSet; -import java.util.stream.Collectors; - -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; - -/** - * Remove all {@link ProgrammingExerciseGitDiffEntry} from the - * {@link ProgrammingExerciseGitDiffReport} of the {@link BehavioralBlackboard} - * that represents consecutive blocks of removed code. - * Entries cannot be generated for removed code, therefore we have to drop them from the git diff report of the blackboard. - */ -public class DropRemovedGitDiffEntries extends BehavioralKnowledgeSource { - - public DropRemovedGitDiffEntries(BehavioralBlackboard blackboard) { - super(blackboard); - } - - @Override - public boolean executeCondition() { - return blackboard.getGitDiffReport() != null && blackboard.getGitDiffReport().getEntries() != null - && blackboard.getGitDiffReport().getEntries().stream().anyMatch(entry -> entry.getStartLine() == null || entry.getLineCount() == null); - } - - @Override - public boolean executeAction() { - var nonRemovedEntries = blackboard.getGitDiffReport().getEntries().stream().filter(entry -> entry.getStartLine() != null && entry.getLineCount() != null) - .collect(Collectors.toCollection(HashSet::new)); - blackboard.getGitDiffReport().setEntries(nonRemovedEntries); - return !nonRemovedEntries.isEmpty(); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/ExtractChangedLines.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/ExtractChangedLines.java deleted file mode 100644 index 3b259d17c21c..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/ExtractChangedLines.java +++ /dev/null @@ -1,35 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import java.util.stream.Collectors; -import java.util.stream.IntStream; - -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; - -/** - * For each {@link GroupedFile}: - * Extracts the lines that were changed in the file (of the GroupedFile) from the - * {@link ProgrammingExerciseGitDiffEntry}s - */ -public class ExtractChangedLines extends BehavioralKnowledgeSource { - - public ExtractChangedLines(BehavioralBlackboard blackboard) { - super(blackboard); - } - - @Override - public boolean executeCondition() { - return blackboard.getGroupedFiles() != null && blackboard.getGroupedFiles().stream().anyMatch(groupedFile -> groupedFile.getChangedLines() == null); - } - - @Override - public boolean executeAction() { - for (GroupedFile groupedFile : blackboard.getGroupedFiles()) { - groupedFile.setChangedLines(groupedFile.getGitDiffEntries().stream() - .flatMapToInt(gitDiffEntry -> IntStream.range(gitDiffEntry.getStartLine(), gitDiffEntry.getStartLine() + gitDiffEntry.getLineCount())).boxed() - .collect(Collectors.toSet())); - } - return !blackboard.getGroupedFiles().stream().allMatch(groupedFile -> groupedFile.getChangedLines().isEmpty()); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/ExtractCoveredLines.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/ExtractCoveredLines.java deleted file mode 100644 index 19894d567205..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/ExtractCoveredLines.java +++ /dev/null @@ -1,36 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import java.util.stream.Collectors; -import java.util.stream.IntStream; - -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; - -/** - * For each {@link GroupedFile}: - * Extracts the lines that were covered by the test case in the file (both of the GroupedFile) from the - * {@link TestwiseCoverageReportEntry}s - */ -public class ExtractCoveredLines extends BehavioralKnowledgeSource { - - public ExtractCoveredLines(BehavioralBlackboard blackboard) { - super(blackboard); - } - - @Override - public boolean executeCondition() { - return blackboard.getGroupedFiles() != null && blackboard.getGroupedFiles().stream().anyMatch(groupedFile -> groupedFile.getCoveredLines() == null); - } - - @Override - public boolean executeAction() { - for (GroupedFile groupedFile : blackboard.getGroupedFiles()) { - groupedFile.setCoveredLines(groupedFile.getCoverageReportEntries().stream() - .flatMapToInt( - coverageReportEntry -> IntStream.range(coverageReportEntry.getStartLine(), coverageReportEntry.getStartLine() + coverageReportEntry.getLineCount())) - .boxed().collect(Collectors.toSet())); - } - return !blackboard.getGroupedFiles().stream().allMatch(groupedFile -> groupedFile.getCoveredLines().isEmpty()); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/FindCommonLines.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/FindCommonLines.java deleted file mode 100644 index cf90f0b9af83..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/FindCommonLines.java +++ /dev/null @@ -1,35 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import java.util.ArrayList; -import java.util.List; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; - -/** - * For each {@link GroupedFile}: - * Creates the intersection of the changed lines and the covered lines. - */ -public class FindCommonLines extends BehavioralKnowledgeSource { - - public FindCommonLines(BehavioralBlackboard blackboard) { - super(blackboard); - } - - @Override - public boolean executeCondition() { - return blackboard.getGroupedFiles() != null && blackboard.getGroupedFiles().stream().noneMatch(groupedFile -> groupedFile.getChangedLines() == null) - && blackboard.getGroupedFiles().stream().noneMatch(groupedFile -> groupedFile.getCoveredLines() == null) - && blackboard.getGroupedFiles().stream().anyMatch(groupedFile -> groupedFile.getCommonLines() == null); - } - - @Override - public boolean executeAction() { - for (GroupedFile groupedFile : blackboard.getGroupedFiles()) { - List commonLines = new ArrayList<>(groupedFile.getCoveredLines()); - commonLines.retainAll(groupedFile.getChangedLines()); - groupedFile.setCommonLines(commonLines); - } - return !blackboard.getGroupedFiles().stream().allMatch(groupedFile -> groupedFile.getCommonLines().isEmpty()); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/GroupGitDiffAndCoverageEntriesByFilePathAndTestCase.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/GroupGitDiffAndCoverageEntriesByFilePathAndTestCase.java deleted file mode 100644 index c04311732a2b..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/GroupGitDiffAndCoverageEntriesByFilePathAndTestCase.java +++ /dev/null @@ -1,70 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import java.util.Collections; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Set; -import java.util.SortedSet; -import java.util.TreeSet; -import java.util.stream.Collectors; - -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageFileReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; - -/** - * Knowledge source that takes care of creating the {@link GroupedFile}s used by all other knowledge sources. - * These GroupedFiles are created by grouping all coverage entries and git-diff entries together that belong the same file and test case. - */ -public class GroupGitDiffAndCoverageEntriesByFilePathAndTestCase extends BehavioralKnowledgeSource { - - public GroupGitDiffAndCoverageEntriesByFilePathAndTestCase(BehavioralBlackboard blackboard) { - super(blackboard); - } - - @Override - public boolean executeCondition() { - return blackboard.getGroupedFiles() == null; - } - - @Override - public boolean executeAction() { - var gitDiffEntriesPerFile = blackboard.getGitDiffReport().getEntries().stream() - .collect(Collectors.toMap(ProgrammingExerciseGitDiffEntry::getFilePath, Collections::singleton, (set1, set2) -> { - var entries = new HashSet<>(set1); - entries.addAll(set2); - return entries; - })); - var coverageEntriesPerFile = blackboard.getCoverageReport().getFileReports().stream() - .collect(Collectors.toMap(CoverageFileReport::getFilePath, CoverageFileReport::getTestwiseCoverageEntries)); - - var commonFilePaths = new TreeSet<>(gitDiffEntriesPerFile.keySet()); - commonFilePaths.retainAll(coverageEntriesPerFile.keySet()); - - List groupedFiles = createGroupedFiles(gitDiffEntriesPerFile, coverageEntriesPerFile, commonFilePaths); - if (groupedFiles.isEmpty()) { - return false; - } - else { - blackboard.setGroupedFiles(groupedFiles); - return true; - } - } - - private List createGroupedFiles(Map> gitDiffEntriesPerFile, - Map> coverageEntriesPerFile, SortedSet commonFilePaths) { - return commonFilePaths.stream().flatMap(filePath -> { - var gitDiffEntries = gitDiffEntriesPerFile.get(filePath); - var coverageReportEntries = coverageEntriesPerFile.get(filePath); - return coverageReportEntries.stream().collect(Collectors.toMap(TestwiseCoverageReportEntry::getTestCase, Collections::singleton, (set1, set2) -> { - var entries = new HashSet<>(set1); - entries.addAll(set2); - return entries; - })).entrySet().stream().map(entry -> new GroupedFile(filePath, entry.getKey(), gitDiffEntries, entry.getValue())); - }).toList(); - } - -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/InsertFileContents.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/InsertFileContents.java deleted file mode 100644 index 43296df40286..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/behavioral/knowledgesource/InsertFileContents.java +++ /dev/null @@ -1,34 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralSolutionEntryGenerationException; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; - -/** - * For each {@link GroupedFile}: - * Inserts the contents of the file into the GroupedFile using the filePath. - */ -public class InsertFileContents extends BehavioralKnowledgeSource { - - public InsertFileContents(BehavioralBlackboard blackboard) { - super(blackboard); - } - - @Override - public boolean executeCondition() { - return blackboard.getGroupedFiles() != null && blackboard.getGroupedFiles().stream().anyMatch(groupedFile -> groupedFile.getFileContent() == null); - } - - @Override - public boolean executeAction() throws BehavioralSolutionEntryGenerationException { - for (GroupedFile groupedFile : blackboard.getGroupedFiles()) { - var fileContent = blackboard.getSolutionRepoFiles().get(groupedFile.getFilePath()); - if (fileContent == null) { - throw new BehavioralSolutionEntryGenerationException( - String.format("Unable to find file '%s' in the solution repo despite it being referenced in the git-diff and coverage", groupedFile.getFilePath())); - } - groupedFile.setFileContent(fileContent); - } - return !blackboard.getGroupedFiles().isEmpty(); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralAttribute.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralAttribute.java deleted file mode 100644 index e1b776cf2b1e..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralAttribute.java +++ /dev/null @@ -1,79 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.structural; - -import java.util.ArrayList; -import java.util.List; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.thoughtworks.qdox.model.JavaClass; -import com.thoughtworks.qdox.model.JavaField; - -/** - * Element of the test.json file representing the properties of an attribute of a class - * Used for the generation of solution entries for structural test cases - */ -@JsonIgnoreProperties(ignoreUnknown = true) -class StructuralAttribute implements StructuralElement { - - @JsonProperty(required = true) - private String name; - - @JsonProperty(required = true) - private String type; - - private List modifiers = new ArrayList<>(); - - private List annotations = new ArrayList<>(); - - @Override - public String getSourceCode(StructuralClassElements structuralClassElements, JavaClass solutionClass) { - JavaField solutionAttribute = getSolutionAttribute(solutionClass); - String attributeCode = ""; - if (!this.getAnnotations().isEmpty()) { - attributeCode += getAnnotationsString(this.getAnnotations(), solutionAttribute); - } - if (!this.getModifiers().isEmpty()) { - attributeCode += formatModifiers(this.getModifiers()) + " "; - } - attributeCode += (solutionAttribute == null ? this.getType() : solutionAttribute.getType().getGenericValue()) + " "; - attributeCode += this.getName(); - attributeCode += ";"; - return attributeCode; - } - - private JavaField getSolutionAttribute(JavaClass solutionClass) { - return solutionClass == null ? null : solutionClass.getFields().stream().filter(field -> field.getName().equals(this.getName())).findFirst().orElse(null); - } - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - - public List getModifiers() { - return modifiers; - } - - public void setModifiers(List modifiers) { - this.modifiers = modifiers; - } - - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } - - public List getAnnotations() { - return annotations; - } - - public void setAnnotations(List annotations) { - this.annotations = annotations; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralClass.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralClass.java deleted file mode 100644 index 4cea7c67c954..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralClass.java +++ /dev/null @@ -1,145 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.structural; - -import java.util.ArrayList; -import java.util.List; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.thoughtworks.qdox.model.JavaClass; - -/** - * Element of the test.json file representing the properties of a class - * Used for the generation of solution entries for structural test cases - */ -@JsonIgnoreProperties(ignoreUnknown = true) -class StructuralClass implements StructuralElement { - - @JsonProperty(required = true) - private String name; - - @JsonProperty(value = "package", required = true) - private String packageName; - - private String superclass; - - private List modifiers = new ArrayList<>(); - - @JsonProperty(defaultValue = "false") - private boolean isInterface; - - @JsonProperty(defaultValue = "false") - private boolean isEnum; - - private List interfaces = new ArrayList<>(); - - private List annotations = new ArrayList<>(); - - @Override - public String getSourceCode(StructuralClassElements structuralClassElements, JavaClass solutionClass) { - String classSolutionCode = "package " + this.getPackageName() + ";\n\n"; - - if (!this.getAnnotations().isEmpty()) { - classSolutionCode += getAnnotationsString(this.getAnnotations(), solutionClass); - } - classSolutionCode += getClassHeaderCode(solutionClass); - // Class Body - classSolutionCode += "{\n"; - classSolutionCode += SINGLE_INDENTATION; - if (this.isEnum()) { - classSolutionCode += String.join(", ", structuralClassElements.getEnumValues()); - } - classSolutionCode += "\n}"; - - return classSolutionCode; - } - - private String getClassHeaderCode(JavaClass solutionClass) { - String classHeaderCode = ""; - if (!this.getModifiers().isEmpty()) { - classHeaderCode += formatModifiers(this.getModifiers()) + " "; - } - classHeaderCode += (this.isInterface() ? "interface" : (this.isEnum() ? "enum" : "class")) + " "; - classHeaderCode += this.getName(); - if (solutionClass != null && !solutionClass.getTypeParameters().isEmpty()) { - classHeaderCode += getGenericTypesString(solutionClass.getTypeParameters()); - } - classHeaderCode += " "; - classHeaderCode += getInheritanceCode(); - return classHeaderCode; - } - - private String getInheritanceCode() { - String inheritanceCode = ""; - if (this.getSuperclass() != null) { - inheritanceCode += "extends " + this.getSuperclass() + " "; - } - if (!this.getInterfaces().isEmpty()) { - inheritanceCode += "implements " + String.join(", ", this.getInterfaces()) + " "; - } - return inheritanceCode; - } - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - - public String getPackageName() { - return packageName; - } - - public void setPackageName(String packageName) { - this.packageName = packageName; - } - - public String getSuperclass() { - return superclass; - } - - public void setSuperclass(String superclass) { - this.superclass = superclass; - } - - public boolean isInterface() { - return isInterface; - } - - public void setIsInterface(boolean anInterface) { - isInterface = anInterface; - } - - public boolean isEnum() { - return isEnum; - } - - public void setIsEnum(boolean anEnum) { - isEnum = anEnum; - } - - public List getInterfaces() { - return interfaces; - } - - public void setInterfaces(List interfaces) { - this.interfaces = interfaces; - } - - public List getModifiers() { - return modifiers; - } - - public void setModifiers(List modifiers) { - this.modifiers = modifiers; - } - - public List getAnnotations() { - return annotations; - } - - public void setAnnotations(List annotations) { - this.annotations = annotations; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralClassElements.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralClassElements.java deleted file mode 100644 index a5d6535b5090..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralClassElements.java +++ /dev/null @@ -1,66 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.structural; - -import java.util.ArrayList; -import java.util.List; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * Root element of the test.json file - * Used for the generation of solution entries for structural test cases - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class StructuralClassElements { - - @JsonProperty(value = "class", required = true) - private StructuralClass structuralClass; - - private List methods = new ArrayList<>(); - - private List attributes = new ArrayList<>(); - - private List constructors = new ArrayList<>(); - - private List enumValues = new ArrayList<>(); - - public StructuralClass getStructuralClass() { - return structuralClass; - } - - public void setStructuralClass(StructuralClass structuralClass) { - this.structuralClass = structuralClass; - } - - public List getMethods() { - return methods; - } - - public void setMethods(List methods) { - this.methods = methods; - } - - public List getAttributes() { - return attributes; - } - - public void setAttributes(List attributes) { - this.attributes = attributes; - } - - public List getConstructors() { - return constructors; - } - - public void setConstructors(List constructors) { - this.constructors = constructors; - } - - public List getEnumValues() { - return enumValues; - } - - public void setEnumValues(List enumValues) { - this.enumValues = enumValues; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralConstructor.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralConstructor.java deleted file mode 100644 index 6f01793c4595..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralConstructor.java +++ /dev/null @@ -1,77 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.structural; - -import java.util.ArrayList; -import java.util.List; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.thoughtworks.qdox.model.JavaClass; -import com.thoughtworks.qdox.model.JavaConstructor; - -/** - * Element of the test.json file representing the properties of a constructor of a class - * Used for the generation of solution entries for structural test cases - */ -@JsonIgnoreProperties(ignoreUnknown = true) -class StructuralConstructor implements StructuralElement { - - private List modifiers = new ArrayList<>(); - - private List parameters = new ArrayList<>(); - - private List annotations = new ArrayList<>(); - - @Override - public String getSourceCode(StructuralClassElements structuralClassElements, JavaClass solutionClass) { - JavaConstructor solutionConstructor = getSolutionConstructor(solutionClass); - String constructorSolutionCode = ""; - if (!this.getAnnotations().isEmpty()) { - constructorSolutionCode += getAnnotationsString(this.getAnnotations(), solutionConstructor); - } - if (!this.getModifiers().isEmpty()) { - constructorSolutionCode += formatModifiers(this.getModifiers()) + " "; - } - constructorSolutionCode += structuralClassElements.getStructuralClass().getName(); - constructorSolutionCode += generateParametersString(this.getParameters(), solutionConstructor) + " "; - constructorSolutionCode += "{\n" + SINGLE_INDENTATION + "\n}"; - return constructorSolutionCode; - } - - /** - * Extracts the parameters from a constructor - * - * @param solutionClass The QDox class instance - * @return The parameters of the constructor - */ - private JavaConstructor getSolutionConstructor(JavaClass solutionClass) { - if (solutionClass == null) { - return null; - } - return solutionClass.getConstructors().stream() - .filter(javaConstructor -> doParametersMatch(this.getParameters(), javaConstructor.getParameters(), solutionClass.getTypeParameters())).findFirst().stream() - .findFirst().orElse(null); - } - - public List getModifiers() { - return modifiers; - } - - public void setModifiers(List modifiers) { - this.modifiers = modifiers; - } - - public List getParameters() { - return parameters; - } - - public void setParameters(List parameters) { - this.parameters = parameters; - } - - public List getAnnotations() { - return annotations; - } - - public void setAnnotations(List annotations) { - this.annotations = annotations; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralElement.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralElement.java deleted file mode 100644 index c62e54e7c02b..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralElement.java +++ /dev/null @@ -1,135 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.structural; - -import java.util.Collections; -import java.util.HashSet; -import java.util.List; -import java.util.Set; -import java.util.stream.Collectors; - -import com.thoughtworks.qdox.model.JavaAnnotatedElement; -import com.thoughtworks.qdox.model.JavaClass; -import com.thoughtworks.qdox.model.JavaExecutable; -import com.thoughtworks.qdox.model.JavaGenericDeclaration; -import com.thoughtworks.qdox.model.JavaParameter; -import com.thoughtworks.qdox.model.JavaType; -import com.thoughtworks.qdox.model.JavaTypeVariable; - -public interface StructuralElement { - - String SINGLE_INDENTATION = " "; - - /** - * Generates well formatted Java code for a structural element - * - * @param structuralClassElements The elements of the class from the test.json - * @param solutionClass The class read by QDox - * @return The code for the element - */ - String getSourceCode(StructuralClassElements structuralClassElements, JavaClass solutionClass); - - /** - * Generates the code for annotations that are required by the test.json file. - * Annotations that are present in the source code, but not in the test.json file will be excluded. - * - * @param structuralAnnotations The annotation names from the test.json - * @param annotatedElement The annotated element (e.g. method) read by QDox - * @return The code for the annotations - */ - default String getAnnotationsString(List structuralAnnotations, JavaAnnotatedElement annotatedElement) { - if (annotatedElement == null) { - return String.join("\n", structuralAnnotations) + "\n"; - } - else { - Set structuralAnnotationsSet = new HashSet<>(structuralAnnotations); - return annotatedElement.getAnnotations().stream() - .filter(solutionAnnotation -> structuralAnnotationsSet.contains(solutionAnnotation.getType().getSimpleName()) - || "Override".equals(solutionAnnotation.getType().getSimpleName())) - .map(annotation -> annotation.getCodeBlock().replace(annotation.getType().getGenericCanonicalName(), annotation.getType().getSimpleName())) - .collect(Collectors.joining()); - } - } - - /** - * Creates the String representation of a generics declaration - * - * @param typeParameters The generic type parameters - * @return The String representation - */ - default String getGenericTypesString(List> typeParameters) { - return "<" + typeParameters.stream().map(JavaType::getGenericValue).map(type -> type.substring(1, type.length() - 1)).collect(Collectors.joining(", ")) + ">"; - } - - /** - * Formats the modifiers properly. - * Currently, it only removes the 'optional: ' tags and joins them together. - * - * @param modifiers The modifiers array - * @return The formatted modifiers - */ - default String formatModifiers(List modifiers) { - if (modifiers == null) { - return ""; - } - return modifiers.stream().map(modifier -> modifier.replace("optional: ", "")).collect(Collectors.joining(" ")); - } - - /** - * Checks if the parameters from the source files and those from the test.json match. - * This is used for methods and constructor parameters. - * Contains special handling for generics - * - * @param parameters The parameters from the test.json file - * @param solutionParameters The parameters from the source code - * @param genericDeclarations The current generic declarations - * @return false if any parameter does not match - */ - default boolean doParametersMatch(List parameters, List solutionParameters, List> genericDeclarations) { - if (parameters == null) { - return solutionParameters.isEmpty(); - } - if (parameters.size() != solutionParameters.size()) { - return false; - } - for (int i = 0; i < parameters.size(); i++) { - var typeMatches = parameters.get(i).equals(solutionParameters.get(i).getType().getValue()); - var isGeneric = false; - for (JavaTypeVariable type : genericDeclarations) { - if (type.getName().equals(solutionParameters.get(i).getType().getValue()) || (type.getName() + "[]").equals(solutionParameters.get(i).getType().getValue())) { - isGeneric = true; - break; - } - } - if (!typeMatches && !isGeneric) { - return false; - } - } - return true; - } - - /** - * Generates the string representing the source code of a parameter list - * - * @param parameterTypes The parameters from the test.json file - * @param javaExecutable The executable (e.g. method) read by QDox to take the parameters from - * @return The parameter source code - */ - default String generateParametersString(List parameterTypes, JavaExecutable javaExecutable) { - List solutionParameters = javaExecutable != null ? javaExecutable.getParameters() : Collections.emptyList(); - String result = "("; - if (parameterTypes != null) { - for (int i = 0; i < parameterTypes.size(); i++) { - if (solutionParameters.size() > i) { - // Use original parameter names - parameterTypes.set(i, solutionParameters.get(i).getType().getGenericValue() + " " + solutionParameters.get(i).getName()); - } - else { - // Use var[i] as a fallback - parameterTypes.set(i, parameterTypes.get(i) + " var" + i); - } - } - result += String.join(", ", parameterTypes); - } - result += ")"; - return result; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralMethod.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralMethod.java deleted file mode 100644 index 66fa6eed3bee..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralMethod.java +++ /dev/null @@ -1,140 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.structural; - -import java.util.ArrayList; -import java.util.List; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.google.common.collect.Lists; -import com.thoughtworks.qdox.model.JavaClass; -import com.thoughtworks.qdox.model.JavaMethod; - -/** - * Element of the test.json file representing the properties of a method of a class - * Used for the generation of solution entries for structural test cases - */ -@JsonIgnoreProperties(ignoreUnknown = true) -class StructuralMethod implements StructuralElement { - - @JsonProperty(required = true) - private String name; - - private List modifiers = new ArrayList<>(); - - private List parameters = new ArrayList<>(); - - private List annotations = new ArrayList<>(); - - @JsonProperty(defaultValue = "void") - private String returnType; - - @Override - public String getSourceCode(StructuralClassElements structuralClassElements, JavaClass solutionClass) { - JavaMethod solutionMethod = getSolutionMethod(solutionClass); - String methodSolutionCode = ""; - boolean isAbstract = this.getModifiers().contains("abstract"); - - if (!this.getAnnotations().isEmpty()) { - methodSolutionCode += getAnnotationsString(this.getAnnotations(), solutionMethod); - } - - methodSolutionCode += formatModifiers(structuralClassElements, isAbstract); - - // Generics - if (solutionMethod != null && !solutionMethod.getTypeParameters().isEmpty()) { - methodSolutionCode += getGenericTypesString(solutionMethod.getTypeParameters()) + " "; - } - - // Return type - methodSolutionCode += solutionMethod != null ? solutionMethod.getReturnType().getGenericValue() + " " : this.getReturnType() + " "; - // Name - methodSolutionCode += this.getName(); - // Parameters - methodSolutionCode += generateParametersString(this.getParameters(), solutionMethod); - // Body - methodSolutionCode += isAbstract ? ";" : " {\n" + SINGLE_INDENTATION + "\n}"; - - return methodSolutionCode; - } - - /** - * Formats the modifiers of this method. - * - * @param structuralClassElements The elements of the class from the test.json - * @param isAbstract Whether the method is abstract - * @return The modifiers as Java code - */ - private String formatModifiers(StructuralClassElements structuralClassElements, boolean isAbstract) { - var modifiers = Lists.newArrayList(this.getModifiers()); - // Adjust modifiers for interfaces - if (structuralClassElements.getStructuralClass().isInterface()) { - if (isAbstract) { - modifiers.remove("abstract"); - } - else { - modifiers.addFirst("default"); - } - } - if (!modifiers.isEmpty()) { - return formatModifiers(modifiers) + " "; - } - return ""; - } - - /** - * Finds the QDox method in a given class by its test.json specification - * - * @param solutionClass The QDox class instance - * @return The QDox method instance or null if not found - */ - private JavaMethod getSolutionMethod(JavaClass solutionClass) { - if (solutionClass == null) { - return null; - } - return solutionClass.getMethods().stream().filter(javaMethod -> javaMethod.getName().equals(this.getName())).filter(javaMethod -> { - var genericTypes = new ArrayList<>(solutionClass.getTypeParameters()); - genericTypes.addAll(javaMethod.getTypeParameters()); - return doParametersMatch(this.getParameters(), javaMethod.getParameters(), genericTypes); - }).findFirst().orElse(null); - } - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - - public List getModifiers() { - return modifiers; - } - - public void setModifiers(List modifiers) { - this.modifiers = modifiers; - } - - public List getParameters() { - return parameters; - } - - public void setParameters(List parameters) { - this.parameters = parameters; - } - - public List getAnnotations() { - return annotations; - } - - public void setAnnotations(List annotations) { - this.annotations = annotations; - } - - public String getReturnType() { - return returnType; - } - - public void setReturnType(String returnType) { - this.returnType = returnType; - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralSolutionEntryGenerationException.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralSolutionEntryGenerationException.java deleted file mode 100644 index 3b9614b9e2aa..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralSolutionEntryGenerationException.java +++ /dev/null @@ -1,18 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.structural; - -/** - * Exception used for the generation of solution entries for structural test cases - * It is thrown if there was an error while generating the solution entries - */ -public class StructuralSolutionEntryGenerationException extends Exception { - - private static final String MESSAGE_PREFIX = "Unable to generate structural solution entries: "; - - public StructuralSolutionEntryGenerationException(String message) { - super(MESSAGE_PREFIX + message); - } - - public StructuralSolutionEntryGenerationException(String message, Throwable cause) { - super(MESSAGE_PREFIX + message, cause); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralTestCaseService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralTestCaseService.java deleted file mode 100644 index 889405573cff..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/hestia/structural/StructuralTestCaseService.java +++ /dev/null @@ -1,254 +0,0 @@ -package de.tum.cit.aet.artemis.programming.service.hestia.structural; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.io.IOException; -import java.net.URISyntaxException; -import java.nio.file.FileSystems; -import java.nio.file.Files; -import java.nio.file.Path; -import java.util.Arrays; -import java.util.Collections; -import java.util.List; -import java.util.Map; -import java.util.Objects; -import java.util.Optional; -import java.util.Set; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -import org.eclipse.jgit.api.errors.GitAPIException; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Profile; -import org.springframework.stereotype.Service; - -import com.fasterxml.jackson.databind.ObjectMapper; -import com.thoughtworks.qdox.JavaProjectBuilder; -import com.thoughtworks.qdox.model.JavaClass; -import com.thoughtworks.qdox.model.JavaType; - -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.Repository; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTestCaseType; -import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; -import de.tum.cit.aet.artemis.programming.repository.SolutionProgrammingExerciseParticipationRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseSolutionEntryRepository; -import de.tum.cit.aet.artemis.programming.service.GitService; - -/** - * Service for handling Solution Entries of structural Test Cases. - */ -@Profile(PROFILE_CORE) -@Service -public class StructuralTestCaseService { - - private static final Logger log = LoggerFactory.getLogger(StructuralTestCaseService.class); - - private final GitService gitService; - - private final ProgrammingExerciseTestCaseRepository testCaseRepository; - - private final ProgrammingExerciseSolutionEntryRepository solutionEntryRepository; - - private final SolutionProgrammingExerciseParticipationRepository solutionProgrammingExerciseParticipationRepository; - - public StructuralTestCaseService(GitService gitService, ProgrammingExerciseTestCaseRepository testCaseRepository, - ProgrammingExerciseSolutionEntryRepository solutionEntryRepository, - SolutionProgrammingExerciseParticipationRepository solutionProgrammingExerciseParticipationRepository) { - this.gitService = gitService; - this.testCaseRepository = testCaseRepository; - this.solutionEntryRepository = solutionEntryRepository; - this.solutionProgrammingExerciseParticipationRepository = solutionProgrammingExerciseParticipationRepository; - } - - /** - * Generates the solution entries for all structural test cases of a programming exercise. - * This includes solution entries for classes, attributes, methods and constructors. - * - * @param programmingExercise The programming exercise - * @return an unmodifiable list of new structural solution entries - * @throws StructuralSolutionEntryGenerationException If there was an error while generating the solution entries - */ - public List generateStructuralSolutionEntries(ProgrammingExercise programmingExercise) throws StructuralSolutionEntryGenerationException { - log.debug("Generating the structural SolutionEntries for the following programmingExercise: {} {}", programmingExercise.getId(), programmingExercise.getProjectName()); - - var testCases = testCaseRepository.findByExerciseIdWithSolutionEntriesAndActive(programmingExercise.getId(), true); - testCases.removeIf(testCase -> testCase.getType() != ProgrammingExerciseTestCaseType.STRUCTURAL); - - // No test cases = no solution entries needed - if (testCases.isEmpty()) { - return Collections.emptyList(); - } - - // Checkout the solution and test repositories - Repository solutionRepository; - Repository testRepository; - try { - var solutionParticipation = solutionProgrammingExerciseParticipationRepository.findByProgrammingExerciseId(programmingExercise.getId()); - if (solutionParticipation.isEmpty()) { - return Collections.emptyList(); - } - solutionRepository = gitService.getOrCheckoutRepository(solutionParticipation.get().getVcsRepositoryUri(), true); - testRepository = gitService.getOrCheckoutRepository(programmingExercise.getVcsTestRepositoryUri(), true); - - gitService.resetToOriginHead(solutionRepository); - gitService.pullIgnoreConflicts(solutionRepository); - gitService.resetToOriginHead(testRepository); - gitService.pullIgnoreConflicts(testRepository); - } - catch (GitAPIException e) { - var error = "Error while checking out repositories"; - log.error(error, e); - throw new StructuralSolutionEntryGenerationException(error, e); - } - - var classElements = readStructureOracleFile(testRepository.getLocalPath()); - var solutionClasses = getClassesFromFiles(retrieveJavaSourceFiles(solutionRepository.getLocalPath())); - - // Create new solution entries - List newSolutionEntries = generateStructuralSolutionEntries(testCases, solutionRepository, classElements, solutionClasses); - - // Get all old solution entries - var oldSolutionEntries = newSolutionEntries.stream().map(ProgrammingExerciseSolutionEntry::getTestCase).flatMap(testCase -> testCase.getSolutionEntries().stream()) - .distinct().toList(); - - // Save new solution entries - newSolutionEntries = solutionEntryRepository.saveAll(newSolutionEntries); - - // Delete old solution entries - solutionEntryRepository.deleteAll(oldSolutionEntries); - - return newSolutionEntries; - } - - /** - * Private method that takes care of the actual generation of structural solution entries. - * - * @param testCases The test cases of the programming exercise - * @param solutionRepository The solution repository of the programming exercise - * @param classElements The entries from the test.json file - * @param solutionClasses The classes read with QDox - * @return an unmodifiable list of new structural solution entries - */ - private List generateStructuralSolutionEntries(Set testCases, Repository solutionRepository, - StructuralClassElements[] classElements, Map solutionClasses) { - return Arrays.stream(classElements).flatMap(classElement -> { - var packageName = classElement.getStructuralClass().getPackageName(); - var name = classElement.getStructuralClass().getName(); - var solutionClass = solutionClasses.get(packageName + "." + name); - String filePath = "src/" + packageName.replaceAll("\\.", "/") + "/" + name + ".java"; - if (solutionClass != null) { - try { - filePath = solutionRepository.getLocalPath().toAbsolutePath().toUri().relativize(solutionClass.getSource().getURL().toURI()).toString(); - } - catch (URISyntaxException e) { - log.warn("Unable to create file path for class", e); - } - } - - String classSolutionCode = classElement.getStructuralClass().getSourceCode(classElement, solutionClass); - List constructorsSolutionCode = classElement.getConstructors().stream().map(constructor -> constructor.getSourceCode(classElement, solutionClass)).toList(); - List methodsSolutionCode = classElement.getMethods().stream().map(method -> method.getSourceCode(classElement, solutionClass)).toList(); - List attributesSolutionCode = classElement.getAttributes().stream().map(attribute -> attribute.getSourceCode(classElement, solutionClass)).toList(); - return Stream.of(createSolutionEntry(filePath, classSolutionCode, findStructuralTestCase("Class", name, testCases)), - createSolutionEntry(filePath, String.join("\n\n", attributesSolutionCode), findStructuralTestCase("Attributes", name, testCases)), - createSolutionEntry(filePath, String.join("\n\n", constructorsSolutionCode), findStructuralTestCase("Constructors", name, testCases)), - createSolutionEntry(filePath, String.join("\n\n", methodsSolutionCode), findStructuralTestCase("Methods", name, testCases))); - }).filter(Objects::nonNull).toList(); - } - - /** - * Finds a structural test case of a specific type and class in the list of all test cases of an exercise. - * - * @param type The type of the structural test case (e.g. Class, Methods) - * @param className The name of the Class - * @param testCases The list of test cases - * @return The matching test case or empty in none found - */ - private Optional findStructuralTestCase(String type, String className, Set testCases) { - return testCases.stream().filter(testCase -> testCase.getTestName().equals("test" + type + "[" + className + "]")).findFirst(); - } - - /** - * Helper method for creating a solution entry. - * If the given test case is not present this will return null. - * - * @param filePath The filePath of the solution entry - * @param code The code of the solution entry - * @param testCase The test case of the solution entry - * @return A SolutionEntry if testCase is present otherwise null - */ - private ProgrammingExerciseSolutionEntry createSolutionEntry(String filePath, String code, Optional testCase) { - return testCase.map(actualTestCase -> { - var solutionEntry = new ProgrammingExerciseSolutionEntry(); - solutionEntry.setFilePath(filePath); - solutionEntry.setPreviousLine(null); - solutionEntry.setPreviousCode(null); - solutionEntry.setLine(1); - solutionEntry.setCode(code); - solutionEntry.setTestCase(actualTestCase); - return solutionEntry; - }).orElse(null); - } - - /** - * Finds, reads and parses the test.json file from the test repository - * - * @param testRepoPath The base path of the test repository - * @return The parsed test.json file - * @throws StructuralSolutionEntryGenerationException If the test.json does not exist or could not be read - */ - private StructuralClassElements[] readStructureOracleFile(Path testRepoPath) throws StructuralSolutionEntryGenerationException { - try (Stream files = Files.walk(testRepoPath)) { - var testJsonFile = files.filter(Files::isRegularFile).filter(path -> "test.json".equals(path.getFileName().toString())).findFirst(); - if (testJsonFile.isEmpty()) { - throw new StructuralSolutionEntryGenerationException("Unable to locate test.json"); - } - else { - String jsonContent = Files.readString(testJsonFile.get()); - var objectMapper = new ObjectMapper(); - return objectMapper.readValue(jsonContent, StructuralClassElements[].class); - } - } - catch (IOException e) { - throw new StructuralSolutionEntryGenerationException("Error while reading test.json", e); - } - } - - /** - * Collects all java source files in a given path. - * - * @param start The base path - * @return The paths to all java source files - * @throws StructuralSolutionEntryGenerationException If there was an IOException - */ - private List retrieveJavaSourceFiles(Path start) throws StructuralSolutionEntryGenerationException { - var matcher = FileSystems.getDefault().getPathMatcher("glob:**/*.java"); - try (Stream files = Files.walk(start)) { - return files.filter(Files::isRegularFile).filter(matcher::matches).toList(); - } - catch (IOException e) { - var error = "Could not retrieve the project files to generate the structural solution entries"; - log.error(error, e); - throw new StructuralSolutionEntryGenerationException(error, e); - } - } - - private Map getClassesFromFiles(List javaSourceFiles) throws StructuralSolutionEntryGenerationException { - JavaProjectBuilder builder = new JavaProjectBuilder(); - try { - for (Path source : javaSourceFiles) { - builder.addSource(source.toFile()); - } - } - catch (IOException e) { - var error = "Could not add java source to builder"; - log.error(error, e); - throw new StructuralSolutionEntryGenerationException(error, e); - } - return builder.getClasses().stream().collect(Collectors.toMap(JavaType::getFullyQualifiedName, clazz -> clazz)); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsProgrammingLanguageFeatureService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsProgrammingLanguageFeatureService.java index cca2995cdba0..6ad7b4c65aec 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsProgrammingLanguageFeatureService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsProgrammingLanguageFeatureService.java @@ -37,21 +37,21 @@ public class JenkinsProgrammingLanguageFeatureService extends ProgrammingLanguag public JenkinsProgrammingLanguageFeatureService() { // Must be extended once a new programming language is added - programmingLanguageFeatures.put(EMPTY, new ProgrammingLanguageFeature(EMPTY, false, false, false, false, false, List.of(), false, false)); - programmingLanguageFeatures.put(C, new ProgrammingLanguageFeature(C, false, false, true, false, false, List.of(FACT, GCC), false, false)); - programmingLanguageFeatures.put(C_PLUS_PLUS, new ProgrammingLanguageFeature(C_PLUS_PLUS, false, false, true, false, false, List.of(), false, false)); - programmingLanguageFeatures.put(C_SHARP, new ProgrammingLanguageFeature(C_SHARP, false, false, true, false, false, List.of(), false, false)); - programmingLanguageFeatures.put(GO, new ProgrammingLanguageFeature(GO, false, false, true, true, false, List.of(), false, false)); - programmingLanguageFeatures.put(HASKELL, new ProgrammingLanguageFeature(HASKELL, false, false, false, false, true, List.of(), false, false)); + programmingLanguageFeatures.put(EMPTY, new ProgrammingLanguageFeature(EMPTY, false, false, false, false, false, List.of(), false)); + programmingLanguageFeatures.put(C, new ProgrammingLanguageFeature(C, false, false, true, false, false, List.of(FACT, GCC), false)); + programmingLanguageFeatures.put(C_PLUS_PLUS, new ProgrammingLanguageFeature(C_PLUS_PLUS, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(C_SHARP, new ProgrammingLanguageFeature(C_SHARP, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(GO, new ProgrammingLanguageFeature(GO, false, false, true, true, false, List.of(), false)); + programmingLanguageFeatures.put(HASKELL, new ProgrammingLanguageFeature(HASKELL, false, false, false, false, true, List.of(), false)); programmingLanguageFeatures.put(JAVA, - new ProgrammingLanguageFeature(JAVA, true, true, true, true, false, List.of(PLAIN_GRADLE, GRADLE_GRADLE, PLAIN_MAVEN, MAVEN_MAVEN, MAVEN_BLACKBOX), true, false)); - programmingLanguageFeatures.put(JAVASCRIPT, new ProgrammingLanguageFeature(JAVASCRIPT, false, false, true, false, false, List.of(), false, false)); - programmingLanguageFeatures.put(KOTLIN, new ProgrammingLanguageFeature(KOTLIN, true, false, true, true, false, List.of(), true, false)); - programmingLanguageFeatures.put(PYTHON, new ProgrammingLanguageFeature(PYTHON, false, false, true, false, false, List.of(), false, false)); - programmingLanguageFeatures.put(R, new ProgrammingLanguageFeature(R, false, false, true, false, false, List.of(), false, false)); - programmingLanguageFeatures.put(RUST, new ProgrammingLanguageFeature(RUST, false, false, true, false, false, List.of(), false, false)); + new ProgrammingLanguageFeature(JAVA, true, true, true, true, false, List.of(PLAIN_GRADLE, GRADLE_GRADLE, PLAIN_MAVEN, MAVEN_MAVEN, MAVEN_BLACKBOX), true)); + programmingLanguageFeatures.put(JAVASCRIPT, new ProgrammingLanguageFeature(JAVASCRIPT, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(KOTLIN, new ProgrammingLanguageFeature(KOTLIN, true, false, true, true, false, List.of(), true)); + programmingLanguageFeatures.put(PYTHON, new ProgrammingLanguageFeature(PYTHON, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(R, new ProgrammingLanguageFeature(R, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(RUST, new ProgrammingLanguageFeature(RUST, false, false, true, false, false, List.of(), false)); // Jenkins is not supporting XCODE at the moment - programmingLanguageFeatures.put(SWIFT, new ProgrammingLanguageFeature(SWIFT, false, true, true, true, false, List.of(PLAIN), false, false)); - programmingLanguageFeatures.put(TYPESCRIPT, new ProgrammingLanguageFeature(TYPESCRIPT, false, false, true, false, false, List.of(), false, false)); + programmingLanguageFeatures.put(SWIFT, new ProgrammingLanguageFeature(SWIFT, false, true, true, true, false, List.of(PLAIN), false)); + programmingLanguageFeatures.put(TYPESCRIPT, new ProgrammingLanguageFeature(TYPESCRIPT, false, false, true, false, false, List.of(), false)); } } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsResultService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsResultService.java index 9c27aa4c5733..2730c9e82f9a 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsResultService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsResultService.java @@ -10,21 +10,17 @@ import org.springframework.context.annotation.Profile; import org.springframework.stereotype.Service; -import de.tum.cit.aet.artemis.assessment.repository.FeedbackRepository; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; import de.tum.cit.aet.artemis.programming.domain.ProjectType; import de.tum.cit.aet.artemis.programming.domain.build.BuildLogEntry; -import de.tum.cit.aet.artemis.programming.dto.AbstractBuildResultNotificationDTO; +import de.tum.cit.aet.artemis.programming.dto.BuildResultNotification; import de.tum.cit.aet.artemis.programming.repository.BuildLogStatisticsEntryRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseBuildConfigRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; -import de.tum.cit.aet.artemis.programming.repository.ProgrammingSubmissionRepository; -import de.tum.cit.aet.artemis.programming.service.BuildLogEntryService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseFeedbackCreationService; import de.tum.cit.aet.artemis.programming.service.ci.AbstractContinuousIntegrationResultService; import de.tum.cit.aet.artemis.programming.service.ci.notification.dto.TestResultsDTO; -import de.tum.cit.aet.artemis.programming.service.hestia.TestwiseCoverageService; @Profile("jenkins") @Service @@ -32,15 +28,13 @@ public class JenkinsResultService extends AbstractContinuousIntegrationResultSer private static final Logger log = LoggerFactory.getLogger(JenkinsResultService.class); - public JenkinsResultService(ProgrammingSubmissionRepository programmingSubmissionRepository, FeedbackRepository feedbackRepository, BuildLogEntryService buildLogService, - BuildLogStatisticsEntryRepository buildLogStatisticsEntryRepository, TestwiseCoverageService testwiseCoverageService, - ProgrammingExerciseFeedbackCreationService feedbackCreationService, ProgrammingExerciseTestCaseRepository testCaseRepository, - ProgrammingExerciseBuildConfigRepository programmingExerciseBuildConfigRepository) { - super(testCaseRepository, buildLogStatisticsEntryRepository, testwiseCoverageService, feedbackCreationService, programmingExerciseBuildConfigRepository); + public JenkinsResultService(BuildLogStatisticsEntryRepository buildLogStatisticsEntryRepository, ProgrammingExerciseFeedbackCreationService feedbackCreationService, + ProgrammingExerciseTestCaseRepository testCaseRepository, ProgrammingExerciseBuildConfigRepository programmingExerciseBuildConfigRepository) { + super(testCaseRepository, buildLogStatisticsEntryRepository, feedbackCreationService, programmingExerciseBuildConfigRepository); } @Override - public AbstractBuildResultNotificationDTO convertBuildResult(Object requestBody) { + public BuildResultNotification convertBuildResult(Object requestBody) { return TestResultsDTO.convert(requestBody); } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsService.java index 4a4e42f602a4..887bef4857b9 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsService.java @@ -30,9 +30,9 @@ import de.tum.cit.aet.artemis.programming.domain.VcsRepositoryUri; import de.tum.cit.aet.artemis.programming.domain.build.BuildPlanType; import de.tum.cit.aet.artemis.programming.dto.CheckoutDirectoriesDTO; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseBuildConfigRepository; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusTemplateService; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; import de.tum.cit.aet.artemis.programming.service.ci.AbstractContinuousIntegrationService; import de.tum.cit.aet.artemis.programming.service.ci.CIPermission; import de.tum.cit.aet.artemis.programming.service.ci.notification.dto.TestResultsDTO; diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/build_plan/JenkinsBuildPlanService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/build_plan/JenkinsBuildPlanService.java index 1264ba9b643f..c54b617d5546 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/build_plan/JenkinsBuildPlanService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/build_plan/JenkinsBuildPlanService.java @@ -48,12 +48,13 @@ import de.tum.cit.aet.artemis.programming.domain.ProjectType; import de.tum.cit.aet.artemis.programming.domain.RepositoryType; import de.tum.cit.aet.artemis.programming.domain.VcsRepositoryUri; +import de.tum.cit.aet.artemis.programming.dto.aeolus.AeolusRepository; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; +import de.tum.cit.aet.artemis.programming.dto.aeolus.WindfileMetadata; import de.tum.cit.aet.artemis.programming.repository.BuildPlanRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseBuildConfigRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusBuildPlanService; -import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusRepository; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationService; import de.tum.cit.aet.artemis.programming.service.ci.notification.dto.TestResultsDTO; import de.tum.cit.aet.artemis.programming.service.jenkins.JenkinsEndpoints; @@ -287,7 +288,7 @@ private void postBuildPlanConfigChange(String buildPlanKey, String buildProjectK * @return the build plan key */ public String getBuildPlanKeyFromTestResults(TestResultsDTO testResultsDTO) throws JsonProcessingException { - final var nameParams = testResultsDTO.getFullName().split(" "); + final var nameParams = testResultsDTO.fullName().split(" "); /* * Jenkins gives the full name of a job as » E.g. the third build of an exercise (projectKey = TESTEXC) for its solution build * (TESTEXC-SOLUTION) would be: TESTEXC » TESTEXC-SOLUTION #3 ==> This would mean that at index 2, we have the actual job/plan key, i.e. TESTEXC-SOLUTION @@ -507,8 +508,9 @@ private String createCustomAeolusBuildPlanForExercise(ProgrammingExercise progra buildConfig.getBranch(), buildConfig.getCheckoutSolutionRepository(), repositoryUri, testRepositoryUri, solutionRepositoryUri, List.of()); String resultHookUrl = artemisServerUrl + NEW_RESULT_RESOURCE_API_PATH; - windfile.setPreProcessingMetadata(buildPlanId, programmingExercise.getProjectName(), this.vcsCredentials, resultHookUrl, "planDescription", repositories, - this.artemisAuthenticationTokenKey); + var metadata = new WindfileMetadata(programmingExercise.getProjectName(), buildPlanId, "planDescription", null, vcsCredentials, null, resultHookUrl, + artemisAuthenticationTokenKey); + windfile = new Windfile(windfile, metadata, repositories); String generatedKey = aeolusBuildPlanService.get().publishBuildPlan(windfile, AeolusTarget.JENKINS); if (generatedKey != null && generatedKey.contains("-")) { diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/build_plan/JenkinsPipelineScriptCreator.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/build_plan/JenkinsPipelineScriptCreator.java index 92fca529c68b..a18f3ea8ffc5 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/build_plan/JenkinsPipelineScriptCreator.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/build_plan/JenkinsPipelineScriptCreator.java @@ -36,8 +36,6 @@ public class JenkinsPipelineScriptCreator extends AbstractBuildPlanCreator { private static final String REPLACE_IS_STATIC_CODE_ANALYSIS_ENABLED = "#isStaticCodeAnalysisEnabled"; - private static final String REPLACE_TESTWISE_COVERAGE = "#testWiseCoverage"; - private final ResourceLoaderService resourceLoaderService; private final ProgrammingLanguageConfiguration programmingLanguageConfiguration; @@ -58,8 +56,7 @@ protected String generateDefaultBuildPlan(final ProgrammingExercise exercise) { final String pipelineScript = loadPipelineScript(exercise, projectType); final boolean isStaticCodeAnalysisEnabled = exercise.isStaticCodeAnalysisEnabled(); - final boolean isTestwiseCoverageAnalysisEnabled = exercise.getBuildConfig().isTestwiseCoverageEnabled(); - final var replacements = getReplacements(programmingLanguage, projectType, isStaticCodeAnalysisEnabled, isTestwiseCoverageAnalysisEnabled); + final var replacements = getReplacements(programmingLanguage, projectType, isStaticCodeAnalysisEnabled); return replaceVariablesInBuildPlanTemplate(replacements, pipelineScript); } @@ -86,12 +83,10 @@ private String loadPipelineScript(final ProgrammingExercise exercise, final Opti } } - private Map getReplacements(final ProgrammingLanguage programmingLanguage, final Optional projectType, final boolean isStaticCodeAnalysisEnabled, - final boolean isTestwiseCoverageAnalysisEnabled) { + private Map getReplacements(final ProgrammingLanguage programmingLanguage, final Optional projectType, final boolean isStaticCodeAnalysisEnabled) { final Map replacements = new HashMap<>(); replacements.put(REPLACE_IS_STATIC_CODE_ANALYSIS_ENABLED, String.valueOf(isStaticCodeAnalysisEnabled)); - replacements.put(REPLACE_TESTWISE_COVERAGE, String.valueOf(isTestwiseCoverageAnalysisEnabled)); replacements.put(REPLACE_DOCKER_IMAGE_NAME, programmingLanguageConfiguration.getImage(programmingLanguage, projectType)); replacements.put(REPLACE_DOCKER_ARGS, String.join(" ", programmingLanguageConfiguration.getDefaultDockerFlags())); diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIBuildConfigurationService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIBuildConfigurationService.java index e4803619c97d..498f416f61ed 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIBuildConfigurationService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIBuildConfigurationService.java @@ -11,10 +11,10 @@ import de.tum.cit.aet.artemis.core.exception.LocalCIException; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseBuildConfig; +import de.tum.cit.aet.artemis.programming.dto.aeolus.ScriptAction; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; import de.tum.cit.aet.artemis.programming.service.BuildScriptProviderService; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusTemplateService; -import de.tum.cit.aet.artemis.programming.service.aeolus.ScriptAction; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; @Service @Profile(PROFILE_LOCALCI) @@ -57,18 +57,18 @@ public String createBuildScript(ProgrammingExercise programmingExercise) { windfile = aeolusTemplateService.getDefaultWindfileFor(programmingExercise); } if (windfile != null) { - actions = windfile.getScriptActions(); + actions = windfile.scriptActions(); } else { throw new LocalCIException("No windfile found for programming exercise " + programmingExercise.getId()); } actions.forEach(action -> { - String workdir = action.getWorkdir(); + String workdir = action.workdir(); if (workdir != null) { buildScriptBuilder.append("cd ").append(LOCALCI_WORKING_DIRECTORY).append("/testing-dir/").append(workdir).append("\n"); } - buildScriptBuilder.append(action.getScript()).append("\n"); + buildScriptBuilder.append(action.script()).append("\n"); if (workdir != null) { buildScriptBuilder.append("cd ").append(LOCALCI_WORKING_DIRECTORY).append("/testing-dir\n"); } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIProgrammingLanguageFeatureService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIProgrammingLanguageFeatureService.java index d764193795c9..b8478cf65daf 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIProgrammingLanguageFeatureService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIProgrammingLanguageFeatureService.java @@ -43,23 +43,23 @@ public class LocalCIProgrammingLanguageFeatureService extends ProgrammingLanguag public LocalCIProgrammingLanguageFeatureService() { // Must be extended once a new programming language is added - programmingLanguageFeatures.put(EMPTY, new ProgrammingLanguageFeature(EMPTY, false, false, false, false, false, List.of(), false, true)); - programmingLanguageFeatures.put(ASSEMBLER, new ProgrammingLanguageFeature(ASSEMBLER, false, false, false, false, false, List.of(), false, true)); - programmingLanguageFeatures.put(C, new ProgrammingLanguageFeature(C, false, true, true, false, false, List.of(FACT, GCC), false, true)); - programmingLanguageFeatures.put(C_PLUS_PLUS, new ProgrammingLanguageFeature(C_PLUS_PLUS, false, false, true, false, false, List.of(), false, true)); - programmingLanguageFeatures.put(C_SHARP, new ProgrammingLanguageFeature(C_SHARP, false, false, true, false, false, List.of(), false, true)); - programmingLanguageFeatures.put(GO, new ProgrammingLanguageFeature(GO, false, false, true, true, false, List.of(), false, true)); - programmingLanguageFeatures.put(HASKELL, new ProgrammingLanguageFeature(HASKELL, true, false, false, false, true, List.of(), false, true)); + programmingLanguageFeatures.put(EMPTY, new ProgrammingLanguageFeature(EMPTY, false, false, false, false, false, List.of(), false)); + programmingLanguageFeatures.put(ASSEMBLER, new ProgrammingLanguageFeature(ASSEMBLER, false, false, false, false, false, List.of(), false)); + programmingLanguageFeatures.put(C, new ProgrammingLanguageFeature(C, false, true, true, false, false, List.of(FACT, GCC), false)); + programmingLanguageFeatures.put(C_PLUS_PLUS, new ProgrammingLanguageFeature(C_PLUS_PLUS, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(C_SHARP, new ProgrammingLanguageFeature(C_SHARP, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(GO, new ProgrammingLanguageFeature(GO, false, false, true, true, false, List.of(), false)); + programmingLanguageFeatures.put(HASKELL, new ProgrammingLanguageFeature(HASKELL, true, false, false, false, true, List.of(), false)); programmingLanguageFeatures.put(JAVA, - new ProgrammingLanguageFeature(JAVA, true, true, true, true, false, List.of(PLAIN_GRADLE, GRADLE_GRADLE, PLAIN_MAVEN, MAVEN_MAVEN), false, true)); - programmingLanguageFeatures.put(JAVASCRIPT, new ProgrammingLanguageFeature(JAVASCRIPT, false, false, true, false, false, List.of(), false, true)); - programmingLanguageFeatures.put(KOTLIN, new ProgrammingLanguageFeature(KOTLIN, false, false, true, true, false, List.of(), false, true)); - programmingLanguageFeatures.put(OCAML, new ProgrammingLanguageFeature(OCAML, false, false, false, false, true, List.of(), false, true)); - programmingLanguageFeatures.put(PYTHON, new ProgrammingLanguageFeature(PYTHON, false, true, true, false, false, List.of(), false, true)); - programmingLanguageFeatures.put(R, new ProgrammingLanguageFeature(R, false, false, true, false, false, List.of(), false, true)); - programmingLanguageFeatures.put(RUST, new ProgrammingLanguageFeature(RUST, false, false, true, false, false, List.of(), false, true)); - programmingLanguageFeatures.put(SWIFT, new ProgrammingLanguageFeature(SWIFT, false, false, true, true, false, List.of(PLAIN), false, true)); - programmingLanguageFeatures.put(TYPESCRIPT, new ProgrammingLanguageFeature(TYPESCRIPT, false, false, true, false, false, List.of(), false, true)); - programmingLanguageFeatures.put(VHDL, new ProgrammingLanguageFeature(VHDL, false, false, false, false, false, List.of(), false, true)); + new ProgrammingLanguageFeature(JAVA, true, true, true, true, false, List.of(PLAIN_GRADLE, GRADLE_GRADLE, PLAIN_MAVEN, MAVEN_MAVEN), false)); + programmingLanguageFeatures.put(JAVASCRIPT, new ProgrammingLanguageFeature(JAVASCRIPT, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(KOTLIN, new ProgrammingLanguageFeature(KOTLIN, false, false, true, true, false, List.of(), false)); + programmingLanguageFeatures.put(OCAML, new ProgrammingLanguageFeature(OCAML, false, false, false, false, true, List.of(), false)); + programmingLanguageFeatures.put(PYTHON, new ProgrammingLanguageFeature(PYTHON, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(R, new ProgrammingLanguageFeature(R, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(RUST, new ProgrammingLanguageFeature(RUST, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(SWIFT, new ProgrammingLanguageFeature(SWIFT, false, false, true, true, false, List.of(PLAIN), false)); + programmingLanguageFeatures.put(TYPESCRIPT, new ProgrammingLanguageFeature(TYPESCRIPT, false, false, true, false, false, List.of(), false)); + programmingLanguageFeatures.put(VHDL, new ProgrammingLanguageFeature(VHDL, false, false, false, false, false, List.of(), false)); } } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIQueueWebsocketService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIQueueWebsocketService.java index 872057e95145..f82777abddb2 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIQueueWebsocketService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIQueueWebsocketService.java @@ -182,8 +182,8 @@ private static List removeUnnecessaryInformation(List getTestResultPaths(Windfile windfile) throws IllegalArgumentException { - List testResultPaths = new ArrayList<>(); - for (AeolusResult testResultPath : windfile.getResults()) { - testResultPaths.add(LOCALCI_WORKING_DIRECTORY + "/testing-dir/" + testResultPath.path()); - } - return testResultPaths; + private List getTestResultPaths(Windfile windfile) { + return windfile.results().stream().map(result -> LOCALCI_WORKING_DIRECTORY + "/testing-dir/" + result.path()).toList(); } /** @@ -319,13 +313,12 @@ private BuildConfig getBuildConfig(ProgrammingExerciseParticipation participatio ProjectType projectType = programmingExercise.getProjectType(); boolean staticCodeAnalysisEnabled = programmingExercise.isStaticCodeAnalysisEnabled(); boolean sequentialTestRunsEnabled = buildConfig.hasSequentialTestRuns(); - boolean testwiseCoverageEnabled = buildConfig.isTestwiseCoverageEnabled(); Windfile windfile; String dockerImage; try { windfile = buildConfig.getWindfile(); - dockerImage = windfile.getMetadata().docker().getFullImageName(); + dockerImage = windfile.metadata().docker().getFullImageName(); } catch (NullPointerException e) { log.warn("Could not retrieve windfile for programming exercise {}. Using default windfile instead.", programmingExercise.getId()); @@ -343,8 +336,8 @@ private BuildConfig getBuildConfig(ProgrammingExerciseParticipation participatio String buildScript = localCIBuildConfigurationService.createBuildScript(programmingExercise); return new BuildConfig(buildScript, dockerImage, commitHashToBuild, assignmentCommitHash, testCommitHash, branch, programmingLanguage, projectType, - staticCodeAnalysisEnabled, sequentialTestRunsEnabled, testwiseCoverageEnabled, resultPaths, buildConfig.getTimeoutSeconds(), - buildConfig.getAssignmentCheckoutPath(), buildConfig.getTestCheckoutPath(), buildConfig.getSolutionCheckoutPath(), dockerRunConfig); + staticCodeAnalysisEnabled, sequentialTestRunsEnabled, resultPaths, buildConfig.getTimeoutSeconds(), buildConfig.getAssignmentCheckoutPath(), + buildConfig.getTestCheckoutPath(), buildConfig.getSolutionCheckoutPath(), dockerRunConfig); } private ProgrammingExerciseBuildConfig loadBuildConfig(ProgrammingExercise programmingExercise) { diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseExportImportResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseExportImportResource.java index da74833808c0..7079d2e2994e 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseExportImportResource.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseExportImportResource.java @@ -73,7 +73,7 @@ import de.tum.cit.aet.artemis.programming.domain.RepositoryType; import de.tum.cit.aet.artemis.programming.repository.AuxiliaryRepositoryRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseTaskRepository; +import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTaskRepository; import de.tum.cit.aet.artemis.programming.service.ConsistencyCheckService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseExportService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseImportFromFileService; @@ -214,8 +214,7 @@ public ResponseEntity importProgrammingExercise(@PathVariab programmingExerciseRepository.validateCourseSettings(newExercise, course); final var originalProgrammingExercise = programmingExerciseRepository - .findByIdWithEagerBuildConfigTestCasesStaticCodeAnalysisCategoriesHintsAndTemplateAndSolutionParticipationsAndAuxReposAndSolutionEntriesAndBuildConfig( - sourceExerciseId) + .findByIdWithEagerBuildConfigTestCasesStaticCodeAnalysisCategoriesAndTemplateAndSolutionParticipationsAndAuxReposAndAndBuildConfig(sourceExerciseId) .orElseThrow(() -> new EntityNotFoundException("ProgrammingExercise", sourceExerciseId)); var consistencyErrors = consistencyCheckService.checkConsistencyOfProgrammingExercise(originalProgrammingExercise); @@ -261,7 +260,6 @@ public ResponseEntity importProgrammingExercise(@PathVariab importedProgrammingExercise.setStaticCodeAnalysisCategories(null); importedProgrammingExercise.setTemplateParticipation(null); importedProgrammingExercise.setSolutionParticipation(null); - importedProgrammingExercise.setExerciseHints(null); importedProgrammingExercise.setTasks(null); competencyProgressApi.updateProgressByLearningObjectAsync(importedProgrammingExercise); diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ProgrammingExerciseGitDiffReportResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseGitDiffReportResource.java similarity index 98% rename from src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ProgrammingExerciseGitDiffReportResource.java rename to src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseGitDiffReportResource.java index 5dad0330b670..b561b6b331bb 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ProgrammingExerciseGitDiffReportResource.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseGitDiffReportResource.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.web.hestia; +package de.tum.cit.aet.artemis.programming.web; import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; @@ -25,15 +25,15 @@ import de.tum.cit.aet.artemis.exercise.repository.ParticipationRepository; import de.tum.cit.aet.artemis.exercise.service.ParticipationAuthorizationCheckService; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseGitDiffReport; import de.tum.cit.aet.artemis.programming.domain.RepositoryType; import de.tum.cit.aet.artemis.programming.domain.VcsRepositoryUri; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; import de.tum.cit.aet.artemis.programming.dto.ProgrammingExerciseGitDiffReportDTO; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingSubmissionRepository; import de.tum.cit.aet.artemis.programming.service.CommitHistoryService; +import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseGitDiffReportService; import de.tum.cit.aet.artemis.programming.service.RepositoryService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseGitDiffReportService; /** * REST controller for managing ProgrammingExerciseGitDiffReports and its entries. diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseResource.java index 0eaab82ce448..9d472c2cfbf1 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseResource.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseResource.java @@ -86,10 +86,10 @@ import de.tum.cit.aet.artemis.programming.service.GitService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseRepositoryService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseService; +import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseTaskService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseTestCaseService; import de.tum.cit.aet.artemis.programming.service.StaticCodeAnalysisService; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseTaskService; import de.tum.cit.aet.artemis.programming.service.vcs.VersionControlService; import io.jsonwebtoken.lang.Arrays; @@ -305,10 +305,6 @@ public ResponseEntity updateProgrammingExercise(@RequestBod if (!Objects.equals(programmingExerciseBeforeUpdate.isStaticCodeAnalysisEnabled(), updatedProgrammingExercise.isStaticCodeAnalysisEnabled())) { throw new BadRequestAlertException("Static code analysis enabled flag must not be changed", ENTITY_NAME, "staticCodeAnalysisCannotChange"); } - if (!Objects.equals(programmingExerciseBeforeUpdate.getBuildConfig().isTestwiseCoverageEnabled(), - updatedProgrammingExercise.getBuildConfig().isTestwiseCoverageEnabled())) { - throw new BadRequestAlertException("Testwise coverage enabled flag must not be changed", ENTITY_NAME, "testwiseCoverageCannotChange"); - } // Check if theia Profile is enabled if (Arrays.asList(this.environment.getActiveProfiles()).contains(PROFILE_THEIA)) { // Require 1 / 3 participation modes to be enabled @@ -791,9 +787,7 @@ public ResponseEntity reEvaluateAndUpdateProgrammingExercis } /** - * DELETE programming-exercises/:exerciseId/tasks : Delete all tasks and solution entries for an existing ProgrammingExercise. - * Note: This endpoint exists only for testing purposes and will be removed at a later stage of the development of HESTIA - * (automatic generation of code hints for programming exercises in Java). + * DELETE programming-exercises/:exerciseId/tasks : Delete all tasks for an existing ProgrammingExercise. * * @param exerciseId of the exercise * @return the {@link ResponseEntity} with status {@code 204}, @@ -802,12 +796,12 @@ public ResponseEntity reEvaluateAndUpdateProgrammingExercis @DeleteMapping("programming-exercises/{exerciseId}/tasks") @EnforceAtLeastEditor @FeatureToggle(Feature.ProgrammingExercises) - public ResponseEntity deleteTaskWithSolutionEntries(@PathVariable Long exerciseId) { - log.debug("REST request to delete ProgrammingExerciseTasks with ProgrammingExerciseSolutionEntries for ProgrammingExercise with id : {}", exerciseId); + public ResponseEntity deleteTasks(@PathVariable Long exerciseId) { + log.debug("REST request to delete tasks for ProgrammingExercise with id : {}", exerciseId); ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.EDITOR, exercise, null); - programmingExerciseService.deleteTasksWithSolutionEntries(exercise.getId()); + programmingExerciseService.deleteTasks(exercise.getId()); return ResponseEntity.noContent().build(); } @@ -859,30 +853,6 @@ public ModelAndView redirectGetTemplateRepositoryFiles(@PathVariable Long exerci return new ModelAndView("forward:/api/repository/" + participation.getId() + "/files-content"); } - /** - * GET programming-exercises/:exerciseId/solution-file-names - *

- * Returns the solution repository file names for a given programming exercise. - * Note: This endpoint redirects the request to the ProgrammingExerciseParticipationService. This is required if - * the solution participation id is not known for the client. - * - * @param exerciseId the exercise for which the solution repository files should be retrieved - * @return a redirect to the endpoint returning the files with content - */ - @GetMapping("programming-exercises/{exerciseId}/file-names") - @EnforceAtLeastTutor - @FeatureToggle(Feature.ProgrammingExercises) - public ModelAndView redirectGetSolutionRepositoryFilesWithoutContent(@PathVariable Long exerciseId) { - log.debug("REST request to get latest solution repository file names for ProgrammingExercise with id : {}", exerciseId); - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.TEACHING_ASSISTANT, exercise, null); - - var participation = solutionProgrammingExerciseParticipationRepository.findByProgrammingExerciseIdElseThrow(exerciseId); - - // TODO: We want to get rid of ModelAndView and use ResponseEntity instead. Define an appropriate service method and then call it here and in the referenced endpoint. - return new ModelAndView("forward:/api/repository/" + participation.getId() + "/file-names"); - } - /** * GET programming-exercises/:exerciseId/build-log-statistics *

diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ProgrammingExerciseTaskResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseTaskResource.java similarity index 94% rename from src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ProgrammingExerciseTaskResource.java rename to src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseTaskResource.java index 379ebfacb035..64b0a34c2b16 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ProgrammingExerciseTaskResource.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExerciseTaskResource.java @@ -1,4 +1,4 @@ -package de.tum.cit.aet.artemis.programming.web.hestia; +package de.tum.cit.aet.artemis.programming.web; import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; @@ -18,9 +18,9 @@ import de.tum.cit.aet.artemis.core.security.annotations.EnforceAtLeastTutor; import de.tum.cit.aet.artemis.core.service.AuthorizationCheckService; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseTaskService; +import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseTaskService; /** * REST controller for managing {@link ProgrammingExerciseTask}. diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/CodeHintResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/CodeHintResource.java deleted file mode 100644 index 122aa11b7a17..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/CodeHintResource.java +++ /dev/null @@ -1,124 +0,0 @@ -package de.tum.cit.aet.artemis.programming.web.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.util.List; -import java.util.Objects; -import java.util.Set; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Profile; -import org.springframework.http.ResponseEntity; -import org.springframework.web.bind.annotation.DeleteMapping; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.PostMapping; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RequestParam; -import org.springframework.web.bind.annotation.RestController; - -import de.tum.cit.aet.artemis.core.exception.AccessForbiddenException; -import de.tum.cit.aet.artemis.core.exception.ConflictException; -import de.tum.cit.aet.artemis.core.security.annotations.enforceRoleInExercise.EnforceAtLeastEditorInExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CodeHintRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseSolutionEntryRepository; -import de.tum.cit.aet.artemis.programming.service.hestia.CodeHintService; - -/** - * REST controller for managing {@link CodeHint}. - */ -@Profile(PROFILE_CORE) -@RestController -@RequestMapping("api/") -public class CodeHintResource { - - private static final Logger log = LoggerFactory.getLogger(CodeHintResource.class); - - private final ProgrammingExerciseRepository programmingExerciseRepository; - - private final ProgrammingExerciseSolutionEntryRepository solutionEntryRepository; - - private final CodeHintRepository codeHintRepository; - - private final CodeHintService codeHintService; - - public CodeHintResource(ProgrammingExerciseRepository programmingExerciseRepository, ProgrammingExerciseSolutionEntryRepository solutionEntryRepository, - CodeHintRepository codeHintRepository, CodeHintService codeHintService) { - this.programmingExerciseRepository = programmingExerciseRepository; - this.solutionEntryRepository = solutionEntryRepository; - this.codeHintRepository = codeHintRepository; - this.codeHintService = codeHintService; - } - - /** - * GET programming-exercises/{exerciseId}/code-hints: Retrieve all code hints for a programming exercise. - * - * @param exerciseId of the exercise - * @return the {@link ResponseEntity} with status {@code 200 (Ok)} and with body the code hints for the exercise - */ - @GetMapping("programming-exercises/{exerciseId}/code-hints") - @EnforceAtLeastEditorInExercise - public ResponseEntity> getAllCodeHints(@PathVariable Long exerciseId) { - var result = codeHintRepository.findByExerciseId(exerciseId); - return ResponseEntity.ok(result); - } - - /** - * {@code POST programming-exercises/:exerciseId/code-hints} : Create a new exerciseHint for an exercise. - * - * @param exerciseId the exerciseId of the exercise of which to create the exerciseHint - * @param deleteOldCodeHints Whether old code hints should be deleted - * @return the {@link ResponseEntity} with status {@code 200 (Ok)} and with body the new code hints - */ - @PostMapping("programming-exercises/{exerciseId}/code-hints") - @EnforceAtLeastEditorInExercise - public ResponseEntity> generateCodeHintsForExercise(@PathVariable Long exerciseId, - @RequestParam(value = "deleteOldCodeHints", defaultValue = "true") boolean deleteOldCodeHints) { - log.debug("REST request to generate CodeHints for ProgrammingExercise: {}", exerciseId); - - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - - // Hints for exam exercises are not supported at the moment - if (exercise.isExamExercise()) { - throw new AccessForbiddenException("Code hints for exams are currently not supported"); - } - - var codeHints = codeHintService.generateCodeHintsForExercise(exercise, deleteOldCodeHints); - return ResponseEntity.ok(codeHints); - } - - /** - * {@code DELETE programming-exercises/:exerciseId/code-hints/:codeHintId/solution-entries/:solutionEntryId} : - * Removes a solution entry from a code hint. - * - * @param exerciseId The id of the exercise of the code hint - * @param codeHintId The id of the code hint - * @param solutionEntryId The id of the solution entry - * @return 204 No Content - */ - @DeleteMapping("programming-exercises/{exerciseId}/code-hints/{codeHintId}/solution-entries/{solutionEntryId}") - @EnforceAtLeastEditorInExercise - public ResponseEntity removeSolutionEntryFromCodeHint(@PathVariable Long exerciseId, @PathVariable Long codeHintId, @PathVariable Long solutionEntryId) { - log.debug("REST request to remove SolutionEntry {} from CodeHint {} in ProgrammingExercise {}", solutionEntryId, codeHintId, exerciseId); - - var codeHint = codeHintRepository.findByIdWithSolutionEntriesElseThrow(codeHintId); - if (!Objects.equals(codeHint.getExercise().getId(), exerciseId)) { - throw new ConflictException("The code hint does not belong to the exercise", "CodeHint", "codeHintExerciseConflict"); - } - - var solutionEntry = codeHint.getSolutionEntries().stream().filter(solutionEntry1 -> solutionEntry1.getId().equals(solutionEntryId)).findFirst().orElse(null); - if (solutionEntry == null) { - throw new ConflictException("The solution entry does not belong to the code hint", "SolutionEntry", "solutionEntryCodeHintConflict"); - } - - solutionEntry.setCodeHint(null); - solutionEntryRepository.save(solutionEntry); - codeHint.getSolutionEntries().remove(solutionEntry); - - return ResponseEntity.noContent().build(); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/CoverageReportResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/CoverageReportResource.java deleted file mode 100644 index ad2c9a574268..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/CoverageReportResource.java +++ /dev/null @@ -1,68 +0,0 @@ -package de.tum.cit.aet.artemis.programming.web.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Profile; -import org.springframework.http.ResponseEntity; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RestController; - -import de.tum.cit.aet.artemis.core.exception.EntityNotFoundException; -import de.tum.cit.aet.artemis.core.security.annotations.enforceRoleInExercise.EnforceAtLeastTutorInExercise; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageReport; -import de.tum.cit.aet.artemis.programming.service.hestia.TestwiseCoverageService; - -/** - * REST controller for managing ProgrammingExerciseTestwiseCoverageReports and its entries. - */ -@Profile(PROFILE_CORE) -@RestController -@RequestMapping("api/") -public class CoverageReportResource { - - private static final Logger log = LoggerFactory.getLogger(CoverageReportResource.class); - - private final TestwiseCoverageService testwiseCoverageService; - - public CoverageReportResource(TestwiseCoverageService testwiseCoverageService) { - this.testwiseCoverageService = testwiseCoverageService; - } - - /** - * {@code GET exercises/:exerciseId/full-testwise-coverage-report} : Get the latest coverage report for a solution submission - * of a programming exercise with all file reports and descendants. - * - * @param exerciseId the exerciseId of the exercise of which to retrieve the testwise coverage report for the latest solution submission - * @return the {@link ResponseEntity} with status {@code 200 (Ok)} and with body the coverage report - */ - @GetMapping("programming-exercises/{exerciseId}/full-testwise-coverage-report") - @EnforceAtLeastTutorInExercise - public ResponseEntity getLatestFullCoverageReport(@PathVariable Long exerciseId) { - log.debug("REST request to get the latest Full Testwise CoverageReport for exercise {}", exerciseId); - - var optionalReportWithFileReports = testwiseCoverageService.getFullCoverageReportForLatestSolutionSubmissionFromProgrammingExercise(exerciseId) - .orElseThrow(() -> new EntityNotFoundException("Coverage report for exercise " + exerciseId + " not found.")); - return ResponseEntity.ok(optionalReportWithFileReports); - } - - /** - * {@code GET exercises/:exerciseId/testwise-coverage-report} : Get the latest coverage report for a solution submission - * of a programming exercise without the actual file reports. - * - * @param exerciseId the exerciseId of the exercise of which to retrieve the testwise coverage report for the latest solution submission - * @return the {@link ResponseEntity} with status {@code 200 (Ok)} and with body the coverage report - */ - @GetMapping("programming-exercises/{exerciseId}/testwise-coverage-report") - @EnforceAtLeastTutorInExercise - public ResponseEntity getLatestCoverageReport(@PathVariable Long exerciseId) { - log.debug("REST request to get the latest Testwise CoverageReport for exercise {}", exerciseId); - - var optionalReportWithoutFileReports = testwiseCoverageService.getCoverageReportForLatestSolutionSubmissionFromProgrammingExercise(exerciseId) - .orElseThrow(() -> new EntityNotFoundException("Coverage report for exercise " + exerciseId + " not found.")); - return ResponseEntity.ok(optionalReportWithoutFileReports); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ExerciseHintResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ExerciseHintResource.java deleted file mode 100644 index 029a374a2a87..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ExerciseHintResource.java +++ /dev/null @@ -1,349 +0,0 @@ -package de.tum.cit.aet.artemis.programming.web.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.net.URI; -import java.net.URISyntaxException; -import java.util.Set; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.beans.factory.annotation.Value; -import org.springframework.context.annotation.Profile; -import org.springframework.http.ResponseEntity; -import org.springframework.web.bind.annotation.DeleteMapping; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.PostMapping; -import org.springframework.web.bind.annotation.PutMapping; -import org.springframework.web.bind.annotation.RequestBody; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RestController; - -import de.tum.cit.aet.artemis.core.exception.BadRequestAlertException; -import de.tum.cit.aet.artemis.core.exception.ConflictException; -import de.tum.cit.aet.artemis.core.repository.UserRepository; -import de.tum.cit.aet.artemis.core.security.annotations.EnforceAtLeastStudent; -import de.tum.cit.aet.artemis.core.security.annotations.enforceRoleInExercise.EnforceAtLeastEditorInExercise; -import de.tum.cit.aet.artemis.core.security.annotations.enforceRoleInExercise.EnforceAtLeastStudentInExercise; -import de.tum.cit.aet.artemis.core.security.annotations.enforceRoleInExercise.EnforceAtLeastTutorInExercise; -import de.tum.cit.aet.artemis.exercise.domain.Exercise; -import de.tum.cit.aet.artemis.exercise.repository.ExerciseRepository; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; -import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ExerciseHintRepository; -import de.tum.cit.aet.artemis.programming.service.hestia.CodeHintService; -import de.tum.cit.aet.artemis.programming.service.hestia.ExerciseHintService; -import tech.jhipster.web.util.HeaderUtil; - -/** - * REST controller for managing {@link ExerciseHint}. - */ -@Profile(PROFILE_CORE) -@RestController -@RequestMapping("api/") -public class ExerciseHintResource { - - private static final String EXERCISE_HINT_ENTITY_NAME = "exerciseHint"; - - private static final String CODE_HINT_ENTITY_NAME = "codeHint"; - - private static final Logger log = LoggerFactory.getLogger(ExerciseHintResource.class); - - private final ExerciseHintService exerciseHintService; - - private final ExerciseHintRepository exerciseHintRepository; - - private final ProgrammingExerciseRepository programmingExerciseRepository; - - private final ExerciseRepository exerciseRepository; - - private final CodeHintService codeHintService; - - private final UserRepository userRepository; - - @Value("${jhipster.clientApp.name}") - private String applicationName; - - public ExerciseHintResource(ExerciseHintService exerciseHintService, ExerciseHintRepository exerciseHintRepository, ProgrammingExerciseRepository programmingExerciseRepository, - ExerciseRepository exerciseRepository, CodeHintService codeHintService, UserRepository userRepository) { - this.exerciseHintService = exerciseHintService; - this.exerciseHintRepository = exerciseHintRepository; - this.programmingExerciseRepository = programmingExerciseRepository; - this.exerciseRepository = exerciseRepository; - this.codeHintService = codeHintService; - this.userRepository = userRepository; - } - - /** - * {@code POST programming-exercises/:exerciseId/exercise-hints} : Create a new exerciseHint for an exercise. - * - * @param exerciseHint the exerciseHint to create - * @param exerciseId the exerciseId of the exercise of which to create the exerciseHint - * @return the {@link ResponseEntity} with status {@code 201 (Created)} and with body the new exerciseHint, - * or with status {@code 409 (Conflict)} if the exerciseId is invalid, - * @throws URISyntaxException if the Location URI syntax is incorrect. - */ - @PostMapping("programming-exercises/{exerciseId}/exercise-hints") - @EnforceAtLeastEditorInExercise - public ResponseEntity createExerciseHint(@RequestBody ExerciseHint exerciseHint, @PathVariable Long exerciseId) throws URISyntaxException { - log.debug("REST request to save ExerciseHint : {}", exerciseHint); - - // Reload the exercise from the database as we can't trust data from the client - Exercise exercise = exerciseRepository.findByIdElseThrow(exerciseId); - - if (exerciseHint instanceof CodeHint) { - throw new BadRequestAlertException("A code hint cannot be created manually.", CODE_HINT_ENTITY_NAME, "manualCodeHintOperation"); - } - if (exerciseHint.getExercise() == null) { - throw new ConflictException("An exercise hint can only be created if the exercise is defined.", EXERCISE_HINT_ENTITY_NAME, "exerciseNotDefined"); - } - - if (!exerciseHint.getExercise().getId().equals(exerciseId)) { - throw new ConflictException("An exercise hint can only be created if the exerciseIds match.", EXERCISE_HINT_ENTITY_NAME, "exerciseIdMismatch"); - } - - // Hints for exam exercises are not supported at the moment - if (exercise.isExamExercise()) { - throw new BadRequestAlertException("Exercise hints for exams are currently not supported", EXERCISE_HINT_ENTITY_NAME, "exerciseHintNotSupported"); - } - ExerciseHint result = exerciseHintRepository.save(exerciseHint); - return ResponseEntity.created(new URI("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + result.getId())) - .headers(HeaderUtil.createEntityCreationAlert(applicationName, true, EXERCISE_HINT_ENTITY_NAME, result.getId().toString())).body(result); - } - - /** - * {@code PUT programming-exercises/:exerciseId/exercise-hints/:exerciseHintId} : Updates an existing exerciseHint. - * - * @param exerciseHint the exerciseHint to update - * @param exerciseId the exerciseId of the exercise of which to update the exerciseHint - * @param exerciseHintId the id to the exerciseHint - * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body the updated exerciseHint, - * or with status {@code 409 (Conflict} if the exerciseHint or exerciseId are not valid, - * or with status {@code 500 (Internal Server Error)} if the exerciseHint couldn't be updated. - */ - @PutMapping("programming-exercises/{exerciseId}/exercise-hints/{exerciseHintId}") - @EnforceAtLeastEditorInExercise - public ResponseEntity updateExerciseHint(@RequestBody ExerciseHint exerciseHint, @PathVariable Long exerciseHintId, @PathVariable Long exerciseId) { - log.debug("REST request to update ExerciseHint : {}", exerciseHint); - - // Reload the exercise from the database as we can't trust data from the client - Exercise exercise = exerciseRepository.findByIdElseThrow(exerciseId); - var hintBeforeSaving = exerciseHintRepository.findByIdWithRelationsElseThrow(exerciseHintId); - - if (!exerciseHint.getClass().equals(hintBeforeSaving.getClass())) { - throw new BadRequestAlertException("A code hint cannot be converted to or from a normal hint.", CODE_HINT_ENTITY_NAME, "manualCodeHintOperation"); - } - - if (exerciseHint.getId() == null || !exerciseHintId.equals(exerciseHint.getId()) || exerciseHint.getExercise() == null) { - throw new ConflictException("An exercise hint can only be changed if it has an ID and if the exercise is not null.", EXERCISE_HINT_ENTITY_NAME, "exerciseNotDefined"); - } - - if (!exerciseHint.getExercise().getId().equals(exerciseId)) { - throw new ConflictException("An exercise hint can only be updated if the exerciseIds match.", EXERCISE_HINT_ENTITY_NAME, "exerciseIdsMismatch"); - } - - // Hints for exam exercises are not supported at the moment - if (exercise.isExamExercise()) { - throw new BadRequestAlertException("Exercise hints for exams are currently not supported", EXERCISE_HINT_ENTITY_NAME, "exerciseHintNotSupported"); - } - - if (exerciseHint instanceof CodeHint codeHint && codeHint.getSolutionEntries() != null) { - codeHintService.updateSolutionEntriesForCodeHint(codeHint); - } - exerciseHint.setExerciseHintActivations(hintBeforeSaving.getExerciseHintActivations()); - ExerciseHint result = exerciseHintRepository.save(exerciseHint); - return ResponseEntity.ok().headers(HeaderUtil.createEntityUpdateAlert(applicationName, true, EXERCISE_HINT_ENTITY_NAME, exerciseHint.getId().toString())).body(result); - } - - /** - * {@code GET programming-exercises/:exerciseId/exercise-hints/:exerciseHintId/title} : Returns the title of the hint with the given id - * - * @param exerciseHintId the id of the exerciseHint - * @param exerciseId the exerciseId of the exercise of which to retrieve the exerciseHints' title - * @return the title of the hint wrapped in an ResponseEntity or 404 Not Found if no hint with that id exists - * or with status {@code 409 (Conflict)} if the exerciseId is not valid. - */ - @GetMapping("programming-exercises/{exerciseId}/exercise-hints/{exerciseHintId}/title") - @EnforceAtLeastStudent - public ResponseEntity getHintTitle(@PathVariable Long exerciseId, @PathVariable Long exerciseHintId) { - var title = exerciseHintService.getExerciseHintTitle(exerciseId, exerciseHintId); - return title == null ? ResponseEntity.notFound().build() : ResponseEntity.ok(title); - } - - /** - * {@code GET programming-exercises/:exerciseId/exercise-hints/:exerciseHintId} : get the exerciseHint with the given id. - * - * @param exerciseHintId the id of the exerciseHint to retrieve. - * @param exerciseId the exerciseId of the exercise of which to retrieve the exerciseHint - * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body the exerciseHint, - * or with status {@code 404 (Not Found)}, - * or with status {@code 409 (Conflict)} if the exerciseId is not valid. - */ - @GetMapping("programming-exercises/{exerciseId}/exercise-hints/{exerciseHintId}") - @EnforceAtLeastTutorInExercise - public ResponseEntity getExerciseHint(@PathVariable Long exerciseId, @PathVariable Long exerciseHintId) { - log.debug("REST request to get ExerciseHint : {}", exerciseHintId); - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - if (exercise.isExamExercise()) { - // not allowed for exam exercises - throw new BadRequestAlertException("Exercise hints for exams are currently not supported", EXERCISE_HINT_ENTITY_NAME, "exerciseHintNotSupported"); - } - - var exerciseHint = exerciseHintRepository.findByIdWithRelationsElseThrow(exerciseHintId); - - if (!exerciseHint.getExercise().getId().equals(exerciseId)) { - throw new ConflictException("An exercise hint can only be retrieved if the exerciseIds match.", EXERCISE_HINT_ENTITY_NAME, "exerciseIdsMismatch"); - } - - return ResponseEntity.ok().body(exerciseHint); - } - - /** - * {@code GET programming-exercises/:exerciseId/exercise-hints} : get the exerciseHints of a provided exercise. - * - * @param exerciseId the exercise id of which to retrieve the exercise hints. - * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body the exerciseHint, - * or with status {@code 404 (Not Found)}, - * or with status {@code 409 (Conflict)} if the exerciseId is not valid. - */ - @GetMapping("programming-exercises/{exerciseId}/exercise-hints") - @EnforceAtLeastTutorInExercise - public ResponseEntity> getExerciseHintsForExercise(@PathVariable Long exerciseId) { - log.debug("REST request to get ExerciseHints : {}", exerciseId); - var exerciseHints = exerciseHintRepository.findByExerciseIdWithRelations(exerciseId); - return ResponseEntity.ok(exerciseHints); - } - - /** - * {@code GET programming-exercises/:exerciseId/exercise-hints/activated} : get the exercise hints of a provided exercise that the user has activated - * - * @param exerciseId the exercise id of which to retrieve the exercise hints. - * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body the exercise hints, - * or with status {@code 404 (Not Found)} - */ - @GetMapping("programming-exercises/{exerciseId}/exercise-hints/activated") - @EnforceAtLeastStudentInExercise - public ResponseEntity> getActivatedExerciseHintsForExercise(@PathVariable Long exerciseId) { - log.debug("REST request to get activated ExerciseHints : {}", exerciseId); - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - if (exercise.isExamExercise()) { - // not allowed for exam exercises - throw new BadRequestAlertException("Exercise hints for exams are currently not supported", EXERCISE_HINT_ENTITY_NAME, "exerciseHintNotSupported"); - } - - var user = userRepository.getUserWithGroupsAndAuthorities(); - var exerciseHints = exerciseHintService.getActivatedExerciseHints(exercise, user); - return ResponseEntity.ok(exerciseHints); - } - - /** - * {@code GET programming-exercises/:exerciseId/available-exercise-hints} : get the available exercise hints. - * - * @param exerciseId the exerciseId of the exercise of which to retrieve the exercise hint - * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body the exercise hints - */ - @GetMapping("programming-exercises/{exerciseId}/exercise-hints/available") - @EnforceAtLeastStudentInExercise - public ResponseEntity> getAvailableExerciseHintsForExercise(@PathVariable Long exerciseId) { - log.debug("REST request to get a CodeHint for programming exercise : {}", exerciseId); - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - if (exercise.isExamExercise()) { - // not allowed for exam exercises - throw new BadRequestAlertException("Exercise hints for exams are currently not supported", EXERCISE_HINT_ENTITY_NAME, "exerciseHintNotSupported"); - } - - var user = userRepository.getUserWithGroupsAndAuthorities(); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, user); - availableExerciseHints.forEach(ExerciseHint::removeContent); - - return ResponseEntity.ok().body(availableExerciseHints); - } - - /** - * {@code POST programming-exercises/:exerciseId/exercise-hints/:exerciseHintId/activate} - * Activates a single exercise hint of an exercise for the logged-in user - * - * @param exerciseId The id of the exercise of which to activate the exercise hint - * @param exerciseHintId The id of the exercise hint to activate - * @return The {@link ResponseEntity} with status {@code 200 (OK)} and with body the activated exercise hint with content - * or with status {@code 400 (BAD_REQUEST)} if the hint could not be activated - */ - @PostMapping("programming-exercises/{exerciseId}/exercise-hints/{exerciseHintId}/activate") - @EnforceAtLeastStudentInExercise - public ResponseEntity activateExerciseHint(@PathVariable Long exerciseId, @PathVariable Long exerciseHintId) { - log.debug("REST request to activate ExerciseHint : {}", exerciseHintId); - var exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - var user = userRepository.getUserWithGroupsAndAuthorities(); - - var exerciseHint = exerciseHintRepository.findByIdWithRelationsElseThrow(exerciseHintId); - - if (!exerciseHint.getExercise().getId().equals(exercise.getId())) { - throw new ConflictException("An exercise hint can only be deleted if the exerciseIds match.", EXERCISE_HINT_ENTITY_NAME, "exerciseIdsMismatch"); - } - - if (exerciseHintService.activateHint(exerciseHint, user)) { - return ResponseEntity.ok(exerciseHint); - } - else { - throw new BadRequestAlertException("Unable to activate exercise hint", EXERCISE_HINT_ENTITY_NAME, "exerciseHintIdActivationFailed"); - } - } - - /** - * {@code POST programming-exercises/:exerciseId/exercise-hints/:exerciseHintId/rating/:ratingValue}: Rates an exercise hint - * - * @param exerciseId The id of the exercise of which to activate the exercise hint - * @param exerciseHintId The id of the exercise hint to activate - * @param ratingValue The value of the rating - * @return The {@link ResponseEntity} with status {@code 200 (OK)} - */ - @PostMapping("programming-exercises/{exerciseId}/exercise-hints/{exerciseHintId}/rating/{ratingValue}") - @EnforceAtLeastStudentInExercise - public ResponseEntity rateExerciseHint(@PathVariable Long exerciseId, @PathVariable Long exerciseHintId, @PathVariable Integer ratingValue) { - log.debug("REST request to rate ExerciseHint : {}", exerciseHintId); - var user = userRepository.getUserWithGroupsAndAuthorities(); - - var exerciseHint = exerciseHintRepository.findByIdWithRelationsElseThrow(exerciseHintId); - if (!exerciseHint.getExercise().getId().equals(exerciseId)) { - throw new ConflictException("An exercise hint can only be deleted if the exerciseIds match.", EXERCISE_HINT_ENTITY_NAME, "exerciseIdsMismatch"); - } - - exerciseHintService.rateExerciseHint(exerciseHint, user, ratingValue); - - return ResponseEntity.ok().build(); - } - - /** - * {@code DELETE programming-exercises/:exerciseId/exercise-hints/:exerciseHintId} : delete the exerciseHint with given id. - * - * @param exerciseHintId the id of the exerciseHint to delete - * @param exerciseId the exercise id of which to delete the exercise hint - * @return the {@link ResponseEntity} with status {@code 204 (NO_CONTENT)}, - * or with status {@code 409 (Conflict)} if the exerciseId is not valid. - */ - @DeleteMapping("programming-exercises/{exerciseId}/exercise-hints/{exerciseHintId}") - @EnforceAtLeastEditorInExercise - public ResponseEntity deleteExerciseHint(@PathVariable Long exerciseId, @PathVariable Long exerciseHintId) { - log.debug("REST request to delete ExerciseHint : {}", exerciseHintId); - var exerciseHint = exerciseHintRepository.findByIdElseThrow(exerciseHintId); - - if (!exerciseHint.getExercise().getId().equals(exerciseId)) { - throw new ConflictException("An exercise hint can only be deleted if the exerciseIds match.", EXERCISE_HINT_ENTITY_NAME, "exerciseIdsMismatch"); - } - - String entityName; - - if (exerciseHint instanceof CodeHint codeHint) { - codeHintService.deleteCodeHint(codeHint); - entityName = CODE_HINT_ENTITY_NAME; - } - else { - exerciseHintRepository.deleteById(exerciseHintId); - entityName = EXERCISE_HINT_ENTITY_NAME; - } - return ResponseEntity.noContent().headers(HeaderUtil.createEntityDeletionAlert(applicationName, true, entityName, exerciseHintId.toString())).build(); - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ProgrammingExerciseSolutionEntryResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ProgrammingExerciseSolutionEntryResource.java deleted file mode 100644 index c18e84806672..000000000000 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/hestia/ProgrammingExerciseSolutionEntryResource.java +++ /dev/null @@ -1,337 +0,0 @@ -package de.tum.cit.aet.artemis.programming.web.hestia; - -import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; - -import java.net.URI; -import java.net.URISyntaxException; -import java.util.List; -import java.util.Set; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.beans.factory.annotation.Value; -import org.springframework.context.annotation.Profile; -import org.springframework.http.ResponseEntity; -import org.springframework.web.bind.annotation.DeleteMapping; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.PostMapping; -import org.springframework.web.bind.annotation.PutMapping; -import org.springframework.web.bind.annotation.RequestBody; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RestController; - -import de.tum.cit.aet.artemis.core.exception.ConflictException; -import de.tum.cit.aet.artemis.core.exception.InternalServerErrorException; -import de.tum.cit.aet.artemis.core.security.Role; -import de.tum.cit.aet.artemis.core.security.annotations.EnforceAtLeastEditor; -import de.tum.cit.aet.artemis.core.security.annotations.EnforceAtLeastStudent; -import de.tum.cit.aet.artemis.core.security.annotations.EnforceAtLeastTutor; -import de.tum.cit.aet.artemis.core.service.AuthorizationCheckService; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseRepository; -import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTestCaseRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CodeHintRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseSolutionEntryRepository; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralSolutionEntryGenerationException; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralTestCaseService; -import de.tum.cit.aet.artemis.programming.service.hestia.structural.StructuralSolutionEntryGenerationException; -import de.tum.cit.aet.artemis.programming.service.hestia.structural.StructuralTestCaseService; -import tech.jhipster.web.util.HeaderUtil; - -/** - * REST controller for managing {@link ProgrammingExerciseSolutionEntry}. - */ -@Profile(PROFILE_CORE) -@RestController -@RequestMapping("api/") -public class ProgrammingExerciseSolutionEntryResource { - - private static final Logger log = LoggerFactory.getLogger(ProgrammingExerciseSolutionEntryResource.class); - - private static final String ENTITY_NAME = "programmingExerciseSolutionEntry"; - - @Value("${jhipster.clientApp.name}") - private String applicationName; - - private final ProgrammingExerciseSolutionEntryRepository programmingExerciseSolutionEntryRepository; - - private final ProgrammingExerciseRepository programmingExerciseRepository; - - private final CodeHintRepository codeHintRepository; - - private final ProgrammingExerciseTestCaseRepository programmingExerciseTestCaseRepository; - - private final AuthorizationCheckService authCheckService; - - private final StructuralTestCaseService structuralTestCaseService; - - private final BehavioralTestCaseService behavioralTestCaseService; - - public ProgrammingExerciseSolutionEntryResource(ProgrammingExerciseSolutionEntryRepository programmingExerciseSolutionEntryRepository, - ProgrammingExerciseRepository programmingExerciseRepository, CodeHintRepository codeHintRepository, - ProgrammingExerciseTestCaseRepository programmingExerciseTestCaseRepository, AuthorizationCheckService authCheckService, - StructuralTestCaseService structuralTestCaseService, BehavioralTestCaseService behavioralTestCaseService) { - this.programmingExerciseSolutionEntryRepository = programmingExerciseSolutionEntryRepository; - this.programmingExerciseRepository = programmingExerciseRepository; - this.codeHintRepository = codeHintRepository; - this.programmingExerciseTestCaseRepository = programmingExerciseTestCaseRepository; - this.authCheckService = authCheckService; - this.structuralTestCaseService = structuralTestCaseService; - this.behavioralTestCaseService = behavioralTestCaseService; - } - - /** - * GET programming-exercises/:exerciseId/solution-entries/:solutionEntryId : Get the solution entry with test cases and programming exercise - * - * @param exerciseId of the exercise - * @param solutionEntryId of the solution entry - * @return the {@link ResponseEntity} with status {@code 200} and with body the solution entries with test cases and exercise, - * or with status {@code 409 (Conflict)} if the exerciseId or solutionEntryId are not valid. - */ - @GetMapping("programming-exercises/{exerciseId}/solution-entries/{solutionEntryId}") - @EnforceAtLeastTutor - public ResponseEntity getSolutionEntry(@PathVariable Long exerciseId, @PathVariable Long solutionEntryId) { - log.debug("REST request to retrieve SolutionEntry : {}", solutionEntryId); - // Reload the exercise from the database as we can't trust data from the client - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.TEACHING_ASSISTANT, exercise, null); - - ProgrammingExerciseSolutionEntry solutionEntry = programmingExerciseSolutionEntryRepository.findByIdWithTestCaseAndProgrammingExerciseElseThrow(solutionEntryId); - - if (!exerciseId.equals(solutionEntry.getTestCase().getExercise().getId())) { - throw new ConflictException("A solution entry can only be retrieved if the exercise match", ENTITY_NAME, "exerciseIdsMismatch"); - } - return ResponseEntity.ok(solutionEntry); - } - - /** - * GET programming-exercises/{exerciseId}/solution-entries: Get all solution entries with test cases for a programming exercise - * - * @param exerciseId of the exercise - * @return the {@link ResponseEntity} with status {@code 200} and with body the solution entries with test cases. - */ - @GetMapping("programming-exercises/{exerciseId}/solution-entries") - @EnforceAtLeastEditor - public ResponseEntity> getAllSolutionEntries(@PathVariable Long exerciseId) { - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.EDITOR, exercise, null); - - var result = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(exercise.getId()); - return ResponseEntity.ok(result); - } - - /** - * GET programming-exercises/:exerciseId/code-hints/:codeHintId/solution-entries : Get all solution entries for a given code hint - * - * @param exerciseId of the exercise - * @param codeHintId of the code hint - * @return the {@link ResponseEntity} with status {@code 200} and with body the solution entries, - * or with status {@code 409 (Conflict)} if the exerciseId or codeHintId are not valid. - */ - @GetMapping("programming-exercises/{exerciseId}/code-hints/{codeHintId}/solution-entries") - @EnforceAtLeastStudent - public ResponseEntity> getSolutionEntriesForCodeHint(@PathVariable Long exerciseId, @PathVariable Long codeHintId) { - log.debug("REST request to retrieve SolutionEntry for CodeHint with id : {}", codeHintId); - // Reload the exercise from the database as we can't trust data from the client - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.STUDENT, exercise, null); - - CodeHint codeHint = codeHintRepository.findByIdElseThrow(codeHintId); - if (!exercise.getId().equals(codeHint.getExercise().getId())) { - throw new ConflictException("A solution entry can only be retrieved if the code hint belongs to the exercise", ENTITY_NAME, "exerciseIdsMismatch"); - } - - Set solutionEntries = programmingExerciseSolutionEntryRepository.findByCodeHintId(codeHintId); - return ResponseEntity.ok(solutionEntries); - } - - /** - * GET programming-exercises/:exerciseId/test-cases/:testCaseId/solution-entries : Get all solution entries for a given test case - * - * @param exerciseId of the exercise - * @param testCaseId of the test case - * @return the {@link ResponseEntity} with status {@code 200} and with body the solution entries, - * or with status {@code 409 (Conflict)} if the exerciseId or testCaseId are not valid. - */ - @GetMapping("programming-exercises/{exerciseId}/test-cases/{testCaseId}/solution-entries") - @EnforceAtLeastStudent - public ResponseEntity> getSolutionEntriesForTestCase(@PathVariable Long exerciseId, @PathVariable Long testCaseId) { - log.debug("REST request to retrieve SolutionEntry for ProgrammingExerciseTestCase with id : {}", testCaseId); - // Reload the exercise from the database as we can't trust data from the client - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.STUDENT, exercise, null); - - ProgrammingExerciseTestCase testCase = programmingExerciseTestCaseRepository.findByIdWithExerciseElseThrow(testCaseId); - if (!exercise.getId().equals(testCase.getExercise().getId())) { - throw new ConflictException("A solution entry can only be retrieved if the test case belongs to the exercise", ENTITY_NAME, "exerciseIdsMismatch"); - } - - Set solutionEntries = programmingExerciseSolutionEntryRepository.findByTestCaseId(testCaseId); - return ResponseEntity.ok(solutionEntries); - } - - /** - * POST programming-exercises/:exerciseId/test-cases/:testCaseId/solution-entries : Create a solution entry for a test case - * - * @param exerciseId of the exercise - * @param testCaseId of the test case - * @param programmingExerciseSolutionEntry the solution entry to be created - * @return the {@link ResponseEntity} with status {@code 201} and with body the created solution entry, - * or with status {@code 409 (Conflict)} if the exerciseId, testcaseId, or solution entry are not valid. - */ - @PostMapping("programming-exercises/{exerciseId}/test-cases/{testCaseId}/solution-entries") - @EnforceAtLeastEditor - public ResponseEntity createSolutionEntryForTestCase(@PathVariable Long exerciseId, @PathVariable Long testCaseId, - @RequestBody ProgrammingExerciseSolutionEntry programmingExerciseSolutionEntry) throws URISyntaxException { - log.debug("REST request to create SolutionEntry : {}", programmingExerciseSolutionEntry); - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.EDITOR, exercise, null); - - ProgrammingExerciseTestCase testCase = programmingExerciseTestCaseRepository.findByIdWithExerciseElseThrow(testCaseId); - checkExerciseContainsTestCaseElseThrow(exercise, testCase); - - ProgrammingExerciseSolutionEntry result = programmingExerciseSolutionEntryRepository.save(programmingExerciseSolutionEntry); - return ResponseEntity.created(new URI("/api/programming-exercises/" + exerciseId + "/solution-entries/" + result.getId())) - .headers(HeaderUtil.createEntityCreationAlert(applicationName, true, ENTITY_NAME, result.getId().toString())).body(result); - } - - /** - * PUT programming-exercises/:exerciseId/test-cases/:testCaseId/solution-entries/:solutionEntryId : Update a solution entry - * - * @param exerciseId of the exercise - * @param testCaseId of the test case - * @param solutionEntryId of the solution entry - * @param solutionEntry the updated solution entry - * @return the {@link ResponseEntity} with status {@code 200} and with body the updated solution entry, - * or with status {@code 409 (Conflict)} if the exerciseId, testcaseId, solutionEntryId, or solution entry are not valid. - */ - @PutMapping("programming-exercises/{exerciseId}/test-cases/{testCaseId}/solution-entries/{solutionEntryId}") - @EnforceAtLeastEditor - public ResponseEntity updateSolutionEntry(@PathVariable Long exerciseId, @PathVariable Long testCaseId, @PathVariable Long solutionEntryId, - @RequestBody ProgrammingExerciseSolutionEntry solutionEntry) { - log.debug("REST request to update SolutionEntry : {}", solutionEntry); - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.EDITOR, exercise, null); - - ProgrammingExerciseTestCase testCase = programmingExerciseTestCaseRepository.findByIdWithExerciseElseThrow(testCaseId); - ProgrammingExerciseSolutionEntry solutionEntryBeforeSaving = programmingExerciseSolutionEntryRepository - .findByIdWithTestCaseAndProgrammingExerciseElseThrow(solutionEntryId); - - checkExerciseContainsTestCaseElseThrow(exercise, testCase); - if (!solutionEntryId.equals(solutionEntryBeforeSaving.getId())) { - throw new ConflictException("A solution entry can only be updated if the solutionEntryIds match", ENTITY_NAME, "solutionEntryError"); - } - - ProgrammingExerciseSolutionEntry solutionEntryAfterSaving = programmingExerciseSolutionEntryRepository.save(solutionEntry); - return ResponseEntity.ok().headers(HeaderUtil.createEntityUpdateAlert(applicationName, true, ENTITY_NAME, solutionEntryAfterSaving.getId().toString())) - .body(solutionEntryAfterSaving); - } - - /** - * DELETE programming-exercises/:exerciseId/test-cases/:testCaseId/solution-entries/:solutionEntryId : Delete a solution entry - * - * @param exerciseId of the exercise - * @param testCaseId of the test case - * @param solutionEntryId of the solution entry that is to be deleted - * @return the {@link ResponseEntity} with status {@code 204}, - * or with status {@code 409 (Conflict)} if the exerciseId, testcaseId, or solutionEntryId are not valid. - */ - @DeleteMapping("programming-exercises/{exerciseId}/test-cases/{testCaseId}/solution-entries/{solutionEntryId}") - @EnforceAtLeastEditor - public ResponseEntity deleteSolutionEntry(@PathVariable Long exerciseId, @PathVariable Long testCaseId, @PathVariable Long solutionEntryId) { - log.debug("REST request to delete SolutionEntry with id : {}", solutionEntryId); - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.EDITOR, exercise, null); - - ProgrammingExerciseTestCase testCase = programmingExerciseTestCaseRepository.findByIdWithExerciseElseThrow(testCaseId); - ProgrammingExerciseSolutionEntry solutionEntry = programmingExerciseSolutionEntryRepository.findByIdWithTestCaseAndProgrammingExerciseElseThrow(solutionEntryId); - - checkExerciseContainsTestCaseElseThrow(exercise, testCase); - checkTestCaseContainsSolutionEntryElseThrow(testCase, solutionEntry); - - programmingExerciseSolutionEntryRepository.deleteById(solutionEntryId); - return ResponseEntity.noContent().headers(HeaderUtil.createEntityDeletionAlert(applicationName, true, ENTITY_NAME, solutionEntry.getId().toString())).build(); - } - - /** - * DELETE programming-exercises/:exerciseId/solution-entries: Delete all solution entries for a programming exercise - * - * @param exerciseId of the exercise - * @return the {@link ResponseEntity} with status {@code 204}, - * or with status {@code 404} if the exerciseId is not valid. - */ - @DeleteMapping("programming-exercises/{exerciseId}/solution-entries") - @EnforceAtLeastEditor - public ResponseEntity deleteAllSolutionEntriesForExercise(@PathVariable Long exerciseId) { - log.debug("REST request to delete all SolutionEntries for exercise with id: {}", exerciseId); - ProgrammingExercise exercise = programmingExerciseRepository.findByIdElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.EDITOR, exercise, null); - - var entriesToDelete = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(exercise.getId()); - - programmingExerciseSolutionEntryRepository.deleteAll(entriesToDelete); - return ResponseEntity.noContent().build(); - } - - /** - * POST programming-exercises/:exerciseId/structural-solution-entries : Create the structural solution entries for a programming exercise - * - * @param exerciseId of the exercise - * @return the {@link ResponseEntity} with status {@code 200} and with body the created solution entries, - */ - @PostMapping("programming-exercises/{exerciseId}/structural-solution-entries") - @EnforceAtLeastEditor - public ResponseEntity> createStructuralSolutionEntries(@PathVariable Long exerciseId) { - log.debug("REST request to create structural solution entries"); - ProgrammingExercise exercise = programmingExerciseRepository.findByIdWithTemplateAndSolutionParticipationElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.EDITOR, exercise, null); - - try { - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - return ResponseEntity.ok(solutionEntries); - } - catch (StructuralSolutionEntryGenerationException e) { - log.error("Unable to create structural solution entries", e); - throw new InternalServerErrorException(e.getMessage()); - } - } - - /** - * POST programming-exercises/:exerciseId/behavioral-solution-entries : Create the behavioral solution entries for a programming exercise - * - * @param exerciseId of the exercise - * @return the {@link ResponseEntity} with status {@code 200} and with body the created solution entries, - */ - @PostMapping("programming-exercises/{exerciseId}/behavioral-solution-entries") - @EnforceAtLeastEditor - public ResponseEntity> createBehavioralSolutionEntries(@PathVariable Long exerciseId) { - log.debug("REST request to create behavioral solution entries"); - ProgrammingExercise exercise = programmingExerciseRepository.findByIdWithTemplateAndSolutionParticipationTeamAssignmentConfigCategoriesAndBuildConfigElseThrow(exerciseId); - authCheckService.checkHasAtLeastRoleForExerciseElseThrow(Role.EDITOR, exercise, null); - - try { - var solutionEntries = behavioralTestCaseService.generateBehavioralSolutionEntries(exercise); - return ResponseEntity.ok(solutionEntries); - } - catch (BehavioralSolutionEntryGenerationException e) { - log.error("Unable to create behavioral solution entries", e); - throw new InternalServerErrorException(e.getMessage()); - } - } - - private void checkExerciseContainsTestCaseElseThrow(ProgrammingExercise exercise, ProgrammingExerciseTestCase testCase) { - if (!exercise.getId().equals(testCase.getExercise().getId())) { - throw new ConflictException("The test case of the solution entry does not belong to the exercise.", ENTITY_NAME, "exerciseIdsMismatch"); - } - } - - private void checkTestCaseContainsSolutionEntryElseThrow(ProgrammingExerciseTestCase testCase, ProgrammingExerciseSolutionEntry solutionEntry) { - if (solutionEntry.getTestCase() == null || !testCase.getId().equals(solutionEntry.getTestCase().getId())) { - throw new ConflictException("The test case of the solution entry does not belong to the solution entry.", ENTITY_NAME, "solutionEntryError"); - } - } -} diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/localci/AeolusTemplateResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/localci/AeolusTemplateResource.java index 9e5530e94813..a20ce1b7c163 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/localci/AeolusTemplateResource.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/web/localci/AeolusTemplateResource.java @@ -21,13 +21,13 @@ import de.tum.cit.aet.artemis.core.security.annotations.EnforceAtLeastEditor; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.ProjectType; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; import de.tum.cit.aet.artemis.programming.service.BuildScriptProviderService; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusTemplateService; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; /** * Service for retrieving aeolus template files based on the programming language, project type, and - * the different options (static analysis, sequential runs, test coverage) as well as the default + * the different options (static analysis, sequential runs) as well as the default * image for the programming language and project type for the artemis instance. */ @Profile("aeolus | localci") @@ -61,21 +61,19 @@ public AeolusTemplateResource(AeolusTemplateService aeolusTemplateService, Build * @param projectType The project type for which the template file should be returned. If omitted, a default depending on the language will be used. * @param staticAnalysis Whether the static analysis template should be used * @param sequentialRuns Whether the sequential runs template should be used - * @param testCoverage Whether the test coverage template should be used * @return The requested file, or 404 if the file doesn't exist */ @GetMapping({ "templates/{language}/{projectType}", "templates/{language}" }) @EnforceAtLeastEditor public ResponseEntity getAeolusTemplate(@PathVariable ProgrammingLanguage language, @PathVariable Optional projectType, @RequestParam(value = "staticAnalysis", defaultValue = "false") boolean staticAnalysis, - @RequestParam(value = "sequentialRuns", defaultValue = "false") boolean sequentialRuns, - @RequestParam(value = "testCoverage", defaultValue = "false") boolean testCoverage) { - log.debug("REST request to get aeolus template for programming language {} and project type {}, static Analysis: {}, sequential Runs {}, testCoverage: {}", language, - projectType, staticAnalysis, sequentialRuns, testCoverage); + @RequestParam(value = "sequentialRuns", defaultValue = "false") boolean sequentialRuns) { + log.debug("REST request to get aeolus template for programming language {} and project type {}, static Analysis: {}, sequential Runs {}", language, projectType, + staticAnalysis, sequentialRuns); String projectTypePrefix = projectType.map(type -> type.name().toLowerCase()).orElse(""); - return getAeolusTemplateFileContentWithResponse(language, projectTypePrefix, staticAnalysis, sequentialRuns, testCoverage); + return getAeolusTemplateFileContentWithResponse(language, projectTypePrefix, staticAnalysis, sequentialRuns); } /** @@ -88,21 +86,19 @@ public ResponseEntity getAeolusTemplate(@PathVariable ProgrammingLanguag * @param projectType The project type for which the template file should be returned. If omitted, a default depending on the language will be used. * @param staticAnalysis Whether the static analysis template should be used * @param sequentialRuns Whether the sequential runs template should be used - * @param testCoverage Whether the test coverage template should be used * @return The requested file, or 404 if the file doesn't exist */ @GetMapping({ "template-scripts/{language}/{projectType}", "template-scripts/{language}" }) @EnforceAtLeastEditor public ResponseEntity getAeolusTemplateScript(@PathVariable ProgrammingLanguage language, @PathVariable Optional projectType, @RequestParam(value = "staticAnalysis", defaultValue = "false") boolean staticAnalysis, - @RequestParam(value = "sequentialRuns", defaultValue = "false") boolean sequentialRuns, - @RequestParam(value = "testCoverage", defaultValue = "false") boolean testCoverage) { - log.debug("REST request to get aeolus template for programming language {} and project type {}, static Analysis: {}, sequential Runs {}, testCoverage: {}", language, - projectType, staticAnalysis, sequentialRuns, testCoverage); + @RequestParam(value = "sequentialRuns", defaultValue = "false") boolean sequentialRuns) { + log.debug("REST request to get aeolus template script for programming language {} and project type {}, static Analysis: {}, sequential Runs {}", language, projectType, + staticAnalysis, sequentialRuns); String projectTypePrefix = projectType.map(type -> type.name().toLowerCase()).orElse(""); - return getAeolusTemplateScriptWithResponse(language, projectTypePrefix, staticAnalysis, sequentialRuns, testCoverage); + return getAeolusTemplateScriptWithResponse(language, projectTypePrefix, staticAnalysis, sequentialRuns); } /** @@ -115,17 +111,16 @@ public ResponseEntity getAeolusTemplateScript(@PathVariable ProgrammingL * @param projectTypePrefix The project type for which the template file should be returned. If omitted, a default depending on the language will be used. * @param staticAnalysis Whether the static analysis template should be used * @param sequentialRuns Whether the sequential runs template should be used - * @param testCoverage Whether the test coverage template should be used * @return The requested file, or 404 if the file doesn't exist */ - private ResponseEntity getAeolusTemplateFileContentWithResponse(ProgrammingLanguage language, String projectTypePrefix, boolean staticAnalysis, boolean sequentialRuns, - boolean testCoverage) { + private ResponseEntity getAeolusTemplateFileContentWithResponse(ProgrammingLanguage language, String projectTypePrefix, boolean staticAnalysis, + boolean sequentialRuns) { try { Optional optionalProjectType = Optional.empty(); if (!projectTypePrefix.isEmpty()) { optionalProjectType = Optional.of(ProjectType.valueOf(projectTypePrefix.toUpperCase())); } - Windfile windfile = aeolusTemplateService.getWindfileFor(language, optionalProjectType, staticAnalysis, sequentialRuns, testCoverage); + Windfile windfile = aeolusTemplateService.getWindfileFor(language, optionalProjectType, staticAnalysis, sequentialRuns); if (windfile == null) { return new ResponseEntity<>(null, null, HttpStatus.NOT_FOUND); } @@ -148,17 +143,15 @@ private ResponseEntity getAeolusTemplateFileContentWithResponse(Programm * @param projectTypePrefix The project type for which the template file should be returned. If omitted, a default depending on the language will be used. * @param staticAnalysis Whether the static analysis template should be used * @param sequentialRuns Whether the sequential runs template should be used - * @param testCoverage Whether the test coverage template should be used * @return The requested file, or 404 if the file doesn't exist */ - private ResponseEntity getAeolusTemplateScriptWithResponse(ProgrammingLanguage language, String projectTypePrefix, boolean staticAnalysis, boolean sequentialRuns, - boolean testCoverage) { + private ResponseEntity getAeolusTemplateScriptWithResponse(ProgrammingLanguage language, String projectTypePrefix, boolean staticAnalysis, boolean sequentialRuns) { try { Optional optionalProjectType = Optional.empty(); if (!projectTypePrefix.isEmpty()) { optionalProjectType = Optional.of(ProjectType.valueOf(projectTypePrefix.toUpperCase())); } - String script = buildScriptProviderService.getScriptFor(language, optionalProjectType, staticAnalysis, sequentialRuns, testCoverage); + String script = buildScriptProviderService.getScriptFor(language, optionalProjectType, staticAnalysis, sequentialRuns); HttpHeaders responseHeaders = new HttpHeaders(); responseHeaders.setContentType(MediaType.TEXT_PLAIN); return new ResponseEntity<>(script, responseHeaders, HttpStatus.OK); diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/repository/RepositoryProgrammingExerciseParticipationResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/repository/RepositoryProgrammingExerciseParticipationResource.java index 6cb9df0b10ee..28515ba970bb 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/repository/RepositoryProgrammingExerciseParticipationResource.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/web/repository/RepositoryProgrammingExerciseParticipationResource.java @@ -7,8 +7,6 @@ import java.util.Map; import java.util.Objects; import java.util.Optional; -import java.util.Set; -import java.util.stream.Collectors; import jakarta.servlet.http.HttpServletRequest; @@ -294,24 +292,6 @@ public ResponseEntity> getFilesWithContent(@PathVariable Lon }); } - /** - * GET /repository/{participationId}/file-names: Gets the file names of the repository - * - * @param participationId participation of the student/template/solution - * @return the ResponseEntity with status 200 (OK) and a set of file names - */ - @GetMapping(value = "repository/{participationId}/file-names", produces = MediaType.APPLICATION_JSON_VALUE) - @EnforceAtLeastTutor - public ResponseEntity> getFileNames(@PathVariable Long participationId) { - return super.executeAndCheckForExceptions(() -> { - Repository repository = getRepository(participationId, RepositoryActionType.READ, true); - var nonFolderFileNames = super.repositoryService.getFiles(repository).entrySet().stream().filter(mapEntry -> mapEntry.getValue().equals(FileType.FILE)) - .map(Map.Entry::getKey).collect(Collectors.toSet()); - - return new ResponseEntity<>(nonFolderFileNames, HttpStatus.OK); - }); - } - @Override @PostMapping(value = "repository/{participationId}/file", produces = MediaType.APPLICATION_JSON_VALUE) @FeatureToggle(Feature.ProgrammingExercises) diff --git a/src/main/resources/templates/aeolus/java/plain_gradle_static_coverage.sh b/src/main/resources/templates/aeolus/java/plain_gradle_static_coverage.sh deleted file mode 100644 index ee2bfdea3d9b..000000000000 --- a/src/main/resources/templates/aeolus/java/plain_gradle_static_coverage.sh +++ /dev/null @@ -1,34 +0,0 @@ -#!/usr/bin/env bash -set -e -export AEOLUS_INITIAL_DIRECTORY=${PWD} -tests () { - echo '⚙️ executing tests' - chmod +x ./gradlew - ./gradlew clean test tiaTests --run-all-tests -} - -static_code_analysis () { - echo '⚙️ executing static_code_analysis' - ./gradlew check -x test -} - -final_aeolus_post_action () { - set +e # from now on, we don't exit on errors - echo '⚙️ executing final_aeolus_post_action' - cd "${AEOLUS_INITIAL_DIRECTORY}" - static_code_analysis -} - -main () { - if [[ "${1}" == "aeolus_sourcing" ]]; then - return 0 # just source to use the methods in the subshell, no execution - fi - local _script_name - _script_name=${BASH_SOURCE[0]:-$0} - trap final_aeolus_post_action EXIT - - cd "${AEOLUS_INITIAL_DIRECTORY}" - bash -c "source ${_script_name} aeolus_sourcing; tests" -} - -main "${@}" diff --git a/src/main/resources/templates/aeolus/java/plain_gradle_static_coverage.yaml b/src/main/resources/templates/aeolus/java/plain_gradle_static_coverage.yaml deleted file mode 100644 index d9d319f2e49f..000000000000 --- a/src/main/resources/templates/aeolus/java/plain_gradle_static_coverage.yaml +++ /dev/null @@ -1,29 +0,0 @@ -api: v0.0.1 -actions: - - name: tests - script: |- - chmod +x ./gradlew - ./gradlew clean test tiaTests --run-all-tests - runAlways: false - - name: static_code_analysis - script: ./gradlew check -x test - runAlways: true - results: - - name: spotbugs - path: target/spotbugsXml.xml - type: static-code-analysis - - name: checkstyle - path: target/checkstyle-result.xml - type: static-code-analysis - - name: pmd - path: target/pmd.xml - type: static-code-analysis - - name: pmd_cpd - path: target/cpd.xml - type: static-code-analysis - - name: testwiseCoverageReport - path: build/reports/testwise-coverage/tiaTests/tiaTests.json - type: testwise-coverage - - name: junit_**/test-results/test/*.xml - path: '**/test-results/test/*.xml' - type: junit diff --git a/src/main/resources/templates/aeolus/java/plain_maven_static_coverage.sh b/src/main/resources/templates/aeolus/java/plain_maven_static_coverage.sh deleted file mode 100644 index d74afc686999..000000000000 --- a/src/main/resources/templates/aeolus/java/plain_maven_static_coverage.sh +++ /dev/null @@ -1,40 +0,0 @@ -#!/usr/bin/env bash -set -e -export AEOLUS_INITIAL_DIRECTORY=${PWD} -maven () { - echo '⚙️ executing maven' - mvn clean test -Pcoverage -} - -move_report_file () { - echo '⚙️ executing move_report_file' - mv target/tia/reports/*/testwise-coverage-*.json target/tia/reports/tiaTests.json -} - -maven_1 () { - echo '⚙️ executing maven_1' - mvn spotbugs:spotbugs checkstyle:checkstyle pmd:pmd pmd:cpd -} - -final_aeolus_post_action () { - set +e # from now on, we don't exit on errors - echo '⚙️ executing final_aeolus_post_action' - cd "${AEOLUS_INITIAL_DIRECTORY}" - maven_1 -} - -main () { - if [[ "${1}" == "aeolus_sourcing" ]]; then - return 0 # just source to use the methods in the subshell, no execution - fi - local _script_name - _script_name=${BASH_SOURCE[0]:-$0} - trap final_aeolus_post_action EXIT - - cd "${AEOLUS_INITIAL_DIRECTORY}" - bash -c "source ${_script_name} aeolus_sourcing; maven" - cd "${AEOLUS_INITIAL_DIRECTORY}" - bash -c "source ${_script_name} aeolus_sourcing; move_report_file" -} - -main "${@}" diff --git a/src/main/resources/templates/aeolus/java/plain_maven_static_coverage.yaml b/src/main/resources/templates/aeolus/java/plain_maven_static_coverage.yaml deleted file mode 100644 index 5017a23d8537..000000000000 --- a/src/main/resources/templates/aeolus/java/plain_maven_static_coverage.yaml +++ /dev/null @@ -1,30 +0,0 @@ -api: v0.0.1 -actions: - - name: maven - script: mvn clean test -Pcoverage - runAlways: false - - name: move_report_file - script: mv target/tia/reports/*/testwise-coverage-*.json target/tia/reports/tiaTests.json - runAlways: false - - name: maven - script: mvn spotbugs:spotbugs checkstyle:checkstyle pmd:pmd pmd:cpd - runAlways: true - results: - - name: spotbugs - path: target/spotbugsXml.xml - type: static-code-analysis - - name: checkstyle - path: target/checkstyle-result.xml - type: static-code-analysis - - name: pmd - path: target/pmd.xml - type: static-code-analysis - - name: pmd_cpd - path: target/cpd.xml - type: static-code-analysis - - name: testwiseCoverageReport - path: target/tia/reports/tiaTests.json - type: testwise-coverage - - name: junit - path: '**/target/surefire-reports/*.xml' - type: junit diff --git a/src/main/resources/templates/aeolus/kotlin/default_coverage.sh b/src/main/resources/templates/aeolus/kotlin/default_coverage.sh deleted file mode 100644 index 1840f600e00c..000000000000 --- a/src/main/resources/templates/aeolus/kotlin/default_coverage.sh +++ /dev/null @@ -1,26 +0,0 @@ -#!/usr/bin/env bash -set -e -export AEOLUS_INITIAL_DIRECTORY=${PWD} -maven () { - echo '⚙️ executing maven' - mvn clean test -Pcoverage -} - -move_report_file () { - echo '⚙️ executing move_report_file' - mv target/tia/reports/*/testwise-coverage-*.json target/tia/reports/tiaTests.json -} - -main () { - if [[ "${1}" == "aeolus_sourcing" ]]; then - return 0 # just source to use the methods in the subshell, no execution - fi - local _script_name - _script_name=${BASH_SOURCE[0]:-$0} - cd "${AEOLUS_INITIAL_DIRECTORY}" - bash -c "source ${_script_name} aeolus_sourcing; maven" - cd "${AEOLUS_INITIAL_DIRECTORY}" - bash -c "source ${_script_name} aeolus_sourcing; move_report_file" -} - -main "${@}" diff --git a/src/main/resources/templates/aeolus/kotlin/default_coverage.yaml b/src/main/resources/templates/aeolus/kotlin/default_coverage.yaml deleted file mode 100644 index 5f2c972b6223..000000000000 --- a/src/main/resources/templates/aeolus/kotlin/default_coverage.yaml +++ /dev/null @@ -1,15 +0,0 @@ -api: v0.0.1 -actions: - - name: maven - script: mvn clean test -Pcoverage - runAlways: false - - name: move_report_file - script: mv target/tia/reports/*/testwise-coverage-*.json target/tia/reports/tiaTests.json - runAlways: false - results: - - name: testwiseCoverageReport - path: target/tia/reports/tiaTests.json - before: false - - name: junit - path: '**/target/surefire-reports/*.xml' - type: junit diff --git a/src/main/resources/templates/iris/hestia.hbs b/src/main/resources/templates/iris/hestia.hbs deleted file mode 100644 index 615534e452a7..000000000000 --- a/src/main/resources/templates/iris/hestia.hbs +++ /dev/null @@ -1,80 +0,0 @@ -{{#system~}}# Your task -You are an assistant to generate long and short descriptions for code hints. -Code hints are hints for programming exercises. -Code hints contain a part of the sample solution code that is needed to solve the exercise. -Code hints contain a short description of the code hint, which describes the purpose of the code hint. -Code hints contain a long description of the code hint, which describes the solution of the hint. -Code hints are made available to the student automatically if they have trouble solving the exercise. -The short description of the code hint is displayed to the student as soon as it is available. -The long description and solution code of the code hint is displayed to the student if they decide to use the code hint. -Your task is to generate the short and long description of a code hint. - -# Context -This is the context of the exercise and code hint that you can use to generate the short and long description. -## Programming exercise -Information about the programming exercise. -### Title -{{exercise.title}} -### Problem statement -```markdown -{{exercise.problemStatement}} -``` -## Code hint -Information about the code hint. -### Title -{{codeHint.title}} -### Solution code -{{#each codeHint.solutionEntries}} -#### {{this.filePath}} -{{#if (contains this "previousLine")}} -**Previous line:** {{this.previousLine}} -**Previous code:** -``` -{{this.previousCode}} -``` -{{/if}} -**New Line:** {{this.line}} -**New code:** -``` -{{this.code}} -``` -{{~/each}} - -## Previous attempts -These are the previous attempts at creating descriptions for this code hint. -The instructor did not like the previous attempts and wants you to generate new and different descriptions. -{{~/system}} -{{#if (contains session "messages")}} - {{#each session.messages}} - {{#if (equal this.sender "LLM")}} - {{#if (equal this.content[0].type "json")}} - {{#assistant~}} - **Short description:** - {{this.content[0].attributes.shortDescription}} - **Long description:** - {{this.content[0].attributes.longDescription}} - {{~/assistant}} - {{/if}} - {{#if (equal this.content[0].type "text")}} - {{#assistant~}} - {{this.content[0].textContent}} - {{~/assistant}} - {{/if}} - {{/if}} - {{~/each}} -{{/if}} -{{#system~}} - -# Your solution -Create your solution here. You can use any markdown you want -## Short description -{{~/system}} -{{#assistant~}} -{{gen 'shortDescription' temperature=0.2 max_tokens=100 stop="\\n"}} -{{~/assistant}} -{{#system~}} -## Long description -{{~/system}} -{{#assistant~}} -{{gen 'longDescription' temperature=0.2 max_tokens=1000}} -{{~/assistant}} diff --git a/src/main/resources/templates/java/maven_maven/test/projectTemplate/pom.xml b/src/main/resources/templates/java/maven_maven/test/projectTemplate/pom.xml index 997728be058b..93720b6f3aee 100644 --- a/src/main/resources/templates/java/maven_maven/test/projectTemplate/pom.xml +++ b/src/main/resources/templates/java/maven_maven/test/projectTemplate/pom.xml @@ -88,14 +88,6 @@ ${project.build.outputDirectory}/sun/ ${project.build.outputDirectory}/org/gradle/ ${project.build.outputDirectory}/worker/org/gradle/ - - ${project.build.outputDirectory}/com/squareup/ - ${project.build.outputDirectory}/com/teamscale/ - ${project.build.outputDirectory}/okhttp3/ - ${project.build.outputDirectory}/okio/ - ${project.build.outputDirectory}/retrofit2/ - ${project.build.outputDirectory}/shadow/ - @@ -180,54 +172,4 @@ - - - - coverage - - - com.teamscale - impacted-test-engine - 33.1.2 - - - - - - org.apache.maven.plugins - maven-surefire-plugin - 3.2.5 - - - okhttp3,com.teamscale,retrofit2,shadow,com.squareup,okio - - - - - com.teamscale - teamscale-maven-plugin - 33.1.2 - - http://localhost - dummy - ${exerciseNamePomXml}-Tests - dummy - testwise - - *${packageName}* - - - - - - prepare-tia-unit-test - - - - - - - - - diff --git a/src/main/resources/templates/java/test/gradle/projectTemplate/build.gradle b/src/main/resources/templates/java/test/gradle/projectTemplate/build.gradle index 97be1ab67bfd..b099c0b33b67 100644 --- a/src/main/resources/templates/java/test/gradle/projectTemplate/build.gradle +++ b/src/main/resources/templates/java/test/gradle/projectTemplate/build.gradle @@ -4,9 +4,7 @@ plugins { id 'pmd' id 'com.github.spotbugs' version '6.0.9' // %static-code-analysis-stop% - // %record-testwise-coverage-start% id 'com.teamscale' version '33.1.2' - // %record-testwise-coverage-stop% } apply plugin: 'java' @@ -97,14 +95,6 @@ def forbiddenPackageFolders = [ //(2) "$studentOutputDir/org/opentest4j/", "$studentOutputDir/sun/", "$studentOutputDir/worker/org/gradle/" - // %record-testwise-coverage-start% - ,"$studentOutputDir/com/teamscale/", - "$studentOutputDir/okhttp3/", - "$studentOutputDir/retrofit2/", - "$studentOutputDir/shadow/", - "$studentOutputDir/com/squareup/", - "$studentOutputDir/okio/" - // %record-testwise-coverage-stop% ] test { doFirst { //(1) @@ -178,20 +168,3 @@ pmd { } } // %static-code-analysis-stop% - - -// %record-testwise-coverage-start% -tasks.register('tiaTests', com.teamscale.TestImpacted) { - systemProperty "ares.security.trustedpackages", "okhttp3,com.teamscale,retrofit2,shadow,com.squareup,okio" - useJUnitPlatform() - filter { - excludeTestsMatching "AttributeTest" - excludeTestsMatching "ConstructorTest" - excludeTestsMatching "ClassTest" - excludeTestsMatching "MethodTest" - } - jacoco { - includes = ["${packageName}.*"] - } -} -// %record-testwise-coverage-stop% diff --git a/src/main/resources/templates/java/test/maven/projectTemplate/pom.xml b/src/main/resources/templates/java/test/maven/projectTemplate/pom.xml index c1a7d42fd3e8..24f9561bc6e9 100644 --- a/src/main/resources/templates/java/test/maven/projectTemplate/pom.xml +++ b/src/main/resources/templates/java/test/maven/projectTemplate/pom.xml @@ -83,14 +83,6 @@ ${project.build.outputDirectory}/sun/ ${project.build.outputDirectory}/org/gradle/ ${project.build.outputDirectory}/worker/org/gradle/ - - ${project.build.outputDirectory}/com/squareup/ - ${project.build.outputDirectory}/com/teamscale/ - ${project.build.outputDirectory}/okhttp3/ - ${project.build.outputDirectory}/okio/ - ${project.build.outputDirectory}/retrofit2/ - ${project.build.outputDirectory}/shadow/ - @@ -175,54 +167,4 @@ - - - - coverage - - - com.teamscale - impacted-test-engine - 33.1.2 - - - - - - org.apache.maven.plugins - maven-surefire-plugin - 3.2.5 - - - okhttp3,com.teamscale,retrofit2,shadow,com.squareup,okio - - - - - com.teamscale - teamscale-maven-plugin - 33.1.2 - - http://localhost - dummy - ${exerciseNamePomXml}-Tests - dummy - testwise - - *${packageName}* - - - - - - prepare-tia-unit-test - - - - - - - - - diff --git a/src/main/resources/templates/jenkins/java/gradle/regularRuns/pipeline.groovy b/src/main/resources/templates/jenkins/java/gradle/regularRuns/pipeline.groovy index b83202251c06..2478c6a8fe40 100644 --- a/src/main/resources/templates/jenkins/java/gradle/regularRuns/pipeline.groovy +++ b/src/main/resources/templates/jenkins/java/gradle/regularRuns/pipeline.groovy @@ -12,7 +12,6 @@ dockerImage = '#dockerImage' dockerFlags = '#dockerArgs' isSolutionBuild = "${env.JOB_NAME}" ==~ /.+-SOLUTION$/ -isTestWiseCoverageEnabled = #testWiseCoverage && isSolutionBuild isStaticCodeAnalysisEnabled = #isStaticCodeAnalysisEnabled /** @@ -37,11 +36,7 @@ private void runTestSteps() { */ void test() { stage('Test') { - if (isTestWiseCoverageEnabled) { - sh './gradlew clean test tiaTests --run-all-tests' - } else { - sh './gradlew clean test' - } + sh './gradlew clean test' } } @@ -66,25 +61,12 @@ private void staticCodeAnalysis() { } } -private void collectTestwiseCoverageReport() { - catchError { - sh ''' - rm -rf testwiseCoverageReport - mkdir testwiseCoverageReport - mv build/reports/testwise-coverage/tiaTests/tiaTests.json testwiseCoverageReport/ - ''' - } -} - /** * Script of the post build tasks aggregating all JUnit files in $WORKSPACE/results. * * Called by Jenkins. */ void postBuildTasks() { - if (isTestWiseCoverageEnabled) { - collectTestwiseCoverageReport() - } sh ''' rm -rf results mkdir results diff --git a/src/main/resources/templates/jenkins/java/maven/regularRuns/pipeline.groovy b/src/main/resources/templates/jenkins/java/maven/regularRuns/pipeline.groovy index cb76cbb4ee84..326b5897fbaa 100644 --- a/src/main/resources/templates/jenkins/java/maven/regularRuns/pipeline.groovy +++ b/src/main/resources/templates/jenkins/java/maven/regularRuns/pipeline.groovy @@ -12,7 +12,6 @@ dockerImage = '#dockerImage' dockerFlags = '#dockerArgs' isSolutionBuild = "${env.JOB_NAME}" ==~ /.+-SOLUTION$/ -isTestwiseCoverageEnabled = #testWiseCoverage && isSolutionBuild isStaticCodeAnalysisEnabled = #isStaticCodeAnalysisEnabled /** @@ -37,11 +36,7 @@ private void runTestSteps() { */ private void test() { stage('Test') { - if (isTestwiseCoverageEnabled) { - sh 'mvn clean test -B -Pcoverage' - } else { - sh 'mvn clean test -B' - } + sh 'mvn clean test -B' } } @@ -66,25 +61,12 @@ private void staticCodeAnalysis() { } } -private void collectTestwiseCoverageReport() { - catchError { - sh ''' - rm -rf testwiseCoverageReport - mkdir testwiseCoverageReport - mv target/tia/reports/*/*.json testwiseCoverageReport/ - ''' - } -} - /** * Script of the post build tasks aggregating all JUnit files in $WORKSPACE/results. * * Called by Jenkins. */ void postBuildTasks() { - if (isTestwiseCoverageEnabled) { - collectTestwiseCoverageReport() - } sh ''' rm -rf results mkdir results diff --git a/src/main/resources/templates/jenkins/kotlin/regularRuns/pipeline.groovy b/src/main/resources/templates/jenkins/kotlin/regularRuns/pipeline.groovy index cb76cbb4ee84..326b5897fbaa 100644 --- a/src/main/resources/templates/jenkins/kotlin/regularRuns/pipeline.groovy +++ b/src/main/resources/templates/jenkins/kotlin/regularRuns/pipeline.groovy @@ -12,7 +12,6 @@ dockerImage = '#dockerImage' dockerFlags = '#dockerArgs' isSolutionBuild = "${env.JOB_NAME}" ==~ /.+-SOLUTION$/ -isTestwiseCoverageEnabled = #testWiseCoverage && isSolutionBuild isStaticCodeAnalysisEnabled = #isStaticCodeAnalysisEnabled /** @@ -37,11 +36,7 @@ private void runTestSteps() { */ private void test() { stage('Test') { - if (isTestwiseCoverageEnabled) { - sh 'mvn clean test -B -Pcoverage' - } else { - sh 'mvn clean test -B' - } + sh 'mvn clean test -B' } } @@ -66,25 +61,12 @@ private void staticCodeAnalysis() { } } -private void collectTestwiseCoverageReport() { - catchError { - sh ''' - rm -rf testwiseCoverageReport - mkdir testwiseCoverageReport - mv target/tia/reports/*/*.json testwiseCoverageReport/ - ''' - } -} - /** * Script of the post build tasks aggregating all JUnit files in $WORKSPACE/results. * * Called by Jenkins. */ void postBuildTasks() { - if (isTestwiseCoverageEnabled) { - collectTestwiseCoverageReport() - } sh ''' rm -rf results mkdir results diff --git a/src/main/resources/templates/kotlin/test/maven/projectTemplate/pom.xml b/src/main/resources/templates/kotlin/test/maven/projectTemplate/pom.xml index e278d59eb3d2..c1c04e44339e 100644 --- a/src/main/resources/templates/kotlin/test/maven/projectTemplate/pom.xml +++ b/src/main/resources/templates/kotlin/test/maven/projectTemplate/pom.xml @@ -138,14 +138,6 @@ ${project.build.outputDirectory}/sun/ ${project.build.outputDirectory}/org/gradle/ ${project.build.outputDirectory}/worker/org/gradle/ - - ${project.build.outputDirectory}/com/squareup/ - ${project.build.outputDirectory}/com/teamscale/ - ${project.build.outputDirectory}/okhttp3/ - ${project.build.outputDirectory}/okio/ - ${project.build.outputDirectory}/retrofit2/ - ${project.build.outputDirectory}/shadow/ - @@ -153,54 +145,4 @@ - - - - coverage - - - com.teamscale - impacted-test-engine - ${teamscale.version} - - - - - - org.apache.maven.plugins - maven-surefire-plugin - 3.2.5 - - - okhttp3,com.teamscale,retrofit2,shadow,com.squareup,okio - - - - - com.teamscale - teamscale-maven-plugin - ${teamscale.version} - - http://localhost - dummy - ${exerciseNamePomXml}-Tests - dummy - testwise - - *${packageName}* - - - - - - prepare-tia-unit-test - - - - - - - - - diff --git a/src/main/webapp/app/course/manage/course-management.module.ts b/src/main/webapp/app/course/manage/course-management.module.ts index 12d386095e31..68b0ce410db3 100644 --- a/src/main/webapp/app/course/manage/course-management.module.ts +++ b/src/main/webapp/app/course/manage/course-management.module.ts @@ -20,7 +20,6 @@ import { ArtemisLectureModule } from 'app/lecture/lecture.module'; import { ArtemisTextExerciseManagementModule } from 'app/exercises/text/manage/text-exercise-management.module'; import { ArtemisDashboardsModule } from 'app/shared/dashboards/dashboards.module'; import { ArtemisParticipationModule } from 'app/exercises/shared/participation/participation.module'; -import { ArtemisExerciseHintManagementModule } from 'app/exercises/shared/exercise-hint/manage/exercise-hint-management.module'; import { ArtemisModelingExerciseManagementModule } from 'app/exercises/modeling/manage/modeling-exercise-management.module'; import { ArtemisCourseScoresModule } from 'app/course/course-scores/course-scores.module'; import { ArtemisExerciseScoresModule } from 'app/exercises/shared/exercise-scores/exercise-scores.module'; @@ -95,7 +94,6 @@ import { ArtemisMarkdownEditorModule } from 'app/shared/markdown-editor/markdown ArtemisModelingExerciseModule, ArtemisColorSelectorModule, ArtemisDashboardsModule, - ArtemisExerciseHintManagementModule, ArtemisParticipationModule, ArtemisComplaintsForTutorModule, ArtemisListOfComplaintsModule, diff --git a/src/main/webapp/app/course/manage/course-update.component.ts b/src/main/webapp/app/course/manage/course-update.component.ts index fd7c49954ea4..8922497168dd 100644 --- a/src/main/webapp/app/course/manage/course-update.component.ts +++ b/src/main/webapp/app/course/manage/course-update.component.ts @@ -128,7 +128,7 @@ export class CourseUpdateComponent implements OnInit { this.course.maxComplaintResponseTextLimit! > 0; this.requestMoreFeedbackEnabled = this.course.maxRequestMoreFeedbackTimeDays! > 0; } else { - this.fileService.getTemplateCodeOfCondcut().subscribe({ + this.fileService.getTemplateCodeOfConduct().subscribe({ next: (res: HttpResponse) => { if (res.body) { this.course.courseInformationSharingMessagingCodeOfConduct = res.body; diff --git a/src/main/webapp/app/course/manage/detail/course-detail.component.ts b/src/main/webapp/app/course/manage/detail/course-detail.component.ts index c6a3e11211ba..4b5ecc80b629 100644 --- a/src/main/webapp/app/course/manage/detail/course-detail.component.ts +++ b/src/main/webapp/app/course/manage/detail/course-detail.component.ts @@ -222,12 +222,6 @@ export class CourseDetailComponent implements OnInit, OnDestroy { data: { course: this.course, disabled: !this.isAdmin, subSettingsType: IrisSubSettingsType.CHAT }, }); } - // TODO: Enable in future PR - // details.push({ - // type: DetailType.ProgrammingIrisEnabled, - // title: 'artemisApp.iris.settings.subSettings.enabled.hesita', - // data: { course: this.course, disabled: !this.isAdmin, subSettingsType: this.HESTIA }, - // }); return irisDetails; } diff --git a/src/main/webapp/app/detail-overview-list/components/programming-diff-report-detail/programming-diff-report-detail.component.ts b/src/main/webapp/app/detail-overview-list/components/programming-diff-report-detail/programming-diff-report-detail.component.ts index 94fb86848f4f..46999aa24a27 100644 --- a/src/main/webapp/app/detail-overview-list/components/programming-diff-report-detail/programming-diff-report-detail.component.ts +++ b/src/main/webapp/app/detail-overview-list/components/programming-diff-report-detail/programming-diff-report-detail.component.ts @@ -3,12 +3,12 @@ import type { ProgrammingDiffReportDetail } from 'app/detail-overview-list/detai import { FeatureToggle } from 'app/shared/feature-toggle/feature-toggle.service'; import { ButtonSize, ButtonType, TooltipPlacement } from 'app/shared/components/button.component'; import { faCodeCompare } from '@fortawesome/free-solid-svg-icons'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; -import { GitDiffReportModalComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report-modal.component'; +import { ProgrammingExerciseGitDiffReport } from 'app/entities/programming-exercise-git-diff-report.model'; +import { GitDiffReportModalComponent } from 'app/exercises/programming/git-diff-report/git-diff-report-modal.component'; import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; import { ArtemisSharedModule } from 'app/shared/shared.module'; import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap'; -import { GitDiffLineStatComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component'; +import { GitDiffLineStatComponent } from 'app/exercises/programming/git-diff-report/git-diff-line-stat.component'; @Component({ selector: 'jhi-programming-diff-report-detail', diff --git a/src/main/webapp/app/detail-overview-list/detail.model.ts b/src/main/webapp/app/detail-overview-list/detail.model.ts index b34f408ef435..995f32024195 100644 --- a/src/main/webapp/app/detail-overview-list/detail.model.ts +++ b/src/main/webapp/app/detail-overview-list/detail.model.ts @@ -5,7 +5,7 @@ import { SolutionProgrammingExerciseParticipation } from 'app/entities/participa import { ProgrammingExerciseInstructorRepositoryType } from 'app/exercises/programming/manage/services/programming-exercise.service'; import { AuxiliaryRepository } from 'app/entities/programming/programming-exercise-auxiliary-repository-model'; import { ProgrammingExerciseParticipationType } from 'app/entities/programming/programming-exercise-participation.model'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from 'app/entities/programming-exercise-git-diff-report.model'; import { BuildLogStatisticsDTO } from 'app/entities/programming/build-log-statistics-dto'; import { DetailType } from 'app/detail-overview-list/detail-overview-list.component'; import { SafeHtml } from '@angular/platform-browser'; diff --git a/src/main/webapp/app/entities/hestia/code-hint-model.ts b/src/main/webapp/app/entities/hestia/code-hint-model.ts deleted file mode 100644 index 310a0a5839a2..000000000000 --- a/src/main/webapp/app/entities/hestia/code-hint-model.ts +++ /dev/null @@ -1,12 +0,0 @@ -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; - -export enum CodeHintGenerationStep { - GIT_DIFF, - COVERAGE, - SOLUTION_ENTRIES, - CODE_HINTS, -} -export class CodeHint extends ExerciseHint { - public solutionEntries?: ProgrammingExerciseSolutionEntry[]; -} diff --git a/src/main/webapp/app/entities/hestia/coverage-file-report.model.ts b/src/main/webapp/app/entities/hestia/coverage-file-report.model.ts deleted file mode 100644 index 03c3b278b6e6..000000000000 --- a/src/main/webapp/app/entities/hestia/coverage-file-report.model.ts +++ /dev/null @@ -1,13 +0,0 @@ -import { BaseEntity } from 'app/shared/model/base-entity'; -import { TestwiseCoverageReportEntry } from 'app/entities/hestia/testwise-coverage-report-entry.model'; - -export class CoverageFileReport implements BaseEntity { - public id?: number; - - public filePath?: string; - public lineCount?: number; - public coveredLineCount?: number; - public testwiseCoverageEntries?: TestwiseCoverageReportEntry[]; - - constructor() {} -} diff --git a/src/main/webapp/app/entities/hestia/coverage-report.model.ts b/src/main/webapp/app/entities/hestia/coverage-report.model.ts deleted file mode 100644 index c60906c135d0..000000000000 --- a/src/main/webapp/app/entities/hestia/coverage-report.model.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { BaseEntity } from 'app/shared/model/base-entity'; -import { CoverageFileReport } from 'app/entities/hestia/coverage-file-report.model'; - -export class CoverageReport implements BaseEntity { - public id?: number; - - public fileReports?: CoverageFileReport[]; - public coveredLineRatio?: number; - - constructor() {} -} diff --git a/src/main/webapp/app/entities/hestia/exercise-hint.model.ts b/src/main/webapp/app/entities/hestia/exercise-hint.model.ts deleted file mode 100644 index fa61b885b7ec..000000000000 --- a/src/main/webapp/app/entities/hestia/exercise-hint.model.ts +++ /dev/null @@ -1,20 +0,0 @@ -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { BaseEntity } from 'app/shared/model/base-entity'; -import { ProgrammingExerciseServerSideTask } from 'app/entities/hestia/programming-exercise-task.model'; - -export enum HintType { - TEXT = 'text', - CODE = 'code', -} - -export class ExerciseHint implements BaseEntity { - public id?: number; - public title?: string; - public description?: string; - public content?: string; - public exercise?: ProgrammingExercise; - public type?: HintType; - public programmingExerciseTask?: ProgrammingExerciseServerSideTask; - public displayThreshold?: number; - public currentUserRating?: number; -} diff --git a/src/main/webapp/app/entities/hestia/programming-exercise-solution-entry.model.ts b/src/main/webapp/app/entities/hestia/programming-exercise-solution-entry.model.ts deleted file mode 100644 index 9380ec858a4f..000000000000 --- a/src/main/webapp/app/entities/hestia/programming-exercise-solution-entry.model.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { BaseEntity } from 'app/shared/model/base-entity'; -import { ProgrammingExerciseTestCase } from 'app/entities/programming/programming-exercise-test-case.model'; -import { CodeHint } from 'app/entities/hestia/code-hint-model'; - -export class ProgrammingExerciseSolutionEntry implements BaseEntity { - id?: number; - filePath?: string; - previousLine?: number; - line?: number; - previousCode?: string; - code?: string; - testCase?: ProgrammingExerciseTestCase; - codeHint?: CodeHint; -} diff --git a/src/main/webapp/app/entities/hestia/testwise-coverage-report-entry.model.ts b/src/main/webapp/app/entities/hestia/testwise-coverage-report-entry.model.ts deleted file mode 100644 index 59234fffbcfe..000000000000 --- a/src/main/webapp/app/entities/hestia/testwise-coverage-report-entry.model.ts +++ /dev/null @@ -1,12 +0,0 @@ -import { BaseEntity } from 'app/shared/model/base-entity'; -import { ProgrammingExerciseTestCase } from 'app/entities/programming/programming-exercise-test-case.model'; - -export class TestwiseCoverageReportEntry implements BaseEntity { - public id?: number; - - public startLine?: number; - public lineCount?: number; - public testCase?: ProgrammingExerciseTestCase; - - constructor() {} -} diff --git a/src/main/webapp/app/entities/hestia/programming-exercise-git-diff-entry.model.ts b/src/main/webapp/app/entities/programming-exercise-git-diff-entry.model.ts similarity index 100% rename from src/main/webapp/app/entities/hestia/programming-exercise-git-diff-entry.model.ts rename to src/main/webapp/app/entities/programming-exercise-git-diff-entry.model.ts diff --git a/src/main/webapp/app/entities/hestia/programming-exercise-git-diff-report.model.ts b/src/main/webapp/app/entities/programming-exercise-git-diff-report.model.ts similarity index 85% rename from src/main/webapp/app/entities/hestia/programming-exercise-git-diff-report.model.ts rename to src/main/webapp/app/entities/programming-exercise-git-diff-report.model.ts index 5fccf4e2f4aa..d688911d1ccd 100644 --- a/src/main/webapp/app/entities/hestia/programming-exercise-git-diff-report.model.ts +++ b/src/main/webapp/app/entities/programming-exercise-git-diff-report.model.ts @@ -1,6 +1,6 @@ import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; import { BaseEntity } from 'app/shared/model/base-entity'; -import { ProgrammingExerciseGitDiffEntry } from 'app/entities/hestia/programming-exercise-git-diff-entry.model'; +import { ProgrammingExerciseGitDiffEntry } from 'app/entities/programming-exercise-git-diff-entry.model'; export class ProgrammingExerciseGitDiffReport implements BaseEntity { public id?: number; diff --git a/src/main/webapp/app/entities/hestia/programming-exercise-task.model.ts b/src/main/webapp/app/entities/programming-exercise-task.model.ts similarity index 100% rename from src/main/webapp/app/entities/hestia/programming-exercise-task.model.ts rename to src/main/webapp/app/entities/programming-exercise-task.model.ts diff --git a/src/main/webapp/app/entities/programming/build-config.model.ts b/src/main/webapp/app/entities/programming/build-config.model.ts index dcae29ec591f..af71acf37afe 100644 --- a/src/main/webapp/app/entities/programming/build-config.model.ts +++ b/src/main/webapp/app/entities/programming/build-config.model.ts @@ -6,6 +6,5 @@ export class BuildConfig { public projectType?: string; public scaEnabled?: boolean; public sequentialTestRunsEnabled?: boolean; - public testwiseCoverageEnabled?: boolean; public resultPaths?: string[]; } diff --git a/src/main/webapp/app/entities/programming/programming-exercise-build.config.ts b/src/main/webapp/app/entities/programming/programming-exercise-build.config.ts index 55e5f5f487e2..d82e5f345c4d 100644 --- a/src/main/webapp/app/entities/programming/programming-exercise-build.config.ts +++ b/src/main/webapp/app/entities/programming/programming-exercise-build.config.ts @@ -11,11 +11,9 @@ export class ProgrammingExerciseBuildConfig { public timeoutSeconds?: number; public dockerFlags?: string; public windfile?: WindFile; - public testwiseCoverageEnabled?: boolean; public theiaImage?: string; constructor() { this.checkoutSolutionRepository = false; // default value - this.testwiseCoverageEnabled = false; // default value } } diff --git a/src/main/webapp/app/entities/programming/programming-exercise-test-case.model.ts b/src/main/webapp/app/entities/programming/programming-exercise-test-case.model.ts index 2d6822dc7ce4..5d43e28c7d16 100644 --- a/src/main/webapp/app/entities/programming/programming-exercise-test-case.model.ts +++ b/src/main/webapp/app/entities/programming/programming-exercise-test-case.model.ts @@ -1,6 +1,5 @@ import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; import { BaseEntity } from 'app/shared/model/base-entity'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; import { TestCaseStats } from './programming-exercise-test-case-statistics.model'; export enum Visibility { @@ -25,7 +24,6 @@ export class ProgrammingExerciseTestCase implements BaseEntity { visibility?: Visibility; exercise?: ProgrammingExercise; type?: ProgrammingExerciseTestCaseType; - solutionEntries?: ProgrammingExerciseSolutionEntry[]; // Utility information that is not actually part of the object stored on the server resultingPoints?: number; diff --git a/src/main/webapp/app/entities/programming/programming-exercise.model.ts b/src/main/webapp/app/entities/programming/programming-exercise.model.ts index 646f7bdfdf4d..bd9246f8569e 100644 --- a/src/main/webapp/app/entities/programming/programming-exercise.model.ts +++ b/src/main/webapp/app/entities/programming/programming-exercise.model.ts @@ -2,8 +2,7 @@ import { AssessmentType } from 'app/entities/assessment-type.model'; import { Course } from 'app/entities/course.model'; import { ExerciseGroup } from 'app/entities/exercise-group.model'; import { Exercise, ExerciseType, resetForImport } from 'app/entities/exercise.model'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from 'app/entities/programming-exercise-git-diff-report.model'; import { SolutionProgrammingExerciseParticipation } from 'app/entities/participation/solution-programming-exercise-participation.model'; import { TemplateProgrammingExerciseParticipation } from 'app/entities/participation/template-programming-exercise-participation.model'; import { BuildLogStatisticsDTO } from 'app/entities/programming/build-log-statistics-dto'; @@ -63,7 +62,6 @@ export class ProgrammingExercise extends Exercise { public showTestNamesToStudents?: boolean; public auxiliaryRepositories?: AuxiliaryRepository[]; public submissionPolicy?: SubmissionPolicy; - public exerciseHints?: ExerciseHint[]; public gitDiffReport?: ProgrammingExerciseGitDiffReport; public buildLogStatistics?: BuildLogStatisticsDTO; public buildConfig?: ProgrammingExerciseBuildConfig; @@ -74,11 +72,6 @@ export class ProgrammingExercise extends Exercise { public projectType?: ProjectType; - // helper attributes - - // this attribute is used to display the covered lines ratio - public coveredLinesRatio?: number; - /** * This attribute is used to generate a programming exercise with no connection to the VCS and CI. * This functionality is only for testing purposes. @@ -126,8 +119,6 @@ export function copyBuildConfigFromExerciseJson(exerciseJson: ProgrammingExercis buildConfig.timeoutSeconds = exerciseJson.timeoutSeconds ?? 0; buildConfig.windfile = exerciseJson.windfile ?? undefined; buildConfig.buildScript = exerciseJson.buildScript ?? ''; - buildConfig.testwiseCoverageEnabled = exerciseJson.testwiseCoverageEnabled ?? false; buildConfig.dockerFlags = exerciseJson.dockerFlags ?? ''; - return buildConfig; } diff --git a/src/main/webapp/app/exam/manage/exam-management.module.ts b/src/main/webapp/app/exam/manage/exam-management.module.ts index 42b34266ba2b..5590adf04e27 100644 --- a/src/main/webapp/app/exam/manage/exam-management.module.ts +++ b/src/main/webapp/app/exam/manage/exam-management.module.ts @@ -67,7 +67,7 @@ import { ArtemisProgrammingExerciseModule } from 'app/exercises/programming/shar import { DetailModule } from 'app/detail-overview-list/detail.module'; import { ArtemisDurationFromSecondsPipe } from 'app/shared/pipes/artemis-duration-from-seconds.pipe'; import { NoDataComponent } from 'app/shared/no-data-component'; -import { GitDiffLineStatComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component'; +import { GitDiffLineStatComponent } from 'app/exercises/programming/git-diff-report/git-diff-line-stat.component'; const ENTITY_STATES = [...examManagementState]; diff --git a/src/main/webapp/app/exam/manage/student-exams/student-exam-timeline/programming-exam-diff/programming-exercise-exam-diff.component.ts b/src/main/webapp/app/exam/manage/student-exams/student-exam-timeline/programming-exam-diff/programming-exercise-exam-diff.component.ts index 81a3f247a809..c3979797994f 100644 --- a/src/main/webapp/app/exam/manage/student-exams/student-exam-timeline/programming-exam-diff/programming-exercise-exam-diff.component.ts +++ b/src/main/webapp/app/exam/manage/student-exams/student-exam-timeline/programming-exam-diff/programming-exercise-exam-diff.component.ts @@ -3,7 +3,7 @@ import { ProgrammingExercise } from 'app/entities/programming/programming-exerci import { ProgrammingSubmission } from 'app/entities/programming/programming-submission.model'; import { FeatureToggle } from 'app/shared/feature-toggle/feature-toggle.service'; import { ButtonSize } from 'app/shared/components/button.component'; -import { GitDiffReportModalComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report-modal.component'; +import { GitDiffReportModalComponent } from 'app/exercises/programming/git-diff-report/git-diff-report-modal.component'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; import { Exercise, IncludedInOverallScore } from 'app/entities/exercise.model'; @@ -11,7 +11,7 @@ import { ExamSubmissionComponent } from 'app/exam/participate/exercises/exam-sub import { Submission } from 'app/entities/submission.model'; import { ProgrammingExerciseStudentParticipation } from 'app/entities/participation/programming-exercise-student-participation.model'; import { faCodeCompare } from '@fortawesome/free-solid-svg-icons'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from 'app/entities/programming-exercise-git-diff-report.model'; import { ExamPageComponent } from 'app/exam/participate/exercises/exam-page.component'; import { Observable, Subject, Subscription, debounceTime, take } from 'rxjs'; import { CachedRepositoryFilesService } from 'app/exercises/programming/manage/services/cached-repository-files.service'; diff --git a/src/main/webapp/app/exam/manage/student-exams/student-exam-timeline/student-exam-timeline.component.ts b/src/main/webapp/app/exam/manage/student-exams/student-exam-timeline/student-exam-timeline.component.ts index 70f4cfe76710..ac7f501a52d5 100644 --- a/src/main/webapp/app/exam/manage/student-exams/student-exam-timeline/student-exam-timeline.component.ts +++ b/src/main/webapp/app/exam/manage/student-exams/student-exam-timeline/student-exam-timeline.component.ts @@ -17,7 +17,7 @@ import { SubmissionVersionService } from 'app/exercises/shared/submission-versio import { ProgrammingExerciseExamDiffComponent } from 'app/exam/manage/student-exams/student-exam-timeline/programming-exam-diff/programming-exercise-exam-diff.component'; import { ProgrammingExerciseParticipationService } from 'app/exercises/programming/manage/services/programming-exercise-participation.service'; import { ExamPageComponent } from 'app/exam/participate/exercises/exam-page.component'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from 'app/entities/programming-exercise-git-diff-report.model'; @Component({ selector: 'jhi-student-exam-timeline', diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel-title.component.html b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel-title.component.html similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel-title.component.html rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel-title.component.html diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel-title.component.scss b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel-title.component.scss similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel-title.component.scss rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel-title.component.scss diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel-title.component.ts b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel-title.component.ts similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel-title.component.ts rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel-title.component.ts diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel.component.html b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel.component.html similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel.component.html rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel.component.html diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel.component.scss b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel.component.scss similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel.component.scss rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel.component.scss diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel.component.ts b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel.component.ts similarity index 90% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel.component.ts rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel.component.ts index 9e01edae26b9..0417e9239618 100644 --- a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file-panel.component.ts +++ b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel.component.ts @@ -1,9 +1,9 @@ import { ChangeDetectionStrategy, Component, ViewEncapsulation, computed, input, output } from '@angular/core'; -import { ProgrammingExerciseGitDiffEntry } from 'app/entities/hestia/programming-exercise-git-diff-entry.model'; +import { ProgrammingExerciseGitDiffEntry } from 'app/entities/programming-exercise-git-diff-entry.model'; import { faAngleDown, faAngleUp } from '@fortawesome/free-solid-svg-icons'; -import { GitDiffFilePanelTitleComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-file-panel-title.component'; -import { GitDiffLineStatComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component'; -import { GitDiffFileComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-file.component'; +import { GitDiffFilePanelTitleComponent } from 'app/exercises/programming/git-diff-report/git-diff-file-panel-title.component'; +import { GitDiffLineStatComponent } from 'app/exercises/programming/git-diff-report/git-diff-line-stat.component'; +import { GitDiffFileComponent } from 'app/exercises/programming/git-diff-report/git-diff-file.component'; import { ArtemisSharedModule } from 'app/shared/shared.module'; @Component({ diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file.component.html b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file.component.html similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file.component.html rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file.component.html diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file.component.ts b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file.component.ts similarity index 92% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file.component.ts rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file.component.ts index 45456a115513..055875de5a6a 100644 --- a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-file.component.ts +++ b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-file.component.ts @@ -1,5 +1,5 @@ import { ChangeDetectionStrategy, Component, ViewEncapsulation, computed, effect, input, output, viewChild } from '@angular/core'; -import { ProgrammingExerciseGitDiffEntry } from 'app/entities/hestia/programming-exercise-git-diff-entry.model'; +import { ProgrammingExerciseGitDiffEntry } from 'app/entities/programming-exercise-git-diff-entry.model'; import { MonacoDiffEditorComponent } from 'app/shared/monaco-editor/monaco-diff-editor.component'; import { TranslateDirective } from 'app/shared/language/translate.directive'; diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component.html b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-line-stat.component.html similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component.html rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-line-stat.component.html diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component.scss b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-line-stat.component.scss similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component.scss rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-line-stat.component.scss diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component.ts b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-line-stat.component.ts similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component.ts rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-line-stat.component.ts diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-report-modal.component.html b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-report-modal.component.html similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-report-modal.component.html rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-report-modal.component.html diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-report-modal.component.ts b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-report-modal.component.ts similarity index 97% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-report-modal.component.ts rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-report-modal.component.ts index 7684573f821f..96a16d6986ee 100644 --- a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-report-modal.component.ts +++ b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-report-modal.component.ts @@ -1,11 +1,11 @@ import { ChangeDetectionStrategy, Component, effect, inject, input, signal, untracked } from '@angular/core'; import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from 'app/entities/programming-exercise-git-diff-report.model'; import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; import { ProgrammingExerciseParticipationService } from 'app/exercises/programming/manage/services/programming-exercise-participation.service'; import { CachedRepositoryFilesService } from 'app/exercises/programming/manage/services/cached-repository-files.service'; import { firstValueFrom } from 'rxjs'; -import { GitDiffReportComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report.component'; +import { GitDiffReportComponent } from 'app/exercises/programming/git-diff-report/git-diff-report.component'; import { TranslateDirective } from 'app/shared/language/translate.directive'; @Component({ diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-report.component.html b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-report.component.html similarity index 100% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-report.component.html rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-report.component.html diff --git a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-report.component.ts b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-report.component.ts similarity index 95% rename from src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-report.component.ts rename to src/main/webapp/app/exercises/programming/git-diff-report/git-diff-report.component.ts index cc916e5daf93..7edc4f32aaf3 100644 --- a/src/main/webapp/app/exercises/programming/hestia/git-diff-report/git-diff-report.component.ts +++ b/src/main/webapp/app/exercises/programming/git-diff-report/git-diff-report.component.ts @@ -1,12 +1,12 @@ import { ChangeDetectionStrategy, Component, computed, effect, input, signal, untracked } from '@angular/core'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; -import { ProgrammingExerciseGitDiffEntry } from 'app/entities/hestia/programming-exercise-git-diff-entry.model'; +import { ProgrammingExerciseGitDiffReport } from 'app/entities/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffEntry } from 'app/entities/programming-exercise-git-diff-entry.model'; import { faSpinner, faTableColumns } from '@fortawesome/free-solid-svg-icons'; import { ButtonSize, ButtonType, TooltipPlacement } from 'app/shared/components/button.component'; -import { GitDiffLineStatComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component'; +import { GitDiffLineStatComponent } from 'app/exercises/programming/git-diff-report/git-diff-line-stat.component'; import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; import { ArtemisSharedModule } from 'app/shared/shared.module'; -import { GitDiffFilePanelComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-file-panel.component'; +import { GitDiffFilePanelComponent } from 'app/exercises/programming/git-diff-report/git-diff-file-panel.component'; interface DiffInformation { path: string; diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.html b/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.html deleted file mode 100644 index 9a62ddda158b..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.html +++ /dev/null @@ -1,52 +0,0 @@ -

- -

-
-@if (isPerformedByStep?.size && currentStep !== undefined && exercise) { -
-
-
- @if (allowBehavioralEntryGeneration ? currentStep !== 0 : currentStep !== 2) { - - } -
-
- -
-
- @if (!isNextStepAvailable()) { - - } - @if (currentStep !== 3) { - - } -
-
-
- - @if (exercise) { - @if (allowBehavioralEntryGeneration) { - - } - @if (allowBehavioralEntryGeneration) { - - } - - - } -
-} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.scss b/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.scss deleted file mode 100644 index 2fa141c0b0d4..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.scss +++ /dev/null @@ -1,49 +0,0 @@ -.button-row { - & > button { - margin-right: 5px; - } -} - -.status-wrapper { - flex-grow: 3; -} - -.butonene-wr .btn[disabled] { - pointer-events: all !important; -} - -.mat-expansion-panel.code-hint-creation-expansion-panel-wrapper, -.mat-expansion-panel.code-hint-creation-expansion-panel-wrapper-header { - .mat-expansion-panel-header { - padding: 10px 10px 10px 0; - border-bottom: 1px solid rgba(0, 0, 0, 0.25); - } - - .mat-expansion-panel-header-title, - .mat-expansion-panel-header-description { - flex-basis: 0; - } - - .mat-expansion-panel-header-description { - justify-content: space-between; - align-items: center; - } -} - -/* prevent resizing of opened expansion panels */ -.mat-expansion-panel-spacing { - margin: 0; -} - -.mat-expansion-panel.code-hint-creation-expansion-panel-wrapper-header { - font-weight: bold; - cursor: default; - - .mat-expansion-panel-header { - border-bottom: 2px solid; - } -} - -.sort-icon { - cursor: pointer; -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.ts b/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.ts deleted file mode 100644 index d039db390a19..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component.ts +++ /dev/null @@ -1,95 +0,0 @@ -import { Component, OnInit } from '@angular/core'; -import { ActivatedRoute, Router } from '@angular/router'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { CoverageReport } from 'app/entities/hestia/coverage-report.model'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { CodeHint, CodeHintGenerationStep } from 'app/entities/hestia/code-hint-model'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; - -@Component({ - selector: 'jhi-code-hint-generation-overview', - templateUrl: './code-hint-generation-overview.component.html', - styleUrls: ['./code-hint-generation-overview.component.scss'], -}) -export class CodeHintGenerationOverviewComponent implements OnInit { - exercise?: ProgrammingExercise; - - currentStep: CodeHintGenerationStep; - isPerformedByStep: Map; - selectedSolutionEntries?: ProgrammingExerciseSolutionEntry[]; - - allowBehavioralEntryGeneration = false; - - readonly GenerationStep = CodeHintGenerationStep; - - constructor( - private route: ActivatedRoute, - private router: Router, - ) {} - - ngOnInit() { - this.route.data.subscribe(({ exercise }) => { - this.exercise = exercise; - // set all steps to unperformed initially - this.isPerformedByStep = new Map(); - this.isPerformedByStep.set(CodeHintGenerationStep.SOLUTION_ENTRIES, false); - this.isPerformedByStep.set(CodeHintGenerationStep.CODE_HINTS, false); - if (exercise.buildConfig?.testwiseCoverageEnabled) { - this.currentStep = CodeHintGenerationStep.GIT_DIFF; - this.allowBehavioralEntryGeneration = true; - this.isPerformedByStep.set(CodeHintGenerationStep.GIT_DIFF, false); - this.isPerformedByStep.set(CodeHintGenerationStep.COVERAGE, false); - } else { - this.currentStep = CodeHintGenerationStep.SOLUTION_ENTRIES; - } - }); - } - - setLatestPerformedStep(latestUpdatedStep: CodeHintGenerationStep) { - if (this.currentStep >= latestUpdatedStep) { - return; - } - const optionalEntry = Array.from(this.isPerformedByStep.entries()) - .filter((a) => a[1]) - .sort((a, b) => b[0] - a[0]) - .first(); - this.currentStep = optionalEntry === undefined ? this.currentStep : optionalEntry![0]; - } - - isNextStepAvailable(): boolean { - return this.isPerformedByStep.get(this.currentStep) ?? false; - } - - onNextStep() { - this.currentStep = this.currentStep + 1; - } - - onPreviousStep() { - this.currentStep = this.currentStep - 1; - } - - onStepChange(step: CodeHintGenerationStep) { - this.currentStep = step; - } - - onDiffReportLoaded(diffReport?: ProgrammingExerciseGitDiffReport) { - this.isPerformedByStep.set(CodeHintGenerationStep.GIT_DIFF, diffReport !== undefined); - this.setLatestPerformedStep(CodeHintGenerationStep.GIT_DIFF); - } - - onCoverageReportLoaded(coverageReport?: CoverageReport) { - this.isPerformedByStep.set(CodeHintGenerationStep.COVERAGE, coverageReport !== undefined); - this.setLatestPerformedStep(CodeHintGenerationStep.COVERAGE); - } - - onSolutionEntryChanges(entries?: ProgrammingExerciseSolutionEntry[]) { - this.selectedSolutionEntries = entries; - this.isPerformedByStep.set(CodeHintGenerationStep.SOLUTION_ENTRIES, entries !== undefined && entries!.length > 0); - this.setLatestPerformedStep(CodeHintGenerationStep.SOLUTION_ENTRIES); - } - - onCodeHintsLoaded(codeHints?: CodeHint[]) { - this.isPerformedByStep.set(CodeHintGenerationStep.CODE_HINTS, codeHints !== undefined && codeHints!.length > 0); - this.setLatestPerformedStep(CodeHintGenerationStep.CODE_HINTS); - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.module.ts b/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.module.ts deleted file mode 100644 index a9d7088b56c9..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.module.ts +++ /dev/null @@ -1,53 +0,0 @@ -import { NgModule } from '@angular/core'; -import { ArtemisSharedModule } from 'app/shared/shared.module'; -import { CodeHintGenerationStatusComponent } from '../code-hint-generation-status/code-hint-generation-status.component'; -import { SolutionEntryDetailsModalComponent } from '../solution-entry-details-modal/solution-entry-details-modal.component'; -import { ArtemisExerciseHintSharedModule } from 'app/exercises/shared/exercise-hint/shared/exercise-hint-shared.module'; -import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; -import { DiffGenerationStepComponent } from '../steps/diff-generation-step/diff-generation-step.component'; -import { CoverageGenerationStepComponent } from '../steps/coverage-generation-step/coverage-generation-step.component'; -import { SolutionEntryGenerationStepComponent } from '../steps/solution-entry-generation-step/solution-entry-generation-step.component'; -import { CodeHintGenerationStepComponent } from '../steps/code-hint-generation-step/code-hint-generation-step.component'; -import { TestwiseCoverageReportModule } from 'app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.module'; -import { CodeHintGenerationOverviewComponent } from 'app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component'; -import { RouterModule } from '@angular/router'; -import { ArtemisMarkdownModule } from 'app/shared/markdown.module'; -import { MatExpansionModule } from '@angular/material/expansion'; -import { CodeHintGenerationStatusStepComponent } from 'app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status-step.component'; -import { ManualSolutionEntryCreationModalComponent } from '../manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component'; -import { GitDiffReportComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report.component'; - -@NgModule({ - imports: [ - RouterModule, - ArtemisSharedModule, - ArtemisExerciseHintSharedModule, - ArtemisSharedComponentModule, - TestwiseCoverageReportModule, - ArtemisMarkdownModule, - MatExpansionModule, - GitDiffReportComponent, - ], - declarations: [ - CodeHintGenerationStatusComponent, - SolutionEntryDetailsModalComponent, - DiffGenerationStepComponent, - CoverageGenerationStepComponent, - SolutionEntryGenerationStepComponent, - CodeHintGenerationStepComponent, - CodeHintGenerationOverviewComponent, - CodeHintGenerationStatusStepComponent, - ManualSolutionEntryCreationModalComponent, - ], - exports: [ - CodeHintGenerationStatusComponent, - SolutionEntryDetailsModalComponent, - DiffGenerationStepComponent, - CoverageGenerationStepComponent, - SolutionEntryGenerationStepComponent, - CodeHintGenerationStepComponent, - CodeHintGenerationOverviewComponent, - CodeHintGenerationStatusStepComponent, - ], -}) -export class ArtemisCodeHintGenerationOverviewModule {} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status-step.component.html b/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status-step.component.html deleted file mode 100644 index 0fb688bdaade..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status-step.component.html +++ /dev/null @@ -1,14 +0,0 @@ -
-
- @if (isPerformed) { - - } - @if (!isPerformed) { - - } -
-
- Current step - -
-
diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status-step.component.ts b/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status-step.component.ts deleted file mode 100644 index d46904d04367..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status-step.component.ts +++ /dev/null @@ -1,24 +0,0 @@ -import { Component, Input } from '@angular/core'; -import { faCheck, faDotCircle } from '@fortawesome/free-solid-svg-icons'; - -@Component({ - selector: 'jhi-code-hint-generation-status-step', - templateUrl: './code-hint-generation-status-step.component.html', - styleUrls: ['./code-hint-generation-status.component.scss'], -}) -export class CodeHintGenerationStatusStepComponent { - @Input() - currentlySelected: boolean; - - @Input() - isPerformed: boolean; - - @Input() - labelTranslationKey: string; - - @Input() - descriptionTranslationKey: string; - - faCheck = faCheck; - faDotCircle = faDotCircle; -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.html b/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.html deleted file mode 100644 index 6cc22ad5429d..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.html +++ /dev/null @@ -1,45 +0,0 @@ -
- - @if (isPerformedByStep.get(GenerationStep.GIT_DIFF) !== undefined) { - - } - @if (isPerformedByStep.get(GenerationStep.GIT_DIFF) !== undefined) { -
- } - - @if (isPerformedByStep.get(GenerationStep.COVERAGE) !== undefined) { - - } - @if (isPerformedByStep.get(GenerationStep.COVERAGE) !== undefined) { -
- } - - -
- - -
diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.scss b/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.scss deleted file mode 100644 index 5a790263e647..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.scss +++ /dev/null @@ -1,65 +0,0 @@ -.code-hint-generation-wrapper { - display: flex; - justify-content: space-between; - - .planned, - .finished, - .check { - color: var(--success); - } - - .unset { - color: var(--secondary); - } - - .current { - color: var(--warning); - } - - li { - display: list-item !important; - } - - .connector { - width: 100%; - min-width: 50px; - border-top: 2px solid; - margin-top: 1rem; - } -} - -.status-step { - width: 25px; - overflow-x: visible; - justify-content: flex-start; - display: flex; - flex-direction: column; - align-items: center; - - .status-step-content { - min-width: fit-content; - white-space: nowrap; - - .selected-label { - font-weight: bold; - } - } - - .header-icon { - z-index: 2; - font-size: 2em; - - & > fa-icon { - cursor: pointer; - background-color: var(--programming-exercise-instruction-step-wizard-card-header-background); - width: 30px; - height: 30px; - text-align: center; - padding: 6px 0; - font-size: 12px; - line-height: 1.428571429; - border-radius: 15px; - border-color: var(--programming-exercise-instruction-step-wizard-btn-border-color); - } - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.ts b/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.ts deleted file mode 100644 index b0c31d9f9676..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component.ts +++ /dev/null @@ -1,26 +0,0 @@ -import { Component, EventEmitter, Input, Output } from '@angular/core'; -import { CodeHintGenerationStep } from 'app/entities/hestia/code-hint-model'; - -@Component({ - selector: 'jhi-code-hint-generation-status', - templateUrl: './code-hint-generation-status.component.html', - styleUrls: ['./code-hint-generation-status.component.scss'], -}) -export class CodeHintGenerationStatusComponent { - @Input() - currentStep: CodeHintGenerationStep; - - @Input() - isPerformedByStep: Map; - - @Output() - onStepChange = new EventEmitter(); - - readonly GenerationStep = CodeHintGenerationStep; - - constructor() {} - - onSelectStep(step: CodeHintGenerationStep) { - this.onStepChange.emit(step); - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component.html b/src/main/webapp/app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component.html deleted file mode 100644 index 842eb668da94..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component.html +++ /dev/null @@ -1,43 +0,0 @@ -
- - -
diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component.ts b/src/main/webapp/app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component.ts deleted file mode 100644 index 0df03f1664b3..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component.ts +++ /dev/null @@ -1,85 +0,0 @@ -import { Component, EventEmitter, OnDestroy, OnInit, ViewChild } from '@angular/core'; -import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; -import { Subject } from 'rxjs'; -import { ProgrammingExerciseTestCase } from 'app/entities/programming/programming-exercise-test-case.model'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { CodeHint } from 'app/entities/hestia/code-hint-model'; -import { ProgrammingExerciseSolutionEntryService } from 'app/exercises/shared/exercise-hint/services/programming-exercise-solution-entry.service'; -import { SolutionEntryComponent } from 'app/exercises/shared/exercise-hint/shared/solution-entry.component'; - -@Component({ - selector: 'jhi-manual-solution-entry-creation-modal', - templateUrl: './manual-solution-entry-creation-modal.component.html', -}) -export class ManualSolutionEntryCreationModalComponent implements OnInit, OnDestroy { - @ViewChild('solutionEntryComponent', { static: false }) solutionEntryComponent: SolutionEntryComponent; - - solutionEntry = new ProgrammingExerciseSolutionEntry(); - - exerciseId: number; - codeHint?: CodeHint; - testCases?: ProgrammingExerciseTestCase[]; - onEntryCreated = new EventEmitter(); - - solutionRepositoryFileNames?: string[]; - - private dialogErrorSource = new Subject(); - dialogError$ = this.dialogErrorSource.asObservable(); - - constructor( - private activeModal: NgbActiveModal, - private exerciseService: ProgrammingExerciseService, - private solutionEntryService: ProgrammingExerciseSolutionEntryService, - ) { - this.solutionEntry.code = ''; - this.solutionEntry.line = 1; - } - - ngOnDestroy(): void { - this.dialogErrorSource.unsubscribe(); - } - - ngOnInit(): void { - this.solutionEntry.codeHint = this.codeHint; - // only load test cases if not set as an input - if (!this.testCases) { - this.exerciseService.getAllTestCases(this.exerciseId).subscribe({ - next: (testCases) => { - this.testCases = testCases; - }, - error: (error) => this.dialogErrorSource.error(error.message), - }); - } - this.exerciseService.getSolutionFileNames(this.exerciseId).subscribe({ - next: (fileNames) => { - this.solutionRepositoryFileNames = fileNames; - }, - error: (error) => this.dialogErrorSource.error(error.message), - }); - } - - /** - * Updates the solution entry editor to refer to the correct file path. - * In particular, this updates the syntax highlighting of the editor. - */ - onUpdateFilePath(): void { - this.solutionEntryComponent.setupEditor(); - } - - clear() { - this.activeModal.close(); - } - - onCreateEntry() { - // eslint-disable-next-line @typescript-eslint/no-non-null-asserted-optional-chain - this.solutionEntryService.createSolutionEntry(this.exerciseId, this.solutionEntry?.testCase?.id!, this.solutionEntry).subscribe({ - next: (createdEntry) => { - this.dialogErrorSource.next(''); - this.onEntryCreated.emit(createdEntry); - this.activeModal.close(); - }, - error: (error) => this.dialogErrorSource.error(error.message), - }); - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/solution-entry-details-modal/solution-entry-details-modal.component.html b/src/main/webapp/app/exercises/programming/hestia/generation-overview/solution-entry-details-modal/solution-entry-details-modal.component.html deleted file mode 100644 index 721b53f675b1..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/solution-entry-details-modal/solution-entry-details-modal.component.html +++ /dev/null @@ -1,21 +0,0 @@ -
- - -
diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/solution-entry-details-modal/solution-entry-details-modal.component.ts b/src/main/webapp/app/exercises/programming/hestia/generation-overview/solution-entry-details-modal/solution-entry-details-modal.component.ts deleted file mode 100644 index 8e2dcbacb3ab..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/solution-entry-details-modal/solution-entry-details-modal.component.ts +++ /dev/null @@ -1,42 +0,0 @@ -import { Component, OnDestroy } from '@angular/core'; -import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { Subject } from 'rxjs'; -import { ProgrammingExerciseSolutionEntryService } from 'app/exercises/shared/exercise-hint/services/programming-exercise-solution-entry.service'; - -@Component({ - selector: 'jhi-solution-entry-details-modal', - templateUrl: './solution-entry-details-modal.component.html', -}) -export class SolutionEntryDetailsModalComponent implements OnDestroy { - exerciseId: number; - solutionEntry: ProgrammingExerciseSolutionEntry; - isEditable: boolean; - - private dialogErrorSource = new Subject(); - dialogError$ = this.dialogErrorSource.asObservable(); - - constructor( - private activeModal: NgbActiveModal, - private solutionEntryService: ProgrammingExerciseSolutionEntryService, - ) {} - - ngOnDestroy(): void { - this.dialogErrorSource.unsubscribe(); - } - - clear() { - this.activeModal.close(); - } - - saveSolutionEntry() { - // eslint-disable-next-line @typescript-eslint/no-non-null-asserted-optional-chain - this.solutionEntryService.updateSolutionEntry(this.exerciseId, this.solutionEntry.testCase?.id!, this.solutionEntry.id!, this.solutionEntry).subscribe({ - next: (updatedEntry) => { - this.solutionEntry = updatedEntry; - this.activeModal.close(); - }, - error: (error) => this.dialogErrorSource.error(error.message), - }); - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/code-hint-generation-step/code-hint-generation-step.component.html b/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/code-hint-generation-step/code-hint-generation-step.component.html deleted file mode 100644 index 70538fa15b02..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/code-hint-generation-step/code-hint-generation-step.component.html +++ /dev/null @@ -1,74 +0,0 @@ -@if (isLoading) { -
- -
-} @else { -
- @if (codeHints?.length) { - - - } @else { - - } -
- @if (codeHints?.length) { - - - - {{ 'artemisApp.codeHint.management.step4.title' | artemisTranslate }} - {{ 'artemisApp.codeHint.management.step4.task' | artemisTranslate }} - - - @for (codeHint of codeHints; track codeHint) { - - - - {{ codeHint.title }} - - - {{ codeHint?.programmingExerciseTask?.taskName }} - - -
- @if (codeHint?.solutionEntries) { -
- -
- } - -
-
- } -
- } @else { -
- -
- } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/code-hint-generation-step/code-hint-generation-step.component.ts b/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/code-hint-generation-step/code-hint-generation-step.component.ts deleted file mode 100644 index 59e8150e2fe8..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/code-hint-generation-step/code-hint-generation-step.component.ts +++ /dev/null @@ -1,67 +0,0 @@ -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { CodeHint } from 'app/entities/hestia/code-hint-model'; -import { faWrench } from '@fortawesome/free-solid-svg-icons'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { CodeHintService } from 'app/exercises/shared/exercise-hint/services/code-hint.service'; -import { AlertService } from 'app/core/util/alert.service'; - -@Component({ - selector: 'jhi-code-hint-generation-step', - templateUrl: './code-hint-generation-step.component.html', - styleUrls: ['../../code-hint-generation-overview/code-hint-generation-overview.component.scss'], -}) -export class CodeHintGenerationStepComponent implements OnInit { - @Input() - exercise: ProgrammingExercise; - - @Output() - onCodeHintsLoaded = new EventEmitter(); - - isLoading = false; - codeHints?: CodeHint[]; - - faWrench = faWrench; - - constructor( - private exerciseService: ProgrammingExerciseService, - private codeHintService: CodeHintService, - private alertService: AlertService, - ) {} - - ngOnInit() { - this.isLoading = true; - this.exerciseService.getCodeHintsForExercise(this.exercise.id!).subscribe({ - next: (codeHints) => { - this.codeHints = codeHints; - this.onCodeHintsLoaded.emit(this.codeHints); - this.isLoading = false; - }, - error: (error) => { - this.isLoading = false; - this.alertService.error(error.message); - }, - }); - } - - generateCodeHints(deleteOldHints: boolean, buttonTranslationKey: string) { - this.isLoading = true; - // eslint-disable-next-line @typescript-eslint/no-non-null-asserted-optional-chain - this.codeHintService.generateCodeHintsForExercise(this.exercise?.id!, deleteOldHints).subscribe({ - next: (generatedHints) => { - if (deleteOldHints) { - this.codeHints = generatedHints; - } else { - generatedHints.forEach((generatedHint) => this.codeHints?.push(generatedHint)); - } - this.onCodeHintsLoaded.emit(this.codeHints); - this.isLoading = false; - this.alertService.success('artemisApp.codeHint.management.step4.' + buttonTranslationKey + '.success'); - }, - error: (error) => { - this.isLoading = false; - this.alertService.error(error.message); - }, - }); - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/coverage-generation-step/coverage-generation-step.component.html b/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/coverage-generation-step/coverage-generation-step.component.html deleted file mode 100644 index 421eb8140c86..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/coverage-generation-step/coverage-generation-step.component.html +++ /dev/null @@ -1,13 +0,0 @@ -@if (isLoading) { -
- -
-} -@if (coverageReport && fileContentByPath) { - -} -@if (!(coverageReport || fileContentByPath)) { -
- -
-} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/coverage-generation-step/coverage-generation-step.component.ts b/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/coverage-generation-step/coverage-generation-step.component.ts deleted file mode 100644 index 0fdc47306615..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/coverage-generation-step/coverage-generation-step.component.ts +++ /dev/null @@ -1,47 +0,0 @@ -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; -import { CoverageReport } from 'app/entities/hestia/coverage-report.model'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { AlertService } from 'app/core/util/alert.service'; - -@Component({ - selector: 'jhi-coverage-generation-step', - templateUrl: './coverage-generation-step.component.html', - styleUrls: ['../../code-hint-generation-overview/code-hint-generation-overview.component.scss'], -}) -export class CoverageGenerationStepComponent implements OnInit { - @Input() - exercise: ProgrammingExercise; - - @Output() - onCoverageLoaded = new EventEmitter(); - - isLoading = false; - coverageReport?: CoverageReport; - fileContentByPath = new Map(); - - constructor( - private exerciseService: ProgrammingExerciseService, - private alertService: AlertService, - ) {} - - ngOnInit(): void { - this.isLoading = true; - this.exerciseService.getSolutionRepositoryTestFilesWithContent(this.exercise.id!).subscribe({ - next: (filesWithContent: Map) => { - this.exerciseService.getLatestFullTestwiseCoverageReport(this.exercise.id!).subscribe({ - next: (coverageReport) => { - this.isLoading = false; - this.onCoverageLoaded.emit(coverageReport); - this.coverageReport = coverageReport; - this.fileContentByPath = filesWithContent; - }, - error: (error) => { - this.isLoading = false; - this.alertService.error(error.message); - }, - }); - }, - }); - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/diff-generation-step/diff-generation-step.component.html b/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/diff-generation-step/diff-generation-step.component.html deleted file mode 100644 index af31d173e0da..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/diff-generation-step/diff-generation-step.component.html +++ /dev/null @@ -1,15 +0,0 @@ -@if (isLoading) { -
- -
-} -@if (!isLoading && gitDiffReport) { -
- -
-} -@if (!isLoading && !gitDiffReport) { -
- -
-} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/diff-generation-step/diff-generation-step.component.ts b/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/diff-generation-step/diff-generation-step.component.ts deleted file mode 100644 index 3903491a1477..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/diff-generation-step/diff-generation-step.component.ts +++ /dev/null @@ -1,54 +0,0 @@ -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { AlertService } from 'app/core/util/alert.service'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; - -@Component({ - selector: 'jhi-diff-generation-step', - templateUrl: './diff-generation-step.component.html', - styleUrls: ['../../code-hint-generation-overview/code-hint-generation-overview.component.scss'], -}) -export class DiffGenerationStepComponent implements OnInit { - @Input() - exercise: ProgrammingExercise; - - @Output() - onGitDiffLoaded = new EventEmitter(); - - isLoading = false; - gitDiffReport?: ProgrammingExerciseGitDiffReport; - templateFileContentByPath: Map; - solutionFileContentByPath: Map; - - constructor( - private exerciseService: ProgrammingExerciseService, - private alertService: AlertService, - ) {} - - ngOnInit() { - this.isLoading = true; - this.exerciseService.getDiffReport(this.exercise.id!).subscribe({ - next: (report) => { - this.gitDiffReport = report; - this.onGitDiffLoaded.emit(report); - this.exerciseService.getTemplateRepositoryTestFilesWithContent(this.exercise.id!).subscribe({ - next: (response: Map) => { - this.templateFileContentByPath = response; - this.isLoading = this.solutionFileContentByPath === undefined; - }, - }); - this.exerciseService.getSolutionRepositoryTestFilesWithContent(this.exercise.id!).subscribe({ - next: (response: Map) => { - this.solutionFileContentByPath = response; - this.isLoading = this.templateFileContentByPath === undefined; - }, - }); - }, - error: (error) => { - this.isLoading = false; - this.alertService.error(error.message); - }, - }); - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/solution-entry-generation-step/solution-entry-generation-step.component.html b/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/solution-entry-generation-step/solution-entry-generation-step.component.html deleted file mode 100644 index 3e3b36f53456..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/solution-entry-generation-step/solution-entry-generation-step.component.html +++ /dev/null @@ -1,97 +0,0 @@ -@if (isLoading) { -
- -
-} -
- - - @if (!!exercise?.buildConfig?.testwiseCoverageEnabled) { - - } - -
-
- - - - - - - - - - - - @for (entry of solutionEntries; track entry) { - - - - - - - - - } - -
- {{ 'artemisApp.programmingExerciseTestCase.testName' | artemisTranslate }} - - File Path - - -
{{ entry?.id }}{{ entry?.testCase?.testName }}{{ entry?.filePath }} - {{ - (entry.code?.split('\n')?.length! === 1 ? 'artemisApp.codeHint.lineOfCode' : 'artemisApp.codeHint.linesOfCode') - | artemisTranslate: { lines: entry.code?.split('\n')?.length! } - }} - {{ 'artemisApp.codeHint.entryAbbreviation.' + entry.testCase?.type?.toString()?.toLocaleLowerCase() | artemisTranslate }} -
- - - -
-
- @if (!solutionEntries?.length) { -
- -
- } -
diff --git a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/solution-entry-generation-step/solution-entry-generation-step.component.ts b/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/solution-entry-generation-step/solution-entry-generation-step.component.ts deleted file mode 100644 index 5d4b3c68f675..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/generation-overview/steps/solution-entry-generation-step/solution-entry-generation-step.component.ts +++ /dev/null @@ -1,178 +0,0 @@ -import { Component, EventEmitter, Input, OnDestroy, OnInit, Output } from '@angular/core'; -import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { SolutionEntryDetailsModalComponent } from 'app/exercises/programming/hestia/generation-overview/solution-entry-details-modal/solution-entry-details-modal.component'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { ProgrammingExerciseTestCaseType } from 'app/entities/programming/programming-exercise-test-case.model'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { AlertService } from 'app/core/util/alert.service'; -import { Subject } from 'rxjs'; -import { faSort, faSortDown, faSortUp, faTimes } from '@fortawesome/free-solid-svg-icons'; -import { CodeHintService } from 'app/exercises/shared/exercise-hint/services/code-hint.service'; -import { ManualSolutionEntryCreationModalComponent } from 'app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component'; -import { SortingOrder } from 'app/shared/table/pageable-table'; -import { ProgrammingExerciseSolutionEntryService } from 'app/exercises/shared/exercise-hint/services/programming-exercise-solution-entry.service'; -import { ConfirmAutofocusModalComponent } from 'app/shared/components/confirm-autofocus-modal.component'; -import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; - -@Component({ - selector: 'jhi-solution-entry-generation-step', - templateUrl: './solution-entry-generation-step.component.html', - styleUrls: ['../../code-hint-generation-overview/code-hint-generation-overview.component.scss'], -}) -export class SolutionEntryGenerationStepComponent implements OnInit, OnDestroy { - @Input() - exercise: ProgrammingExercise; - - @Output() - onEntryUpdate = new EventEmitter(); - - isLoading: boolean; - solutionEntries: ProgrammingExerciseSolutionEntry[]; - faTimes = faTimes; - - testCaseSortOrder?: SortingOrder; - faSort = faSort; - faSortUp = faSortUp; - faSortDown = faSortDown; - - private dialogErrorSource = new Subject(); - dialogError$ = this.dialogErrorSource.asObservable(); - - readonly SortingOrder = SortingOrder; - - constructor( - private modalService: NgbModal, - private exerciseService: ProgrammingExerciseService, - private alertService: AlertService, - private artemisTranslatePipe: ArtemisTranslatePipe, - private codeHintService: CodeHintService, - private solutionEntryService: ProgrammingExerciseSolutionEntryService, - ) {} - - ngOnInit() { - this.isLoading = true; - this.solutionEntryService.getSolutionEntriesForExercise(this.exercise.id!).subscribe({ - next: (solutionEntries: ProgrammingExerciseSolutionEntry[]) => { - this.solutionEntries = solutionEntries; - this.isLoading = false; - this.onEntryUpdate.emit(this.solutionEntries); - }, - error: (error) => { - this.isLoading = false; - this.alertService.error(error.message); - }, - }); - } - - ngOnDestroy(): void { - this.dialogErrorSource.unsubscribe(); - } - - openSolutionEntryModal(solutionEntry: ProgrammingExerciseSolutionEntry, isEditable: boolean) { - const modalRef: NgbModalRef = this.modalService.open(SolutionEntryDetailsModalComponent as Component, { - size: 'lg', - backdrop: 'static', - }); - modalRef.componentInstance.exerciseId = this.exercise.id; - modalRef.componentInstance.solutionEntry = solutionEntry; - modalRef.componentInstance.isEditable = isEditable; - } - - openManualEntryCreationModal() { - const modalRef: NgbModalRef = this.modalService.open(ManualSolutionEntryCreationModalComponent as Component, { - size: 'lg', - backdrop: 'static', - }); - modalRef.componentInstance.exerciseId = this.exercise.id; - modalRef.componentInstance.onEntryCreated.subscribe((createdEntry: ProgrammingExerciseSolutionEntry) => { - this.solutionEntries.push(createdEntry); - this.onEntryUpdate.emit(this.solutionEntries); - }); - } - - onGenerateStructuralSolutionEntries() { - this.exerciseService.createStructuralSolutionEntries(this.exercise.id!).subscribe({ - next: (updatedStructuralEntries) => { - this.alertService.success('artemisApp.programmingExercise.createStructuralSolutionEntriesSuccess'); - // replace all structural entries - const result = this.removeSolutionEntriesOfTypeFromArray(this.solutionEntries, ProgrammingExerciseTestCaseType.STRUCTURAL); - Array.prototype.push.apply(result, updatedStructuralEntries); - this.solutionEntries = result; - this.testCaseSortOrder = undefined; - this.onEntryUpdate.emit(this.solutionEntries); - }, - error: (error) => this.alertService.error(error.message), - }); - } - - onGenerateBehavioralSolutionEntries() { - this.exerciseService.createBehavioralSolutionEntries(this.exercise.id!).subscribe({ - next: (updatedBehavioralEntries) => { - this.alertService.success('artemisApp.programmingExercise.createBehavioralSolutionEntriesSuccess'); - // replace all behavioral entries - const result = this.removeSolutionEntriesOfTypeFromArray(this.solutionEntries, ProgrammingExerciseTestCaseType.BEHAVIORAL); - Array.prototype.push.apply(result, updatedBehavioralEntries); - this.solutionEntries = result; - this.testCaseSortOrder = undefined; - this.onEntryUpdate.emit(this.solutionEntries); - }, - error: (error) => this.alertService.error(error.message), - }); - } - - openBulkDeletionModal() { - const modalRef = this.modalService.open(ConfirmAutofocusModalComponent, { keyboard: true, size: 'lg' }); - modalRef.componentInstance.title = 'artemisApp.codeHint.management.step3.deleteAllEntriesButton.title'; - modalRef.componentInstance.text = this.artemisTranslatePipe.transform('artemisApp.codeHint.management.step3.deleteAllEntriesButton.question'); - modalRef.result.then(() => { - this.deleteAllSolutionEntries(); - }); - } - - deleteAllSolutionEntries() { - // eslint-disable-next-line @typescript-eslint/no-non-null-asserted-optional-chain - this.solutionEntryService.deleteAllSolutionEntriesForExercise(this.exercise?.id!).subscribe({ - next: () => { - this.solutionEntries = []; - this.onEntryUpdate.emit([]); - this.alertService.success('artemisApp.codeHint.management.step3.deleteAllEntriesButton.success'); - this.dialogErrorSource.next(''); - }, - error: (error) => this.dialogErrorSource.error(error.message), - }); - } - - deleteSolutionEntry(entry: ProgrammingExerciseSolutionEntry) { - // eslint-disable-next-line @typescript-eslint/no-non-null-asserted-optional-chain - this.solutionEntryService.deleteSolutionEntry(this.exercise?.id!, entry.testCase?.id!, entry.id!).subscribe({ - next: () => { - this.solutionEntries = this.solutionEntries.filter((existingEntry) => entry !== existingEntry); - this.dialogErrorSource.next(''); - this.onEntryUpdate.emit(this.solutionEntries); - }, - error: (error) => this.dialogErrorSource.error(error.message), - }); - } - - changeTestCaseSortOrder() { - switch (this.testCaseSortOrder) { - case SortingOrder.ASCENDING: - this.solutionEntries = this.solutionEntries.reverse(); - this.testCaseSortOrder = SortingOrder.DESCENDING; - break; - case SortingOrder.DESCENDING: - this.solutionEntries = this.solutionEntries.reverse(); - this.testCaseSortOrder = SortingOrder.ASCENDING; - break; - case undefined: - // eslint-disable-next-line @typescript-eslint/no-non-null-asserted-optional-chain - this.solutionEntries = this.solutionEntries.sort((a, b) => a.testCase?.testName!.localeCompare(b.testCase?.testName!)!); - this.testCaseSortOrder = SortingOrder.ASCENDING; - } - } - - private removeSolutionEntriesOfTypeFromArray(entries: ProgrammingExerciseSolutionEntry[], typeToRemove: ProgrammingExerciseTestCaseType) { - return entries.filter((entry) => entry.testCase?.type !== typeToRemove); - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.html b/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.html deleted file mode 100644 index 3690bfdf0e3b..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.html +++ /dev/null @@ -1,11 +0,0 @@ - - - - {{ fileName }} - - {{ proportionString }} - -
- -
-
diff --git a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.scss b/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.scss deleted file mode 100644 index 915e18b1199e..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.scss +++ /dev/null @@ -1,20 +0,0 @@ -.mat-expansion-panel-header { - padding: 0; -} - -.mat-expansion-panel .mat-expansion-panel-body { - padding: 0 !important; -} - -.mat-expansion-panel-header-description { - flex-grow: 0; -} - -.mat-expansion-panel .mat-expansion-indicator::after { - margin-right: 2px !important; - margin-left: 2px !important; -} - -.covered-line-highlight { - background-color: var(--monaco-editor-test-coverage-highlight); -} diff --git a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.ts b/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.ts deleted file mode 100644 index b9fc28de4fbf..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component.ts +++ /dev/null @@ -1,91 +0,0 @@ -import { Component, Input, OnChanges, OnInit, SimpleChanges, ViewChild, ViewEncapsulation } from '@angular/core'; -import { CoverageFileReport } from 'app/entities/hestia/coverage-file-report.model'; -import { MonacoEditorComponent } from 'app/shared/monaco-editor/monaco-editor.component'; - -@Component({ - selector: 'jhi-testwise-coverage-file', - templateUrl: './testwise-coverage-file.component.html', - styleUrls: ['./testwise-coverage-file.component.scss'], - encapsulation: ViewEncapsulation.None, -}) -export class TestwiseCoverageFileComponent implements OnInit, OnChanges { - @Input() - fileContent: string; - - @Input() - fileName: string; - - @Input() - fileReport: CoverageFileReport; - - @ViewChild('editor', { static: true }) - editor: MonacoEditorComponent; - - proportionCoveredLines: number; - proportionString: string; - editorHeight: number = 20; - - static readonly COVERED_LINE_HIGHLIGHT_CLASS = 'covered-line-highlight'; - - ngOnInit(): void { - this.renderFile(); - this.editorHeight = this.editor.getContentHeight(); - } - - ngOnChanges(changes: SimpleChanges): void { - if (changes.fileReport || changes.fileContent) { - this.renderFile(); - } - } - - private aggregateCoveredLinesBlocks(fileReport: CoverageFileReport): Map { - const coveredLines = new Set(); - const entries = fileReport.testwiseCoverageEntries!; - // retrieve all covered line numbers - entries.forEach((entry) => { - this.getRangeArray(entry.startLine!, entry.lineCount!).forEach((line) => coveredLines.add(line)); - }); - - // build the blocks - const orderedLines = Array.from(coveredLines).sort(); - const startLineByLength = new Map(); - - // set the covered line ratio accordingly - this.proportionCoveredLines = orderedLines.length / this.fileReport!.lineCount!; - this.proportionString = `${(this.proportionCoveredLines * 100).toFixed(1)} %`; - - let index = 0; - while (index < orderedLines.length) { - const currentBlockStartLine = orderedLines[index]; - let currentBlockLength = 1; - let continueBlock = true; - - // count the length of the consecutive blocks - while (continueBlock && index < orderedLines.length) { - if (index + 1 === orderedLines.length) { - continueBlock = false; - } else if (orderedLines[index + 1] === orderedLines[index] + 1) { - currentBlockLength++; - } else { - continueBlock = false; - } - index++; - } - startLineByLength.set(currentBlockStartLine, currentBlockLength); - } - - return startLineByLength; - } - - private getRangeArray(startLine: number, lineCount: number): number[] { - return [...Array(lineCount).keys()].map((i) => i + startLine - 1); - } - - private renderFile() { - this.editor.changeModel(this.fileName, this.fileContent ?? ''); - this.editor.disposeLineHighlights(); - this.aggregateCoveredLinesBlocks(this.fileReport).forEach((blockLength, lineNumber) => { - this.editor.highlightLines(lineNumber + 1, lineNumber + blockLength, 'covered-line-highlight', 'covered-line-highlight'); - }); - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report-modal.component.html b/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report-modal.component.html deleted file mode 100644 index 01e96e404923..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report-modal.component.html +++ /dev/null @@ -1,11 +0,0 @@ -
- - -
diff --git a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report-modal.component.ts b/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report-modal.component.ts deleted file mode 100644 index 2183198c8772..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report-modal.component.ts +++ /dev/null @@ -1,21 +0,0 @@ -import { Component, Input } from '@angular/core'; -import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; -import { CoverageReport } from 'app/entities/hestia/coverage-report.model'; - -@Component({ - selector: 'jhi-testwise-coverage-report-modal', - templateUrl: './testwise-coverage-report-modal.component.html', -}) -export class TestwiseCoverageReportModalComponent { - @Input() - report: CoverageReport; - - @Input() - fileContentByPath: Map; - - constructor(protected activeModal: NgbActiveModal) {} - - close(): void { - this.activeModal.dismiss(); - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.component.html b/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.component.html deleted file mode 100644 index b16a1eccd579..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.component.html +++ /dev/null @@ -1,17 +0,0 @@ - -@for (entry of displayedTestCaseNames | keyvalue; track entry) { -
- - {{ entry.key }} -
-} -
-
-
-
-
-@for (entry of fileReportByFileName | keyvalue; track identifyCoverageFileComponent($index, entry)) { -
- -
-} diff --git a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.component.ts b/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.component.ts deleted file mode 100644 index a5f6b75f05a7..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.component.ts +++ /dev/null @@ -1,75 +0,0 @@ -import { Component, Input, OnInit } from '@angular/core'; -import { CoverageReport } from 'app/entities/hestia/coverage-report.model'; -import { CoverageFileReport } from 'app/entities/hestia/coverage-file-report.model'; - -@Component({ - selector: 'jhi-testwise-coverage-report', - templateUrl: './testwise-coverage-report.component.html', -}) -export class TestwiseCoverageReportComponent implements OnInit { - @Input() - report: CoverageReport; - - @Input() - fileContentByPath: Map; - - displayedTestCaseNames: Map; - - fileReportByFileName: Map; - - constructor() {} - - ngOnInit(): void { - // initially display covered lines for all test cases - this.displayedTestCaseNames = new Map(); - // retrieve all test cases - const testCases = new Set( - this.report?.fileReports - ?.flatMap((report) => report) - .flatMap((fileReport) => fileReport.testwiseCoverageEntries) - .map((entry) => entry!.testCase), - ); - testCases.forEach((testCase) => this.displayedTestCaseNames.set(testCase!.testName!, true)); - - this.setReportsByFileName(); - } - - changeReportsBySelectedTestCases(testCaseName: string): void { - const selected = this.displayedTestCaseNames.get(testCaseName); - this.displayedTestCaseNames.set(testCaseName, !selected); - this.setReportsByFileName(); - } - - private setReportsByFileName(): void { - const result = new Map(); - // create the reports for all files, not only for files that have existing coverage data - this.fileContentByPath?.forEach((content, filePath) => { - // do not include non-java/kotlin files - if (!(filePath.endsWith('.java') || filePath.endsWith('.kt'))) { - return; - } - const matchingFileReport = this.report.fileReports?.filter((fileReport) => fileReport?.filePath === filePath)?.first(); - const copiedFileReport = new CoverageFileReport(); - copiedFileReport.filePath = filePath; - if (matchingFileReport) { - copiedFileReport.lineCount = matchingFileReport.lineCount; - copiedFileReport.coveredLineCount = matchingFileReport.coveredLineCount; - - // filter out entries for the current file report - copiedFileReport.testwiseCoverageEntries = matchingFileReport.testwiseCoverageEntries?.filter((entry) => - this.displayedTestCaseNames.get(entry.testCase!.testName!), - ); - } else { - copiedFileReport.lineCount = content.split('\n').length; - copiedFileReport.coveredLineCount = 0; - copiedFileReport.testwiseCoverageEntries = []; - } - result.set(filePath, copiedFileReport); - }); - this.fileReportByFileName = result; - } - - identifyCoverageFileComponent(index: number, item: any) { - return item.key; - } -} diff --git a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.module.ts b/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.module.ts deleted file mode 100644 index c37bc7a255da..000000000000 --- a/src/main/webapp/app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.module.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { NgModule } from '@angular/core'; -import { ArtemisSharedModule } from 'app/shared/shared.module'; -import { TestwiseCoverageReportModalComponent } from 'app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report-modal.component'; -import { TestwiseCoverageReportComponent } from 'app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.component'; -import { TestwiseCoverageFileComponent } from 'app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component'; -import { MatExpansionModule } from '@angular/material/expansion'; -import { MonacoEditorComponent } from 'app/shared/monaco-editor/monaco-editor.component'; - -@NgModule({ - imports: [ArtemisSharedModule, MatExpansionModule, MonacoEditorComponent], - declarations: [TestwiseCoverageFileComponent, TestwiseCoverageReportComponent, TestwiseCoverageReportModalComponent], - exports: [TestwiseCoverageFileComponent, TestwiseCoverageReportModalComponent, TestwiseCoverageReportComponent], -}) -export class TestwiseCoverageReportModule {} diff --git a/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-grading-tasks-table.component.ts b/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-grading-tasks-table.component.ts index bd491c999df9..ec88a4dc9ae1 100644 --- a/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-grading-tasks-table.component.ts +++ b/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-grading-tasks-table.component.ts @@ -8,7 +8,7 @@ import { ProgrammingExerciseTask } from './programming-exercise-task'; import { Observable, Subject } from 'rxjs'; import { ProgrammingExerciseTestCase } from 'app/entities/programming/programming-exercise-test-case.model'; import { isExamExercise } from 'app/shared/util/utils'; -import { ProgrammingExerciseServerSideTask } from 'app/entities/hestia/programming-exercise-task.model'; +import { ProgrammingExerciseServerSideTask } from 'app/entities/programming-exercise-task.model'; type Sort = { by: 'name' | 'weight' | 'multiplier' | 'bonusPoints' | 'visibility' | 'resulting' | 'type'; diff --git a/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-task.service.ts b/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-task.service.ts index 5f45d551b476..3021e5e548f2 100644 --- a/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-task.service.ts +++ b/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-task.service.ts @@ -1,6 +1,6 @@ import { HttpClient, HttpErrorResponse } from '@angular/common/http'; import { Injectable } from '@angular/core'; -import { ProgrammingExerciseServerSideTask } from 'app/entities/hestia/programming-exercise-task.model'; +import { ProgrammingExerciseServerSideTask } from 'app/entities/programming-exercise-task.model'; import { Observable, catchError, of, tap } from 'rxjs'; import { Exercise } from 'app/entities/exercise.model'; import { ProgrammingExerciseTask } from 'app/exercises/programming/manage/grading/tasks/programming-exercise-task'; diff --git a/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-task.ts b/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-task.ts index 3a5844ddbb9e..101498cc8cf4 100644 --- a/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-task.ts +++ b/src/main/webapp/app/exercises/programming/manage/grading/tasks/programming-exercise-task.ts @@ -1,4 +1,4 @@ -import { ProgrammingExerciseServerSideTask } from 'app/entities/hestia/programming-exercise-task.model'; +import { ProgrammingExerciseServerSideTask } from 'app/entities/programming-exercise-task.model'; import { TestCaseStats } from 'app/entities/programming/programming-exercise-test-case-statistics.model'; import { ProgrammingExerciseTestCase, ProgrammingExerciseTestCaseType, Visibility } from 'app/entities/programming/programming-exercise-test-case.model'; diff --git a/src/main/webapp/app/exercises/programming/manage/programming-exercise-detail.component.html b/src/main/webapp/app/exercises/programming/manage/programming-exercise-detail.component.html index 95da3dea7a71..d3809ff3fdc8 100644 --- a/src/main/webapp/app/exercises/programming/manage/programming-exercise-detail.component.html +++ b/src/main/webapp/app/exercises/programming/manage/programming-exercise-detail.component.html @@ -40,15 +40,6 @@

{{ } - @if (programmingExercise.course && !isExamExercise) { - - - - - } @@ -134,27 +125,6 @@

{{ } - @if (programmingExercise.isAtLeastEditor) { - - @if ( - (programmingExercise.programmingLanguage === ProgrammingLanguage.JAVA || - programmingExercise.programmingLanguage === ProgrammingLanguage.KOTLIN) && - programmingExercise.buildConfig?.testwiseCoverageEnabled - ) { - - - - } - - } @if (programmingExercise.isAtLeastInstructor && !localVCEnabled) { - - } - @if (exercise?.id) { -
- - -
- } - -
- - -
- - - diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint-update.component.ts b/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint-update.component.ts deleted file mode 100644 index 260aba7f73e9..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint-update.component.ts +++ /dev/null @@ -1,196 +0,0 @@ -import { Component, OnDestroy, OnInit } from '@angular/core'; -import { HttpResponse } from '@angular/common/http'; -import { ActivatedRoute } from '@angular/router'; -import { Observable, Subscription, filter, switchMap } from 'rxjs'; -import { AlertService } from 'app/core/util/alert.service'; -import { ExerciseHintService } from '../shared/exercise-hint.service'; -import { ArtemisNavigationUtilService } from 'app/utils/navigation.utils'; -import { faBan, faCircleNotch, faSave } from '@fortawesome/free-solid-svg-icons'; -import { ExerciseHint, HintType } from 'app/entities/hestia/exercise-hint.model'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { ProgrammingExerciseServerSideTask } from 'app/entities/hestia/programming-exercise-task.model'; -import { ManualSolutionEntryCreationModalComponent } from 'app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component'; -import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap'; -import { CodeHint } from 'app/entities/hestia/code-hint-model'; -import { CodeHintService } from 'app/exercises/shared/exercise-hint/services/code-hint.service'; -import { onError } from 'app/shared/util/global.utils'; -import { IrisSettingsService } from 'app/iris/settings/shared/iris-settings.service'; -import { IrisSettings } from 'app/entities/iris/settings/iris-settings.model'; -import { ProfileService } from 'app/shared/layouts/profiles/profile.service'; -import { ButtonType } from 'app/shared/components/button.component'; -import { PROFILE_IRIS } from 'app/app.constants'; -import { FormulaAction } from 'app/shared/monaco-editor/model/actions/formula.action'; -import { MarkdownEditorHeight } from 'app/shared/markdown-editor/monaco/markdown-editor-monaco.component'; - -const DEFAULT_DISPLAY_THRESHOLD = 3; - -@Component({ - selector: 'jhi-exercise-hint-update', - templateUrl: './exercise-hint-update.component.html', -}) -export class ExerciseHintUpdateComponent implements OnInit, OnDestroy { - MarkdownEditorHeight = MarkdownEditorHeight; - - courseId: number; - exercise: ProgrammingExercise; - exerciseHint = new ExerciseHint(); - solutionEntries: ProgrammingExerciseSolutionEntry[]; - - programmingExercise: ProgrammingExercise; - tasks: ProgrammingExerciseServerSideTask[]; - irisSettings?: IrisSettings; - - isSaving: boolean; - isGeneratingDescription: boolean; - paramSub: Subscription; - - domainActions = [new FormulaAction()]; - - // Icons - faCircleNotch = faCircleNotch; - faBan = faBan; - faSave = faSave; - - // Enums - readonly HintType = HintType; - readonly ButtonType = ButtonType; - - constructor( - private route: ActivatedRoute, - protected alertService: AlertService, - private modalService: NgbModal, - protected codeHintService: CodeHintService, - protected exerciseHintService: ExerciseHintService, - private programmingExerciseService: ProgrammingExerciseService, - private navigationUtilService: ArtemisNavigationUtilService, - protected irisSettingsService: IrisSettingsService, - private profileService: ProfileService, - ) {} - - /** - * Fetches the exercise from the server and assigns it on the exercise hint - */ - ngOnInit() { - this.paramSub = this.route.params.subscribe((params) => { - this.courseId = params['courseId']; - this.isSaving = false; - this.isGeneratingDescription = false; - }); - this.route.data.subscribe(({ exerciseHint, exercise }) => { - this.exercise = exercise; - exerciseHint.exercise = exercise; - this.exerciseHint = exerciseHint; - this.exerciseHint.displayThreshold = this.exerciseHint.displayThreshold ?? DEFAULT_DISPLAY_THRESHOLD; - - this.programmingExerciseService.getTasksAndTestsExtractedFromProblemStatement(this.exercise.id!).subscribe((tasks) => { - this.tasks = tasks; - - const selectedTask = this.tasks.find((task) => task.id === this.exerciseHint.programmingExerciseTask?.id); - if (selectedTask) { - this.exerciseHint.programmingExerciseTask = selectedTask; - } else if (tasks.length) { - this.exerciseHint.programmingExerciseTask = this.tasks[0]; - } - }); - - this.profileService - .getProfileInfo() - .pipe( - filter((profileInfo) => profileInfo?.activeProfiles?.includes(PROFILE_IRIS)), - switchMap(() => this.irisSettingsService.getCombinedExerciseSettings(this.exercise.id!)), - ) - .subscribe((settings) => { - this.irisSettings = settings; - }); - }); - } - - openManualEntryCreationModal() { - const codeHint = this.exerciseHint as CodeHint; - const testCasesForCurrentTask = codeHint.programmingExerciseTask?.testCases ?? []; - const modalRef: NgbModalRef = this.modalService.open(ManualSolutionEntryCreationModalComponent as Component, { size: 'lg', backdrop: 'static' }); - modalRef.componentInstance.exerciseId = this.exercise.id!; - modalRef.componentInstance.codeHint = codeHint; - modalRef.componentInstance.testCases = testCasesForCurrentTask; - modalRef.componentInstance.onEntryCreated.subscribe((createdEntry: ProgrammingExerciseSolutionEntry) => { - codeHint!.solutionEntries!.push(createdEntry); - }); - } - - /** - * Unsubscribes from the param subscription - */ - ngOnDestroy(): void { - if (this.paramSub) { - this.paramSub.unsubscribe(); - } - } - - /** - * Setter to update the exercise hint content - * @param newContent New value to set - */ - updateHintContent(newContent: string) { - this.exerciseHint.content = newContent; - } - - /** - * Navigate to the previous page when the user cancels the update process - * Returns to the detail page if there is no previous state and we edited an existing hint - * Returns to the overview page if there is no previous state and we created a new hint - */ - previousState() { - this.navigationUtilService.navigateBackWithOptional( - ['course-management', this.courseId.toString(), 'programming-exercises', this.exercise.id!.toString(), 'hints'], - this.exerciseHint.id?.toString(), - ); - } - - /** - * Saves the exercise hint by creating or updating it on the server - */ - save() { - this.isSaving = true; - if (this.exerciseHint.id !== undefined) { - this.subscribeToSaveResponse(this.exerciseHintService.update(this.exercise.id!, this.exerciseHint)); - } else { - this.subscribeToSaveResponse(this.exerciseHintService.create(this.exercise.id!, this.exerciseHint)); - } - } - - generateDescriptionForCodeHint() { - if (((this.exerciseHint as CodeHint).solutionEntries?.length ?? 0) === 0) { - return; - } - this.isGeneratingDescription = true; - this.codeHintService.generateDescriptionForCodeHint(this.exercise.id!, this.exerciseHint.id!).subscribe({ - next: (response) => { - this.exerciseHint.description = response.body!.description; - this.exerciseHint.content = response.body!.content; - this.isGeneratingDescription = false; - }, - error: (error) => { - this.isGeneratingDescription = false; - onError(this.alertService, error); - }, - }); - } - - protected subscribeToSaveResponse(result: Observable>) { - result.subscribe({ - next: () => this.onSaveSuccess(), - error: () => this.onSaveError(), - }); - } - - protected onSaveSuccess() { - this.isSaving = false; - this.previousState(); - } - - protected onSaveError() { - this.isSaving = false; - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.component.html b/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.component.html deleted file mode 100644 index 59db6e86ff2f..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.component.html +++ /dev/null @@ -1,110 +0,0 @@ -
-

- - @if (exercise?.programmingLanguage === ProgrammingLanguage.JAVA) { - - @if (containsCodeHints) { -
- - Manage Code Hints -
- } - @if (!containsCodeHints) { -
- - -
- } -
- } - - - - -

-
- @if (exerciseHints?.length === 0) { -
- -
- } - @if (exerciseHints && exerciseHints.length > 0) { -
- - - - - - - - - - - - @for (exerciseHint of exerciseHints; track trackId($index, exerciseHint)) { - - - - - - - - } - -
- {{ exerciseHint.id }} - - {{ exerciseHint.title }} -
- @if (exerciseHint.type === HintType.CODE && exerciseHint.exercise?.type === ExerciseType.PROGRAMMING) { - {{ 'artemisApp.codeHint.type' | artemisTranslate }} - } - @if (exerciseHint.type === HintType.TEXT && exerciseHint.exercise?.type === ExerciseType.PROGRAMMING) { - {{ 'artemisApp.exerciseHint.textHint' | artemisTranslate }} - } -
-
- -
- @if (exerciseHint.type === HintType.CODE) { -
- - {{ (exerciseHint | castToCodeHint).solutionEntries?.length ?? 0 }} - {{ - ((exerciseHint | castToCodeHint).solutionEntries?.length === 1 ? 'artemisApp.codeHint.entry' : 'artemisApp.codeHint.entries') - | artemisTranslate - }} - -
- } -
- {{ exerciseHint.exercise?.id }} - -
- - - - - - - - - -
-
-
- } -
diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.component.ts b/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.component.ts deleted file mode 100644 index 3b542f17643b..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.component.ts +++ /dev/null @@ -1,127 +0,0 @@ -import { Component, OnDestroy, OnInit } from '@angular/core'; -import { ActivatedRoute } from '@angular/router'; -import { HttpErrorResponse, HttpResponse } from '@angular/common/http'; -import { Subject, Subscription } from 'rxjs'; -import { filter, map } from 'rxjs/operators'; - -import { ExerciseHintService } from '../shared/exercise-hint.service'; -import { onError } from 'app/shared/util/global.utils'; -import { AlertService } from 'app/core/util/alert.service'; -import { EventManager } from 'app/core/util/event-manager.service'; -import { faArrowsRotate, faCode, faEye, faFont, faPlus, faTimes, faWrench } from '@fortawesome/free-solid-svg-icons'; -import { ExerciseHint, HintType } from 'app/entities/hestia/exercise-hint.model'; -import { ExerciseType } from 'app/entities/exercise.model'; -import { ProgrammingExercise, ProgrammingLanguage } from 'app/entities/programming/programming-exercise.model'; - -@Component({ - selector: 'jhi-exercise-hint', - templateUrl: './exercise-hint.component.html', -}) -export class ExerciseHintComponent implements OnInit, OnDestroy { - readonly HintType = HintType; - ExerciseType = ExerciseType; - exercise: ProgrammingExercise; - exerciseHints: ExerciseHint[]; - containsCodeHints: boolean; - eventSubscriber: Subscription; - - private dialogErrorSource = new Subject(); - dialogError$ = this.dialogErrorSource.asObservable(); - - paramSub: Subscription; - - // Icons - faPlus = faPlus; - faTimes = faTimes; - faEye = faEye; - faWrench = faWrench; - faText = faFont; - faCode = faCode; - faArrowsRotate = faArrowsRotate; - - readonly ProgrammingLanguage = ProgrammingLanguage; - - constructor( - private route: ActivatedRoute, - protected exerciseHintService: ExerciseHintService, - private alertService: AlertService, - protected eventManager: EventManager, - ) {} - - /** - * Subscribes to the route params to act on the currently selected exercise. - */ - ngOnInit() { - this.route.data.subscribe(({ exercise }) => { - this.exercise = exercise; - this.loadAllByExerciseId(); - this.registerChangeInExerciseHints(); - }); - } - - /** - * Unsubscribe from subscriptions - */ - ngOnDestroy() { - if (this.paramSub) { - this.paramSub.unsubscribe(); - } - this.eventManager.destroy(this.eventSubscriber); - this.dialogErrorSource.unsubscribe(); - } - - /** - * Load all exercise hints with the currently selected exerciseId (taken from route params). - */ - loadAllByExerciseId() { - this.exerciseHintService - .findByExerciseId(this.exercise.id!) - .pipe( - filter((res: HttpResponse) => res.ok), - map((res: HttpResponse) => res.body), - ) - .subscribe({ - next: (res: ExerciseHint[]) => { - this.exerciseHints = res; - this.containsCodeHints = this.exerciseHints?.some((hint) => hint.type === HintType.CODE); - }, - error: (res: HttpErrorResponse) => onError(this.alertService, res), - }); - } - - /** - * Returns the track id of an exercise hint - * @param index Index of the item - * @param item Item for which to get the id - */ - trackId(index: number, item: ExerciseHint) { - return item.id; - } - - /** - * (Re-)subscribe to the exercise hint list modification subscription - */ - registerChangeInExerciseHints() { - if (this.eventSubscriber) { - this.eventSubscriber.unsubscribe(); - } - this.eventSubscriber = this.eventManager.subscribe('exerciseHintListModification', () => this.loadAllByExerciseId()); - } - - /** - * Deletes exercise hint - * @param exerciseHintId the id of the exercise hint that we want to delete - */ - deleteExerciseHint(exerciseHintId: number) { - this.exerciseHintService.delete(this.exercise.id!, exerciseHintId).subscribe({ - next: () => { - this.eventManager.broadcast({ - name: 'exerciseHintListModification', - content: 'Deleted an exerciseHint', - }); - this.dialogErrorSource.next(''); - }, - error: (error: HttpErrorResponse) => this.dialogErrorSource.next(error.message), - }); - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.route.ts b/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.route.ts deleted file mode 100644 index dd64cf79d51e..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/manage/exercise-hint.route.ts +++ /dev/null @@ -1,96 +0,0 @@ -import { Injectable } from '@angular/core'; -import { HttpResponse } from '@angular/common/http'; -import { ActivatedRouteSnapshot, Resolve, Routes } from '@angular/router'; -import { UserRouteAccessService } from 'app/core/auth/user-route-access-service'; -import { of } from 'rxjs'; -import { filter, map } from 'rxjs/operators'; -import { ExerciseHintService } from '../shared/exercise-hint.service'; -import { ExerciseHintComponent } from './exercise-hint.component'; -import { ExerciseHintDetailComponent } from './exercise-hint-detail.component'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; -import { Authority } from 'app/shared/constants/authority.constants'; -import { exerciseTypes } from 'app/entities/exercise.model'; -import { ExerciseHintUpdateComponent } from 'app/exercises/shared/exercise-hint/manage/exercise-hint-update.component'; -import { ProgrammingExerciseResolve } from 'app/exercises/programming/manage/programming-exercise-management-routing.module'; - -@Injectable({ providedIn: 'root' }) -export class ExerciseHintResolve implements Resolve { - constructor(private service: ExerciseHintService) {} - - /** - * Resolves the route into an exercise hint id and fetches it from the server - * @param route Route which to resolve - */ - resolve(route: ActivatedRouteSnapshot) { - const exerciseId = route.params['exerciseId'] ? route.params['exerciseId'] : undefined; - const hintId = route.params['hintId'] ? route.params['hintId'] : undefined; - if (exerciseId && hintId) { - return this.service.find(exerciseId, hintId).pipe( - filter((response: HttpResponse) => response.ok), - map((exerciseHint: HttpResponse) => exerciseHint.body!), - ); - } - return of(new ExerciseHint()); - } -} - -export const exerciseHintRoute: Routes = [ - ...exerciseTypes.map((exerciseType) => { - return { - path: ':courseId/' + exerciseType + '-exercises/:exerciseId/exercise-hints/new', - component: ExerciseHintUpdateComponent, - resolve: { - exercise: ProgrammingExerciseResolve, - exerciseHint: ExerciseHintResolve, - }, - data: { - authorities: [Authority.EDITOR, Authority.INSTRUCTOR, Authority.ADMIN], - pageTitle: 'artemisApp.exerciseHint.home.title', - }, - canActivate: [UserRouteAccessService], - }; - }), - ...exerciseTypes.map((exerciseType) => { - return { - path: ':courseId/' + exerciseType + '-exercises/:exerciseId/exercise-hints/:hintId', - component: ExerciseHintDetailComponent, - resolve: { - exerciseHint: ExerciseHintResolve, - }, - data: { - authorities: [Authority.EDITOR, Authority.INSTRUCTOR, Authority.ADMIN], - pageTitle: 'artemisApp.exerciseHint.home.title', - }, - canActivate: [UserRouteAccessService], - }; - }), - ...exerciseTypes.map((exerciseType) => { - return { - path: ':courseId/' + exerciseType + '-exercises/:exerciseId/exercise-hints/:hintId/edit', - component: ExerciseHintUpdateComponent, - resolve: { - exercise: ProgrammingExerciseResolve, - exerciseHint: ExerciseHintResolve, - }, - data: { - authorities: [Authority.EDITOR, Authority.INSTRUCTOR, Authority.ADMIN], - pageTitle: 'artemisApp.exerciseHint.home.title', - }, - canActivate: [UserRouteAccessService], - }; - }), - ...exerciseTypes.map((exerciseType) => { - return { - path: ':courseId/' + exerciseType + '-exercises/:exerciseId/exercise-hints', - component: ExerciseHintComponent, - resolve: { - exercise: ProgrammingExerciseResolve, - }, - data: { - authorities: [Authority.EDITOR, Authority.INSTRUCTOR, Authority.ADMIN], - pageTitle: 'artemisApp.exerciseHint.home.title', - }, - canActivate: [UserRouteAccessService], - }; - }), -]; diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.html b/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.html deleted file mode 100644 index 01e2077a3ae0..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.html +++ /dev/null @@ -1,13 +0,0 @@ -
- @if (availableExerciseHints && activatedExerciseHints && availableExerciseHints.length + activatedExerciseHints.length > 0) { - - } -
diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.scss b/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.scss deleted file mode 100644 index 98247be1b9c1..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.scss +++ /dev/null @@ -1,20 +0,0 @@ -:host { - position: sticky; - bottom: 0; - left: 0; - z-index: 100; - overflow: hidden; - display: flex; - flex-direction: column; - align-items: flex-end; -} - -.btn { - box-shadow: none; -} - -.animated-button { - animation: 0.4s highlight; - animation-delay: 0.8s; - animation-iteration-count: 3; -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.ts b/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.ts deleted file mode 100644 index 07e11494103b..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component.ts +++ /dev/null @@ -1,41 +0,0 @@ -import { Component, EventEmitter, Input, Output } from '@angular/core'; -import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; -import { faCircleQuestion } from '@fortawesome/free-solid-svg-icons'; -import { ExerciseHintStudentDialogComponent } from 'app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component'; -import cloneDeep from 'lodash-es/cloneDeep'; - -@Component({ - selector: 'jhi-exercise-hint-button-overlay', - templateUrl: './exercise-hint-button-overlay.component.html', - styleUrls: ['./exercise-hint-button-overlay.component.scss'], -}) -export class ExerciseHintButtonOverlayComponent { - @Input() - availableExerciseHints?: ExerciseHint[]; - @Input() - activatedExerciseHints?: ExerciseHint[]; - @Output() - onHintActivated = new EventEmitter(); - - faCircleQuestion = faCircleQuestion; - ngbModalRef?: NgbModalRef; - - constructor(private modalService: NgbModal) {} - - openModal() { - this.ngbModalRef = this.modalService.open(ExerciseHintStudentDialogComponent as Component, { size: 'lg', backdrop: 'static' }); - this.ngbModalRef.componentInstance.onHintActivated = this.onHintActivated; - // cloning is required that the lists not change while modal is open - this.ngbModalRef.componentInstance.activatedExerciseHints = cloneDeep(this.activatedExerciseHints!); - this.ngbModalRef.componentInstance.availableExerciseHints = cloneDeep(this.availableExerciseHints!); - this.ngbModalRef.result.then( - () => { - this.ngbModalRef = undefined; - }, - () => { - this.ngbModalRef = undefined; - }, - ); - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.html b/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.html deleted file mode 100644 index 7dc825d8aa95..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.html +++ /dev/null @@ -1,56 +0,0 @@ - - - - - -
-
{{ exerciseHint!.description }}
-
{{ exerciseHint!.programmingExerciseTask?.taskName }}
-
-
- - -
-
- - - - @if (isLoading) { -
-
- -
-
- } @else { -
- @if (exerciseHint?.content) { -
- } - @if (exerciseHint.type === HintType.CODE && (exerciseHint | castToCodeHint)?.solutionEntries) { -
- -
- } -
- How helpful was this hint? - -
-
- } -
diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.scss b/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.scss deleted file mode 100644 index 0768618c0d5d..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.scss +++ /dev/null @@ -1,31 +0,0 @@ -.mat-expansion-panel.exercise-hint-student-expansion-panel-wrapper { - .mat-expansion-panel-header { - padding: 10px 10px 10px 0; - font-weight: bold; - } - - .mat-expansion-panel-header-title { - flex-grow: 0; - } - - .mat-expansion-panel-header-description { - flex-grow: 3; - } - - .task-name { - font-size: 12px; - font-weight: 300; - } - - .mat-expansion-panel-body { - padding: 10px 0 0; - } - - .activated { - background-color: var(--info); - } - - .not_activated { - background-color: var(--success); - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.ts b/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.ts deleted file mode 100644 index bc787e798f87..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component.ts +++ /dev/null @@ -1,58 +0,0 @@ -import { Component, EventEmitter, Input, Output, ViewEncapsulation } from '@angular/core'; -import { faQuestionCircle } from '@fortawesome/free-solid-svg-icons'; - -import { ExerciseHint, HintType } from 'app/entities/hestia/exercise-hint.model'; -import { ExerciseHintResponse, ExerciseHintService } from 'app/exercises/shared/exercise-hint/shared/exercise-hint.service'; -import { StarRatingComponent } from 'app/exercises/shared/rating/star-rating/star-rating.component'; - -/** - * This component is a modal that shows the exercise's hints. - */ -@Component({ - selector: 'jhi-exercise-hint-expandable', - templateUrl: './exercise-hint-expandable.component.html', - styleUrls: ['./exercise-hint-expandable.component.scss'], - encapsulation: ViewEncapsulation.None, -}) -export class ExerciseHintExpandableComponent { - @Input() exerciseHint: ExerciseHint; - @Input() hasUsed: boolean; - @Output() - onHintActivated = new EventEmitter(); - - expanded = false; - isLoading = false; - - faQuestionCircle = faQuestionCircle; - - readonly HintType = HintType; - - constructor(private exerciseHintService: ExerciseHintService) {} - - displayHintContent() { - this.expanded = true; - - if (this.hasUsed) { - // the hint already contains the content - return; - } - - this.isLoading = true; - this.exerciseHintService.activateExerciseHint(this.exerciseHint!.exercise!.id!, this.exerciseHint!.id!).subscribe((res: ExerciseHintResponse) => { - this.exerciseHint = res.body!; - this.hasUsed = true; - this.isLoading = false; - this.onHintActivated.emit(this.exerciseHint); - }); - } - - collapseContent() { - this.expanded = false; - } - - onRate(event: { oldValue: number; newValue: number; starRating: StarRatingComponent }) { - this.exerciseHintService.rateExerciseHint(this.exerciseHint!.exercise!.id!, this.exerciseHint!.id!, event.newValue).subscribe(() => { - this.exerciseHint.currentUserRating = event.newValue; - }); - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-participation.module.ts b/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-participation.module.ts deleted file mode 100644 index 3742abfc8de0..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-participation.module.ts +++ /dev/null @@ -1,26 +0,0 @@ -import { NgModule } from '@angular/core'; -import { MatExpansionModule } from '@angular/material/expansion'; -import { ArtemisSharedModule } from 'app/shared/shared.module'; -import { ExerciseHintStudentDialogComponent } from 'app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component'; -import { ArtemisMarkdownModule } from 'app/shared/markdown.module'; -import { ArtemisExerciseHintManagementModule } from 'app/exercises/shared/exercise-hint/manage/exercise-hint-management.module'; -import { ArtemisExerciseHintSharedModule } from 'app/exercises/shared/exercise-hint/shared/exercise-hint-shared.module'; -import { ExerciseHintExpandableComponent } from 'app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component'; -import { ExerciseHintButtonOverlayComponent } from 'app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component'; -import { RatingModule } from 'app/exercises/shared/rating/rating.module'; -import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; - -@NgModule({ - imports: [ - ArtemisSharedModule, - ArtemisMarkdownModule, - ArtemisExerciseHintManagementModule, - ArtemisExerciseHintSharedModule, - RatingModule, - MatExpansionModule, - ArtemisSharedComponentModule, - ], - declarations: [ExerciseHintStudentDialogComponent, ExerciseHintExpandableComponent, ExerciseHintButtonOverlayComponent], - exports: [ExerciseHintStudentDialogComponent, ExerciseHintExpandableComponent, ExerciseHintExpandableComponent, ExerciseHintButtonOverlayComponent], -}) -export class ArtemisExerciseHintParticipationModule {} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component.html b/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component.html deleted file mode 100644 index 09c17845b507..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component.html +++ /dev/null @@ -1,32 +0,0 @@ - - diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component.ts b/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component.ts deleted file mode 100644 index 4e5d174b28bd..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component.ts +++ /dev/null @@ -1,26 +0,0 @@ -import { Component, EventEmitter, Input, Output } from '@angular/core'; -import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; - -/** - * This component is a modal that shows the exercise's hints. - */ -@Component({ - selector: 'jhi-exercise-hint-student-dialog', - templateUrl: './exercise-hint-student-dialog.component.html', -}) -export class ExerciseHintStudentDialogComponent { - @Input() availableExerciseHints: ExerciseHint[]; - @Input() activatedExerciseHints: ExerciseHint[]; - @Output() - onHintActivated = new EventEmitter(); - - constructor(public activeModal: NgbActiveModal) {} - - /** - * Dismisses the modal - */ - clear() { - this.activeModal.dismiss('cancel'); - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.scss b/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.scss deleted file mode 100644 index a077cc1f55cd..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.scss +++ /dev/null @@ -1,4 +0,0 @@ -.hint-icon { - font-size: 1.5em; - cursor: pointer; -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/services/code-hint-cast.pipe.ts b/src/main/webapp/app/exercises/shared/exercise-hint/services/code-hint-cast.pipe.ts deleted file mode 100644 index c648527cef38..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/services/code-hint-cast.pipe.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { Pipe, PipeTransform } from '@angular/core'; -import { CodeHint } from 'app/entities/hestia/code-hint-model'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; - -/** - * Pipe to transform an ExerciseHint into a CodeHint - */ -@Pipe({ - name: 'castToCodeHint', - pure: true, -}) -export class CastToCodeHintPipe implements PipeTransform { - transform(exerciseHint: ExerciseHint): CodeHint { - return exerciseHint as CodeHint; - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/services/code-hint.service.ts b/src/main/webapp/app/exercises/shared/exercise-hint/services/code-hint.service.ts deleted file mode 100644 index 2796e41b9f2c..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/services/code-hint.service.ts +++ /dev/null @@ -1,68 +0,0 @@ -import { Injectable } from '@angular/core'; -import { HttpClient, HttpResponse } from '@angular/common/http'; -import { Observable } from 'rxjs'; -import { CodeHint } from 'app/entities/hestia/code-hint-model'; -import { createRequestOption } from 'app/shared/util/request.util'; - -export interface ICodeHintService { - /** - * Generates the code hints for a programming exercise - * @param exerciseId Id of the programming exercise - * @param deleteOldCodeHints Whether the old code hints should be deleted - */ - generateCodeHintsForExercise(exerciseId: number, deleteOldCodeHints: boolean): Observable; - - /** - * Generates the description for a code hint using Iris - * @param exerciseId of the programming exercise - * @param codeHintId of the code hint for which the description will be generated - */ - generateDescriptionForCodeHint(exerciseId: number, codeHintId: number): Observable>; - - /** - * Removes a programming exercise solution entry from a code hint - * @param exerciseId of the programming exercise - * @param codeHintId of the code hint from which the solution entry will be removed - * @param solutionEntryId of the solution entry to be removed - */ - removeSolutionEntryFromCodeHint(exerciseId: number, codeHintId: number, solutionEntryId: number): Observable>; -} - -@Injectable({ providedIn: 'root' }) -export class CodeHintService implements ICodeHintService { - public resourceUrl = 'api/programming-exercises'; - - constructor(protected http: HttpClient) {} - - /** - * Generates the code hints for a programming exercise - * @param exerciseId Id of the programming exercise - * @param deleteOldCodeHints Whether the old code hints should be deleted - */ - generateCodeHintsForExercise(exerciseId: number, deleteOldCodeHints: boolean): Observable { - const options = createRequestOption({ deleteOldCodeHints }); - return this.http.post(`${this.resourceUrl}/${exerciseId}/code-hints`, undefined, { - params: options, - }); - } - - /** - * Generates the description for a code hint using Iris - * @param exerciseId of the programming exercise - * @param codeHintId of the code hint for which the description will be generated - */ - generateDescriptionForCodeHint(exerciseId: number, codeHintId: number): Observable> { - return this.http.post(`${this.resourceUrl}/${exerciseId}/code-hints/${codeHintId}/generate-description`, undefined, { observe: 'response' }); - } - - /** - * Removes a programming exercise solution entry from a code hint. Only removes the linkage between, but does not - * delete the entry itself. - * @param exerciseId of the programming exercise - * @param codeHintId of the code hint from which the solution entry will be removed - * @param solutionEntryId of the solution entry to be removed - */ - removeSolutionEntryFromCodeHint(exerciseId: number, codeHintId: number, solutionEntryId: number): Observable> { - return this.http.delete(`${this.resourceUrl}/${exerciseId}/code-hints/${codeHintId}/solution-entries/${solutionEntryId}`, { observe: 'response' }); - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/services/programming-exercise-solution-entry.service.ts b/src/main/webapp/app/exercises/shared/exercise-hint/services/programming-exercise-solution-entry.service.ts deleted file mode 100644 index 45af7c818acb..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/services/programming-exercise-solution-entry.service.ts +++ /dev/null @@ -1,97 +0,0 @@ -import { Injectable } from '@angular/core'; -import { HttpClient } from '@angular/common/http'; -import { Observable } from 'rxjs'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; - -export interface IProgrammingExerciseSolutionEntryService { - /** - * Create a custom solution entry for a programming exercise and test case. - * @param exerciseId of the programming exercise - * @param testCaseId of the test case that the entry relates to - * @param entry the entry to be created - */ - createSolutionEntry(exerciseId: number, testCaseId: number, entry: ProgrammingExerciseSolutionEntry): Observable; - - /** - * Get all solution entries for a programming exercise. - * @param exerciseId of the programming exercise - */ - getSolutionEntriesForExercise(exerciseId: number): Observable; - - /** - * Update a solution entry and returns the updated entry. - * @param exerciseId of the programming exercise - * @param testCaseId of the test case - * @param solutionEntryId of the solution entry to update - * @param entry the entry to update - */ - updateSolutionEntry(exerciseId: number, testCaseId: number, solutionEntryId: number, entry: ProgrammingExerciseSolutionEntry): Observable; - - /** - * Delete a solution entry. - * @param exerciseId of the programming exercise - * @param testCaseId of the test case - * @param solutionEntryId of the solution entry to delete - */ - deleteSolutionEntry(exerciseId: number, testCaseId: number, solutionEntryId: number): Observable; - - /** - * Delete all solution entries for a programming exercise. - * @param exerciseId of the programming exercise - */ - deleteAllSolutionEntriesForExercise(exerciseId: number): Observable; -} - -@Injectable({ providedIn: 'root' }) -export class ProgrammingExerciseSolutionEntryService implements IProgrammingExerciseSolutionEntryService { - public resourceUrl = 'api/programming-exercises'; - - constructor(protected http: HttpClient) {} - - /** - * Create a custom solution entry for a programming exercise and test case. - * @param exerciseId of the programming exercise - * @param testCaseId of the test case that the entry relates to - * @param entry the entry to be created - */ - createSolutionEntry(exerciseId: number, testCaseId: number, entry: ProgrammingExerciseSolutionEntry): Observable { - return this.http.post(`${this.resourceUrl}/${exerciseId}/test-cases/${testCaseId}/solution-entries`, entry); - } - - /** - * Get all solution entries for a programming exercise. - * @param exerciseId of the programming exercise - */ - getSolutionEntriesForExercise(exerciseId: number): Observable { - return this.http.get(`${this.resourceUrl}/${exerciseId}/solution-entries`); - } - - /** - * Update a solution entry and returns the updated entry. - * @param exerciseId of the programming exercise - * @param testCaseId of the test case - * @param solutionEntryId of the solution entry to update - * @param entry the entry to update - */ - updateSolutionEntry(exerciseId: number, testCaseId: number, solutionEntryId: number, entry: ProgrammingExerciseSolutionEntry): Observable { - return this.http.put(`${this.resourceUrl}/${exerciseId}/test-cases/${testCaseId}/solution-entries/${solutionEntryId}`, entry); - } - - /** - * Delete a solution entry. - * @param exerciseId of the programming exercise - * @param testCaseId of the test case - * @param solutionEntryId of the solution entry to delete - */ - deleteSolutionEntry(exerciseId: number, testCaseId: number, solutionEntryId: number) { - return this.http.delete(`${this.resourceUrl}/${exerciseId}/test-cases/${testCaseId}/solution-entries/${solutionEntryId}`); - } - - /** - * Delete all solution entries for a programming exercise. - * @param exerciseId of the programming exercise - */ - deleteAllSolutionEntriesForExercise(exerciseId: number) { - return this.http.delete(`${this.resourceUrl}/${exerciseId}/solution-entries`); - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/shared/code-hint-container.component.html b/src/main/webapp/app/exercises/shared/exercise-hint/shared/code-hint-container.component.html deleted file mode 100644 index d62c18f4e20e..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/shared/code-hint-container.component.html +++ /dev/null @@ -1,25 +0,0 @@ -@for (solutionEntry of sortedSolutionEntries; track solutionEntry) { -
-
- - @if (enableEditing) { -
- -
- } -
-
-} -@if (!sortedSolutionEntries.length) { -
- -
-} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/shared/code-hint-container.component.ts b/src/main/webapp/app/exercises/shared/exercise-hint/shared/code-hint-container.component.ts deleted file mode 100644 index 8bd962711fdb..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/shared/code-hint-container.component.ts +++ /dev/null @@ -1,68 +0,0 @@ -import { Component, Input, OnDestroy, OnInit } from '@angular/core'; -import { ActivatedRoute } from '@angular/router'; -import { Subject } from 'rxjs'; - -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { CodeHint } from 'app/entities/hestia/code-hint-model'; -import { CodeHintService } from 'app/exercises/shared/exercise-hint/services/code-hint.service'; -import { faTimes } from '@fortawesome/free-solid-svg-icons'; - -/** - * Component containing the solution entries for a {@link CodeHint}. - * The entries are sorted by name (primary) and start line number (secondary) - */ -@Component({ - selector: 'jhi-code-hint-container', - templateUrl: './code-hint-container.component.html', -}) -export class CodeHintContainerComponent implements OnInit, OnDestroy { - @Input() - codeHint: CodeHint; - - @Input() - enableEditing = false; - - sortedSolutionEntries: ProgrammingExerciseSolutionEntry[]; - - private dialogErrorSource = new Subject(); - dialogError$ = this.dialogErrorSource.asObservable(); - - faTimes = faTimes; - - constructor( - protected route: ActivatedRoute, - private codeHintService: CodeHintService, - ) {} - - ngOnInit() { - this.setSortedSolutionEntriesForCodeHint(); - } - - ngOnDestroy(): void { - this.dialogErrorSource.unsubscribe(); - } - - setSortedSolutionEntriesForCodeHint() { - this.sortedSolutionEntries = - this.codeHint.solutionEntries?.sort((a, b) => { - return a.filePath?.localeCompare(b.filePath!) || a.line! - b.line!; - }) ?? []; - } - - /** - * Removes a solution entry from the code hint - * @param solutionEntryId of the solution entry to be removed - */ - removeEntryFromCodeHint(solutionEntryId: number) { - this.codeHintService.removeSolutionEntryFromCodeHint(this.codeHint.exercise!.id!, this.codeHint.id!, solutionEntryId).subscribe({ - next: () => { - this.sortedSolutionEntries = this.sortedSolutionEntries.filter((entry) => entry.id !== solutionEntryId); - this.codeHint.solutionEntries = this.sortedSolutionEntries; - this.dialogErrorSource.next(''); - }, - error: (error) => { - this.dialogErrorSource.next(error.message); - }, - }); - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/shared/exercise-hint-shared.module.ts b/src/main/webapp/app/exercises/shared/exercise-hint/shared/exercise-hint-shared.module.ts deleted file mode 100644 index 08a2b1c8ee07..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/shared/exercise-hint-shared.module.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { NgModule } from '@angular/core'; -import { ArtemisSharedModule } from 'app/shared/shared.module'; -import { CastToCodeHintPipe } from 'app/exercises/shared/exercise-hint/services/code-hint-cast.pipe'; -import { SolutionEntryComponent } from 'app/exercises/shared/exercise-hint/shared/solution-entry.component'; -import { CodeHintContainerComponent } from 'app/exercises/shared/exercise-hint/shared/code-hint-container.component'; -import { ArtemisMarkdownModule } from 'app/shared/markdown.module'; -import { MonacoEditorComponent } from 'app/shared/monaco-editor/monaco-editor.component'; - -@NgModule({ - imports: [ArtemisSharedModule, ArtemisMarkdownModule, MonacoEditorComponent], - declarations: [SolutionEntryComponent, CodeHintContainerComponent, CastToCodeHintPipe], - exports: [SolutionEntryComponent, CodeHintContainerComponent, CastToCodeHintPipe], -}) -export class ArtemisExerciseHintSharedModule {} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/shared/exercise-hint.service.ts b/src/main/webapp/app/exercises/shared/exercise-hint/shared/exercise-hint.service.ts deleted file mode 100644 index 88c29bef59a6..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/shared/exercise-hint.service.ts +++ /dev/null @@ -1,179 +0,0 @@ -import { Injectable } from '@angular/core'; -import { HttpClient, HttpResponse } from '@angular/common/http'; -import { Observable, tap } from 'rxjs'; - -import { ExerciseService } from 'app/exercises/shared/exercise/exercise.service'; -import { ExerciseHint, HintType } from 'app/entities/hestia/exercise-hint.model'; -import { EntityTitleService, EntityType } from 'app/shared/layouts/navbar/entity-title.service'; - -export type ExerciseHintResponse = HttpResponse; - -export interface IExerciseHintService { - /** - * Creates an exercise hint - * @param exerciseId of the exercise - * @param exerciseHint Exercise hint to create - */ - create(exerciseId: number, exerciseHint: ExerciseHint): Observable; - - /** - * Updates an exercise hint - * @param exerciseId of the exercise - * @param exerciseHint Exercise hint to update - */ - update(exerciseId: number, exerciseHint: ExerciseHint): Observable; - - /** - * Deletes an exercise hint - * @param exerciseId Id of the exercise of which to delete the hint - * @param exerciseHintId Id of exercise hint to delete - */ - delete(exerciseId: number, exerciseHintId: number): Observable>; - - /** - * Finds an exercise hint - * @param exerciseId Id of the exercise of which to retrieve the hint - * @param exerciseHintId Id of exercise hint to find - */ - find(exerciseId: number, exerciseHintId: number): Observable; - - /** - * Finds all exercise hints by exercise id - * @param exerciseId Id of exercise - */ - findByExerciseId(exerciseId: number): Observable>; - - /** - * Gets all available exercise hints - * @param exerciseId Id of the exercise of which to retrieve all available exercise hints - */ - getAvailableExerciseHints(exerciseId: number): Observable>; - - /** - * Gets all activated exercise hints - * @param exerciseId Id of the exercise of which to retrieve all activated exercise hints - */ - getActivatedExerciseHints(exerciseId: number): Observable>; - - /** - * Activates an exercise hint for the current user - * @param exerciseId Id of the exercise - * @param exerciseHintId Id of the exercise hint - */ - activateExerciseHint(exerciseId: number, exerciseHintId: number): Observable; - - /** - * Activates an exercise hint for the current user - * @param exerciseId Id of the exercise - * @param exerciseHintId Id of the exercise hint - * @param ratingValue Value of the rating - */ - rateExerciseHint(exerciseId: number, exerciseHintId: number, ratingValue: number): Observable>; -} - -@Injectable({ providedIn: 'root' }) -export class ExerciseHintService implements IExerciseHintService { - public resourceUrl = 'api/programming-exercises'; - - constructor( - protected http: HttpClient, - private entityTitleService: EntityTitleService, - ) {} - - /** - * Creates an exercise hint - * @param exerciseId of the exercise - * @param exerciseHint Exercise hint to create - */ - create(exerciseId: number, exerciseHint: ExerciseHint): Observable { - exerciseHint.exercise = ExerciseService.convertExerciseDatesFromClient(exerciseHint.exercise!); - exerciseHint.type = HintType.TEXT; - if (exerciseHint.exercise.categories) { - exerciseHint.exercise.categories = ExerciseService.stringifyExerciseCategories(exerciseHint.exercise); - } - return this.http.post(`${this.resourceUrl}/${exerciseId}/exercise-hints`, exerciseHint, { observe: 'response' }); - } - - /** - * Updates an exercise hint - * @param exerciseId of the exercise - * @param exerciseHint Exercise hint to update - */ - update(exerciseId: number, exerciseHint: ExerciseHint): Observable { - exerciseHint.exercise = ExerciseService.convertExerciseDatesFromClient(exerciseHint.exercise!); - exerciseHint.exercise.categories = ExerciseService.stringifyExerciseCategories(exerciseHint.exercise); - return this.http.put(`${this.resourceUrl}/${exerciseId}/exercise-hints/${exerciseHint.id}`, exerciseHint, { observe: 'response' }); - } - - /** - * Deletes an exercise hint - * @param exerciseId Id of the exercise of which to delete the hint - * @param exerciseHintId Id of exercise hint to delete - */ - delete(exerciseId: number, exerciseHintId: number): Observable> { - return this.http.delete(`${this.resourceUrl}/${exerciseId}/exercise-hints/${exerciseHintId}`, { observe: 'response' }); - } - - /** - * Finds an exercise hint - * @param exerciseId Id of the exercise of which to retrieve the hint - * @param exerciseHintId Id of exercise hint to find - */ - find(exerciseId: number, exerciseHintId: number): Observable { - return this.http - .get(`${this.resourceUrl}/${exerciseId}/exercise-hints/${exerciseHintId}`, { observe: 'response' }) - .pipe(tap((res) => this.sendTitlesToEntityTitleService(res?.body, exerciseId))); - } - - /** - * Finds all exercise hints by exercise id - * Also fetches any relations. This currently only includes the submission entries of a code hint - * @param exerciseId Id of exercise - */ - findByExerciseId(exerciseId: number): Observable> { - return this.http - .get(`${this.resourceUrl}/${exerciseId}/exercise-hints`, { observe: 'response' }) - .pipe(tap((res) => res?.body?.forEach((hint) => this.sendTitlesToEntityTitleService(hint, exerciseId)))); - } - - /** - * Gets all available exercise hints - * @param exerciseId Id of the exercise of which to retrieve all available exercise hints - */ - getAvailableExerciseHints(exerciseId: number): Observable> { - return this.http.get(`${this.resourceUrl}/${exerciseId}/exercise-hints/available`, { observe: 'response' }); - } - - /** - * Gets all activated exercise hints - * @param exerciseId Id of the exercise of which to retrieve all activated exercise hints - */ - getActivatedExerciseHints(exerciseId: number): Observable> { - return this.http - .get(`${this.resourceUrl}/${exerciseId}/exercise-hints/activated`, { observe: 'response' }) - .pipe(tap((res) => res?.body?.forEach((hint) => this.sendTitlesToEntityTitleService(hint, exerciseId)))); - } - - /** - * Activates an exercise hint for the current user - * @param exerciseId Id of the exercise - * @param exerciseHintId Id of the exercise hint - */ - activateExerciseHint(exerciseId: number, exerciseHintId: number): Observable { - return this.http.post(`${this.resourceUrl}/${exerciseId}/exercise-hints/${exerciseHintId}/activate`, {}, { observe: 'response' }); - } - - /** - * Activates an exercise hint for the current user - * @param exerciseId Id of the exercise - * @param exerciseHintId Id of the exercise hint - * @param ratingValue Value of the rating - */ - rateExerciseHint(exerciseId: number, exerciseHintId: number, ratingValue: number): Observable> { - return this.http.post(`${this.resourceUrl}/${exerciseId}/exercise-hints/${exerciseHintId}/rating/${ratingValue}`, {}, { observe: 'response' }); - } - - private sendTitlesToEntityTitleService(hint: ExerciseHint | undefined | null, exerciseId: number) { - this.entityTitleService.setTitle(EntityType.HINT, [hint?.id, exerciseId], hint?.title); - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.html b/src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.html deleted file mode 100644 index 28a935fd18ff..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.html +++ /dev/null @@ -1,12 +0,0 @@ -
-
{{ solutionEntry.filePath }}
-
-
- -
diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.scss b/src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.scss deleted file mode 100644 index 5a51c7d082f5..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.scss +++ /dev/null @@ -1,3 +0,0 @@ -.solution-entry-editor { - border: 1px solid var(--border-color); -} diff --git a/src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.ts b/src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.ts deleted file mode 100644 index 314fce704776..000000000000 --- a/src/main/webapp/app/exercises/shared/exercise-hint/shared/solution-entry.component.ts +++ /dev/null @@ -1,47 +0,0 @@ -import { Component, EventEmitter, Input, OnInit, Output, ViewChild } from '@angular/core'; -import { faTimes } from '@fortawesome/free-solid-svg-icons'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { MonacoEditorComponent } from 'app/shared/monaco-editor/monaco-editor.component'; - -@Component({ - selector: 'jhi-solution-entry', - templateUrl: './solution-entry.component.html', - styleUrls: ['./solution-entry.component.scss'], -}) -export class SolutionEntryComponent implements OnInit { - @ViewChild('editor', { static: true }) - editor: MonacoEditorComponent; - - @Input() - solutionEntry: ProgrammingExerciseSolutionEntry; - @Input() - enableEditing: boolean; - - @Output() - onRemoveEntry: EventEmitter = new EventEmitter(); - - editorHeight = 20; - - protected readonly faTimes = faTimes; - - ngOnInit(): void { - this.setupEditor(); - } - - onEditorContentChange(value: string): void { - this.solutionEntry.code = value; - } - - onContentSizeChange(contentHeight: number): void { - this.editorHeight = contentHeight; - } - - setupEditor(): void { - const startLine = this.solutionEntry.line ?? 1; - this.editor.setStartLineNumber(startLine); - this.editor.changeModel(this.solutionEntry.filePath ?? 'file', this.solutionEntry.code ?? ''); - this.editor.layout(); - // We manually fetch the initial content height, as the editor does not provide it immediately - this.editorHeight = this.editor.getContentHeight(); - } -} diff --git a/src/main/webapp/app/exercises/shared/exercise/exercise.service.ts b/src/main/webapp/app/exercises/shared/exercise/exercise.service.ts index f3247bc26267..b377e0c20572 100644 --- a/src/main/webapp/app/exercises/shared/exercise/exercise.service.ts +++ b/src/main/webapp/app/exercises/shared/exercise/exercise.service.ts @@ -20,7 +20,6 @@ import { FileUploadExercise } from 'app/entities/file-upload-exercise.model'; import { ArtemisMarkdownService } from 'app/shared/markdown.service'; import { SafeHtml } from '@angular/platform-browser'; import { PlagiarismCaseInfo } from 'app/exercises/shared/plagiarism/types/PlagiarismCaseInfo'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; import { IrisExerciseSettings } from 'app/entities/iris/settings/iris-settings.model'; export type EntityResponseType = HttpResponse; @@ -38,8 +37,6 @@ export type ExerciseDetailsType = { exercise: Exercise; irisSettings?: IrisExerciseSettings; plagiarismCaseInfo?: PlagiarismCaseInfo; - availableExerciseHints?: ExerciseHint[]; - activatedExerciseHints?: ExerciseHint[]; }; export type CourseExistingExerciseDetailsType = { @@ -176,9 +173,6 @@ export class ExerciseService { if (res.body.exercise.posts === undefined) { res.body.exercise.posts = []; } - for (const hint of res.body.activatedExerciseHints ?? []) { - this.entityTitleService.setTitle(EntityType.HINT, [hint?.id, exerciseId], hint?.title); - } } return res; }), diff --git a/src/main/webapp/app/localvc/commit-details-view/commit-details-view.component.ts b/src/main/webapp/app/localvc/commit-details-view/commit-details-view.component.ts index 98ced261dc32..b147bd0892cd 100644 --- a/src/main/webapp/app/localvc/commit-details-view/commit-details-view.component.ts +++ b/src/main/webapp/app/localvc/commit-details-view/commit-details-view.component.ts @@ -1,5 +1,5 @@ import { Component, OnDestroy, OnInit } from '@angular/core'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from 'app/entities/programming-exercise-git-diff-report.model'; import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; import { ProgrammingExerciseParticipationService } from 'app/exercises/programming/manage/services/programming-exercise-participation.service'; import { Subscription, throwError } from 'rxjs'; diff --git a/src/main/webapp/app/overview/exercise-details/course-exercise-details.component.html b/src/main/webapp/app/overview/exercise-details/course-exercise-details.component.html index 55391f36192e..05e6e4d9d53e 100644 --- a/src/main/webapp/app/overview/exercise-details/course-exercise-details.component.html +++ b/src/main/webapp/app/overview/exercise-details/course-exercise-details.component.html @@ -190,12 +190,6 @@

@if (exercise.type === PROGRAMMING && !exercise.exerciseGroup && irisSettings?.irisChatSettings?.enabled) { } - @if (plagiarismCaseInfo?.verdict === PlagiarismVerdict.NO_PLAGIARISM) { diff --git a/src/main/webapp/app/overview/exercise-details/course-exercise-details.component.ts b/src/main/webapp/app/overview/exercise-details/course-exercise-details.component.ts index 330e469035a6..6b81fedc9ad0 100644 --- a/src/main/webapp/app/overview/exercise-details/course-exercise-details.component.ts +++ b/src/main/webapp/app/overview/exercise-details/course-exercise-details.component.ts @@ -30,11 +30,8 @@ import { Complaint } from 'app/entities/complaint.model'; import { SubmissionPolicy } from 'app/entities/submission-policy.model'; import { ArtemisMarkdownService } from 'app/shared/markdown.service'; import { IconDefinition, faAngleDown, faAngleUp, faBook, faEye, faFileSignature, faListAlt, faSignal, faTable, faWrench } from '@fortawesome/free-solid-svg-icons'; -import { ExerciseHintService } from 'app/exercises/shared/exercise-hint/shared/exercise-hint.service'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; import { PlagiarismVerdict } from 'app/exercises/shared/plagiarism/types/PlagiarismVerdict'; import { PlagiarismCaseInfo } from 'app/exercises/shared/plagiarism/types/PlagiarismCaseInfo'; -import { ResultService } from 'app/exercises/shared/result/result.service'; import { MAX_RESULT_HISTORY_LENGTH } from 'app/overview/result-history/result-history.component'; import { isCommunicationEnabled, isMessagingEnabled } from 'app/entities/course.model'; import { ExerciseCacheService } from 'app/exercises/shared/exercise/exercise-cache.service'; @@ -68,7 +65,6 @@ export class CourseExerciseDetailsComponent extends AbstractScienceComponent imp readonly MODELING = ExerciseType.MODELING; readonly TEXT = ExerciseType.TEXT; readonly FILE_UPLOAD = ExerciseType.FILE_UPLOAD; - readonly evaluateBadge = ResultService.evaluateBadge; readonly dayjs = dayjs; readonly ChatServiceMode = ChatServiceMode; @@ -99,8 +95,6 @@ export class CourseExerciseDetailsComponent extends AbstractScienceComponent imp submissionPolicy?: SubmissionPolicy; exampleSolutionCollapsed: boolean; plagiarismCaseInfo?: PlagiarismCaseInfo; - availableExerciseHints: ExerciseHint[]; - activatedExerciseHints: ExerciseHint[]; irisSettings?: IrisSettings; paramsSubscription: Subscription; profileSubscription?: Subscription; @@ -120,8 +114,6 @@ export class CourseExerciseDetailsComponent extends AbstractScienceComponent imp faWrench = faWrench; faTable = faTable; faListAlt = faListAlt; - faSignal = faSignal; - faFileSignature = faFileSignature; faAngleDown = faAngleDown; faAngleUp = faAngleUp; @@ -137,7 +129,6 @@ export class CourseExerciseDetailsComponent extends AbstractScienceComponent imp private quizExerciseService: QuizExerciseService, private complaintService: ComplaintService, private artemisMarkdown: ArtemisMarkdownService, - private exerciseHintService: ExerciseHintService, scienceService: ScienceService, ) { super(scienceService, ScienceEventType.EXERCISE__OPEN); @@ -209,8 +200,6 @@ export class CourseExerciseDetailsComponent extends AbstractScienceComponent imp this.irisSettings = newExerciseDetails.irisSettings; } }); - this.availableExerciseHints = newExerciseDetails.availableExerciseHints || []; - this.activatedExerciseHints = newExerciseDetails.activatedExerciseHints || []; } this.showIfExampleSolutionPresent(newExerciseDetails.exercise); @@ -331,25 +320,6 @@ export class CourseExerciseDetailsComponent extends AbstractScienceComponent imp } this.updateStudentParticipations(); this.mergeResultsAndSubmissionsForParticipations(); - - if (ExerciseType.PROGRAMMING === this.exercise?.type) { - this.exerciseHintService.getActivatedExerciseHints(this.exerciseId).subscribe((activatedRes?: HttpResponse) => { - this.activatedExerciseHints = activatedRes!.body!; - - this.exerciseHintService.getAvailableExerciseHints(this.exerciseId).subscribe((availableRes?: HttpResponse) => { - // filter out the activated hints from the available hints - this.availableExerciseHints = availableRes!.body!.filter( - (availableHint) => !this.activatedExerciseHints.some((activatedHint) => availableHint.id === activatedHint.id), - ); - const filteredAvailableExerciseHints = this.availableExerciseHints.filter((hint) => hint.displayThreshold !== 0); - if (filteredAvailableExerciseHints.length) { - this.alertService.info('artemisApp.exerciseHint.availableHintsAlertMessage', { - taskName: filteredAvailableExerciseHints.first()?.programmingExerciseTask?.taskName, - }); - } - }); - }); - } } }); } @@ -429,11 +399,6 @@ export class CourseExerciseDetailsComponent extends AbstractScienceComponent imp this.alertService.error(error); } - onHintActivated(exerciseHint: ExerciseHint) { - this.availableExerciseHints = this.availableExerciseHints.filter((hint) => hint.id !== exerciseHint.id); - this.activatedExerciseHints.push(exerciseHint); - } - /** * Used to change the boolean value for the example solution dropdown menu */ diff --git a/src/main/webapp/app/overview/exercise-details/course-exercise-details.module.ts b/src/main/webapp/app/overview/exercise-details/course-exercise-details.module.ts index 6c5664b7bff2..ec6495463ecc 100644 --- a/src/main/webapp/app/overview/exercise-details/course-exercise-details.module.ts +++ b/src/main/webapp/app/overview/exercise-details/course-exercise-details.module.ts @@ -25,7 +25,6 @@ import { SubmissionResultStatusModule } from 'app/overview/submission-result-sta import { LtiInitializerComponent } from 'app/overview/exercise-details/lti-initializer.component'; import { LtiInitializerModalComponent } from 'app/overview/exercise-details/lti-initializer-modal.component'; import { ArtemisProgrammingExerciseManagementModule } from 'app/exercises/programming/manage/programming-exercise-management.module'; -import { ArtemisExerciseHintParticipationModule } from 'app/exercises/shared/exercise-hint/participate/exercise-hint-participation.module'; import { ProblemStatementComponent } from 'app/overview/exercise-details/problem-statement/problem-statement.component'; import { ArtemisFeedbackModule } from 'app/exercises/shared/feedback/feedback.module'; import { ArtemisExerciseInfoModule } from 'app/exercises/shared/exercise-info/exercise-info.module'; @@ -69,7 +68,6 @@ const standaloneComponents = [ExerciseHeadersInformationComponent]; ArtemisMarkdownModule, SubmissionResultStatusModule, ArtemisProgrammingExerciseManagementModule, - ArtemisExerciseHintParticipationModule, ArtemisFeedbackModule, ArtemisExerciseInfoModule, IrisModule, diff --git a/src/main/webapp/app/shared/http/file.service.ts b/src/main/webapp/app/shared/http/file.service.ts index 1c9f264f41d3..f739c44014ed 100644 --- a/src/main/webapp/app/shared/http/file.service.ts +++ b/src/main/webapp/app/shared/http/file.service.ts @@ -37,32 +37,11 @@ export class FileService { return Promise.resolve(file); } - /** - * Fetches the aeolus template file for the given programming language - * @param {ProgrammingLanguage} language - * @param {ProjectType} projectType (if available) - * @param staticAnalysis (if available) whether static code analysis should be enabled - * @param sequentialRuns (if available) whether sequential test runs should be enabled - * @param coverage (if available) whether test coverage should be enabled - * @returns json test file - */ - getAeolusTemplateFile(language: ProgrammingLanguage, projectType?: ProjectType, staticAnalysis?: boolean, sequentialRuns?: boolean, coverage?: boolean): Observable { - const urlParts: string[] = [language]; - const params: string[] = []; - if (projectType) { - urlParts.push(projectType); - } - params.push('staticAnalysis=' + (staticAnalysis == undefined ? false : staticAnalysis)); - params.push('sequentialRuns=' + (sequentialRuns == undefined ? false : sequentialRuns)); - params.push('testCoverage=' + (coverage == undefined ? false : coverage)); - return this.http.get(`${this.resourceUrl}/aeolus/templates/` + urlParts.join('/') + '?' + params.join('&'), { responseType: 'text' as 'json' }); - } - /** * Fetches the template code of conduct * @returns markdown file */ - getTemplateCodeOfCondcut(): Observable> { + getTemplateCodeOfConduct(): Observable> { return this.http.get(`api/files/templates/code-of-conduct`, { observe: 'response', responseType: 'text' as 'json' }); } diff --git a/src/main/webapp/app/shared/layouts/navbar/entity-title.service.ts b/src/main/webapp/app/shared/layouts/navbar/entity-title.service.ts index a50aab27ddc4..3ab54964001f 100644 --- a/src/main/webapp/app/shared/layouts/navbar/entity-title.service.ts +++ b/src/main/webapp/app/shared/layouts/navbar/entity-title.service.ts @@ -9,7 +9,6 @@ export enum EntityType { EXERCISE = 'EXERCISE', LECTURE = 'LECTURE', COMPETENCY = 'COMPETENCY', - HINT = 'HINT', DIAGRAM = 'DIAGRAM', ORGANIZATION = 'ORGANIZATION', EXAM = 'EXAM', @@ -120,9 +119,6 @@ export class EntityTitleService { case EntityType.COMPETENCY: resourceUrl += 'competencies'; break; - case EntityType.HINT: - resourceUrl += `programming-exercises/${ids[1]}/exercise-hints`; - break; case EntityType.DIAGRAM: resourceUrl += 'apollon-diagrams'; break; diff --git a/src/main/webapp/app/shared/layouts/navbar/navbar.component.ts b/src/main/webapp/app/shared/layouts/navbar/navbar.component.ts index cdf3bbb137ae..77a83dc0990d 100644 --- a/src/main/webapp/app/shared/layouts/navbar/navbar.component.ts +++ b/src/main/webapp/app/shared/layouts/navbar/navbar.component.ts @@ -342,7 +342,6 @@ export class NavbarComponent implements OnInit, OnDestroy { detailed: 'artemisApp.gradingSystem.detailedTab.title', interval: 'artemisApp.gradingSystem.intervalTab.title', plagiarism_cases: 'artemisApp.plagiarism.cases.pageTitle', - code_hint_management: 'artemisApp.codeHint.management.title', tutorial_groups_management: 'artemisApp.pages.tutorialGroupsManagement.title', tutorial_groups: 'artemisApp.breadcrumb.title', registered_students: 'artemisApp.pages.registeredStudents.title', @@ -492,13 +491,6 @@ export class NavbarComponent implements OnInit, OnDestroy { case 'assessment-dashboard': this.addResolvedTitleAsCrumb(EntityType.EXERCISE, [Number(segment)], currentPath, segment); break; - case 'exercise-hints': - // obtain the exerciseId of the current path - // current path of form '/course-management/:courseId/exercises/:exerciseId/... - - const exerciseId = currentPath.split('/')[4]; - this.addResolvedTitleAsCrumb(EntityType.HINT, [Number(segment), Number(exerciseId)], currentPath, segment); - break; case 'apollon-diagrams': this.addResolvedTitleAsCrumb(EntityType.DIAGRAM, [Number(segment)], currentPath, segment); break; diff --git a/src/main/webapp/app/shared/markdown-editor/monaco/markdown-editor-monaco.component.ts b/src/main/webapp/app/shared/markdown-editor/monaco/markdown-editor-monaco.component.ts index 05acc805c47a..4be6605a93b9 100644 --- a/src/main/webapp/app/shared/markdown-editor/monaco/markdown-editor-monaco.component.ts +++ b/src/main/webapp/app/shared/markdown-editor/monaco/markdown-editor-monaco.component.ts @@ -278,7 +278,9 @@ export class MarkdownEditorMonacoComponent implements AfterContentInit, AfterVie color: this.filterDisplayedAction(this.colorAction), domain: { withoutOptions: this.filterDisplayedActions(this.domainActions.filter((action) => !(action instanceof TextEditorDomainActionWithOptions))), - withOptions: this.filterDisplayedActions(this.domainActions.filter((action) => action instanceof TextEditorDomainActionWithOptions)), + withOptions: this.filterDisplayedActions( + this.domainActions.filter((action) => action instanceof TextEditorDomainActionWithOptions), + ) as TextEditorDomainActionWithOptions[], }, lecture: this.filterDisplayedAction(this.lectureReferenceAction), meta: this.filterDisplayedActions(this.metaActions), @@ -521,7 +523,7 @@ export class MarkdownEditorMonacoComponent implements AfterContentInit, AfterVie private processFileUploadResponse(response: FileUploadResponse, file: File): void { const extension = file.name.split('.').last()?.toLocaleLowerCase(); - const attachmentAction: AttachmentAction | undefined = this.defaultActions.find((action) => action instanceof AttachmentAction); + const attachmentAction: AttachmentAction | undefined = this.defaultActions.find((action) => action instanceof AttachmentAction) as AttachmentAction; const urlAction: UrlAction | undefined = this.defaultActions.find((action) => action instanceof UrlAction); if (!attachmentAction || !urlAction || !response.path) { throw new Error('Cannot process file upload.'); diff --git a/src/main/webapp/content/scss/themes/_dark-variables.scss b/src/main/webapp/content/scss/themes/_dark-variables.scss index 19de3c6980ef..97c749b23ac0 100644 --- a/src/main/webapp/content/scss/themes/_dark-variables.scss +++ b/src/main/webapp/content/scss/themes/_dark-variables.scss @@ -452,7 +452,6 @@ $monaco-editor-build-annotation-outdated-background: scale-color($gray-500, $alp $monaco-editor-build-annotation-outdated-glyph: $gray-500; $monaco-editor-diff-highlight-green: $success; $monaco-editor-diff-line-highlight: scale-color($monaco-editor-diff-highlight-green, $alpha: -80%, $lightness: -20%); -$monaco-editor-test-coverage-highlight: scale-color($warning, $alpha: -80%); $monaco-editor-add-feedback-button-background: $gray-600; $monaco-editor-add-feedback-button-hover-background: lighten($gray-600, 10%); $monaco-editor-add-feedback-button-text: $gray-300; diff --git a/src/main/webapp/content/scss/themes/_default-variables.scss b/src/main/webapp/content/scss/themes/_default-variables.scss index f8bbfad5151a..d7aab8d31279 100644 --- a/src/main/webapp/content/scss/themes/_default-variables.scss +++ b/src/main/webapp/content/scss/themes/_default-variables.scss @@ -378,7 +378,6 @@ $monaco-editor-build-annotation-outdated-background: scale-color($gray-500, $alp $monaco-editor-build-annotation-outdated-glyph: $gray-500; $monaco-editor-diff-highlight-green: rgb(63, 185, 80); $monaco-editor-diff-line-highlight: scale-color($monaco-editor-diff-highlight-green, $alpha: -60%); -$monaco-editor-test-coverage-highlight: scale-color($warning, $alpha: -80%); $monaco-editor-add-feedback-button-background: $gray-400; $monaco-editor-add-feedback-button-hover-background: $gray-500; $monaco-editor-add-feedback-button-text: $black; diff --git a/src/main/webapp/i18n/de/codeHint.json b/src/main/webapp/i18n/de/codeHint.json deleted file mode 100644 index 93f2b508fea0..000000000000 --- a/src/main/webapp/i18n/de/codeHint.json +++ /dev/null @@ -1,105 +0,0 @@ -{ - "artemisApp": { - "codeHint": { - "type": "Code-Hinweis", - "entry": "Eintrag", - "entryType": "Typ (V/S)", - "lineOfCode": "1 Code-Zeile", - "linesOfCode": "{{ lines }} Code-Zeilen", - "entryTypeTooltip": "Definiert, ob dieser Eintrag für einen strukturellen (S) oder verhaltensweisen (V) Test erstellt wurde.", - "entryAbbreviation": { - "structural": "S", - "behavioral": "V" - }, - "entries": "Fragmente", - "removeEntryQuestion": "Soll dieses Fragment wirklich entfernt werden?", - "noEntries": "Dieser Code-Hinweis hat keine Fragmente.", - "deleted": "Exercise Hint gelöscht mit ID {{ param }}", - "management": { - "title": "Code-Hinweise verwalten", - "previousButton": { - "label": "Zurück" - }, - "nextButton": { - "label": "Weiter", - "tooltip": "Der aktuelle Schritt muss erfolgreich ausgeführt seien, damit fortgefahren werden kann." - }, - "step1": { - "name": "Git-Diff Report", - "description": "Der Git-Diff Report wird automatisch für das Template- und Lösungs-Repository erstellt.", - "notGenerated": "Git-Diff Report wurde noch nicht erzeugt." - }, - "step2": { - "name": "Code-Abdeckung", - "description": "Der Code-Abdeckungs-Report wird automatisch für jede Einreichung in Lösungs-Repository erstellt.", - "notGenerated": "Der Code-Abdeckungs-Report wurde noch nicht erzeugt." - }, - "step3": { - "name": "Lösungs-Fragmente", - "description": "In diesem Schritt kannst du die einzelnen Fragmente bearbeiten, aus denen Code-Hinweise erstellt werden.", - "notGenerated": "Bisher wurden keine Fragmente generiert.", - "createManualFragmentButton": { - "label": "Erstelle manuelles Fragment", - "tooltip": "Fragment manuell mit zugehörigem Test und Datei erstellen." - }, - "structuralEntriesButton": { - "label": "Strukturelle Fragmente generieren", - "tooltip": "Erstellt Fragmente für strukturelle Tests. Bereits existierende strukturelle Fragmente werden überschrieben." - }, - "behavioralEntriesButton": { - "label": "Verhaltensweise Fragmente generieren", - "tooltip": "Erstellt Fragmente für verhaltensweise Tests. Bereits existierende verhaltensweise Fragmente werden überschrieben." - }, - "deleteAllEntriesButton": { - "label": "Alle Fragmente löschen", - "tooltip": "Löscht alle Fragmente für diese Aufgabe.", - "question": "Bist du dir sicher, dass du alle Fragmente für diese Aufgabe löschen möchtest?", - "title": "Alle Fragmente löschen", - "success": "Alle Fragmente wurden erfolgreich gelöscht" - }, - "deleteIndividualEntryButton": { - "question": "Bist du dir sicher, dass du das Fragment mit ID {{ title }} löschen möchtest?" - } - }, - "step4": { - "name": "Code-Hinweise", - "description": "In diesem Schritt kannst du die Code-Hinweise generieren.", - "title": "Titel", - "task": "Aufgabe", - "notGenerated": "Bisher wurden keine Code-Hinweise generiert.", - "updateHintsButton": { - "label": "Code-Hinweise aktualisieren", - "tooltip": "Erstellt neue Code-Hinweise, indem die existierenden Lösungs-Codeschnipsel neu zugewiesen werden. Titel, Beschreibung und Inhalt bleiben unverändert beim ursprünglichen Hinweis.", - "success": "Code-Hinweise erfolgreich aktualisiert. Die Hinweise sind direkt sichtbar für teilnehmende Studierende." - }, - "createHintsButton": { - "label": "Code-Hinweise erstellen", - "tooltip": "Erstellt Code-Hinweise aus den existierenden Lösungs-Codeschnipseln.", - "success": "Code-Hinweise erfolgreich erstellt. Die Hinweise sind direkt sichtbar für teilnehmende Studierende." - }, - "recreateHintsButton": { - "label": "Code-Hinweise neu erstellen", - "tooltip": "Erstellt Code-Hinweise aus den existierenden Fragmenten. Existierende Hinweise werden überschrieben.", - "success": "Code-Hinweise erfolgreich neu erstellt. Die Hinweise sind direkt sichtbar für teilnehmende Studierende." - }, - "iris": { - "generateDescription": { - "label": "Beschreibungen generieren", - "tooltip": "Generiert eine kurze und lange Beschreibung basierend auf dem Lösungs-Codeschnipseln mithilfe von Iris." - } - } - } - } - }, - "programmingExerciseSolutionEntry": { - "simplifiedName": "Lösungs-Codeschnipsel", - "created": "Neues Lösungs-Codeschnipsel erstellt mit ID {{ param }}", - "updated": "Lösungs-Codeschnipsel aktualisiert mit ID {{ param }}", - "deleted": "Lösungs-Codeschnipsel gelöscht mit ID {{ param }}", - "test": "Test", - "testOnlyForCurrentTask": "Du kannst nur Tests auswählen, die in der aktuellen Task enthalten sind.", - "file": "Datei", - "changeContentNote": "Hinweis: Änderungen an diesem Fragment werden für Hinweise mit diesem Fragment automatisch übernommen." - } - } -} diff --git a/src/main/webapp/i18n/de/error.json b/src/main/webapp/i18n/de/error.json index c60e87ec42a3..7306229a03fb 100644 --- a/src/main/webapp/i18n/de/error.json +++ b/src/main/webapp/i18n/de/error.json @@ -83,7 +83,6 @@ }, "idNull": "Ungültige ID", "unexpectedError": "Ein unerwarteter Fehler ist aufgetreten: {{error}}", - "manualCodeHintOperation": "Dieser Endpunkt erlaubt keine manuelle Bearbeitung von Codehinweisen.", "exerciseNotDefined": "Die Aufgabe ist nicht definiert.", "exerciseIdMismatch": "Die Aufgaben-IDs stimmen nicht überein.", "solutionEntryError": "Fehler beim Laden der extrahierten Aufgaben aus der Aufgabenstellung.", diff --git a/src/main/webapp/i18n/de/iris.json b/src/main/webapp/i18n/de/iris.json index 95fa36c63909..2c01b1e3679d 100644 --- a/src/main/webapp/i18n/de/iris.json +++ b/src/main/webapp/i18n/de/iris.json @@ -31,7 +31,6 @@ "title": "Vorlesungen Erfassung Einstellungen", "autoIngestOnAttachmentUpload": "Vorlesungen automatisch an Pyris senden" }, - "hestiaSettings": "Hestia Einstellungen", "competencyGenerationSettings": "Kompetenzgenerierung Einstellungen", "proactivityBuildFailedEventEnabled": { "label": "Build-Fehler überwachen", @@ -70,8 +69,7 @@ "enabled": { "on": "Aktiviert", "off": "Deaktiviert", - "chat": "Iris Chat", - "hestia": "Hestia Integration" + "chat": "Iris Chat" }, "disabled": { "course": "IRIS ist in den Kurs-IRIS-Einstellungen deaktiviert", diff --git a/src/main/webapp/i18n/de/programmingExercise.json b/src/main/webapp/i18n/de/programmingExercise.json index 2fb203e24900..dbb1c466df22 100644 --- a/src/main/webapp/i18n/de/programmingExercise.json +++ b/src/main/webapp/i18n/de/programmingExercise.json @@ -218,21 +218,6 @@ "combineTemplateCommitsError": "Template-Repository-Commits konnten nicht zusammengeführt werden.", "diffReportError": "Der Diff Report konnte nicht erstellt werden.", "combineTemplateCommitsSuccess": "Template-Repository-Commits wurden zusammengeführt.", - "extractTasksFromProblemStatementWarning": "Nur zu Testzwecken: Extrahiere alle Aufgaben und Tests aus der Aufgabenstellung.", - "extractTasksFromProblemStatementSuccess": "{{numberTasks}} extrahierte Aufgaben mit {{numberTestCases}} Tests geladen:\n{{detailedResult}}", - "extractTasksFromProblemStatementTitle": "Lade extrahierte Aufgaben", - "deleteTasksAndSolutionEntriesSuccess": "Alle Aufgaben und Lösungs-Codeschnipsel wurden gelöscht.", - "deleteTasksAndSolutionEntriesWarning": "Nur zu Testzwecken: Lösche alle Aufgaben und Lösungs-Codeschnipsel.", - "deleteTasksAndSolutionEntriesTitle": "Lösche Aufgaben", - "generateCodeHintsTooltip": "Generiert Code Hinweise für alle Tasks dieser Aufgabe unter Verwendung der bereits erstellten Lösungs-Codeschnipsel", - "generateCodeHintsTitle": "Generiere Code Hinweise", - "generateCodeHintsSuccess": "Code Hinweise wurden erfolgreich generiert", - "createStructuralSolutionEntriesTooltip": "Erstellt Lösungs-Codeschnipsel für alle strukturellen Tests dieser Aufgabe.", - "createStructuralSolutionEntriesTitle": "Erstelle strukturelle Lösungs-Codeschnipsel", - "createStructuralSolutionEntriesSuccess": "Strukturelle Lösungs-Codeschnipsel wurden erfolgreich erstellt", - "createBehavioralSolutionEntriesTooltip": "Erstellt Lösungs-Codeschnipsel für alle Verhaltenstests dieser Aufgabe.", - "createBehavioralSolutionEntriesTitle": "Erstelle verhaltensweise Lösungs-Codeschnipsel", - "createBehavioralSolutionEntriesSuccess": "Verhaltensweise Lösungs-Codeschnipsel wurden erfolgreich erstellt", "editable": { "unsaved": "Ungespeichert.", "unsavedTooltip": "Es gibt ungespeicherte Änderungen in der Aufgabenstellung.", @@ -706,20 +691,9 @@ }, "withDependencies": "Mit beispielhafter Abhängigkeit", "withDependenciesTooltip": "Fügt dem generierten Projekt eine externe Apache commons-lang-Abhängigkeit als Beispiel dafür hinzu, wie Maven-Abhängigkeiten mit Artemis-Aufgaben verwendet werden sollten.", - "recordTestwiseCoverage": "Testbasierte Code-Abdeckung aufzeichnen", - "recordTestwiseCoverageTooltip": "Durch Aktivierung dieser Option wird die testbasierte Code-Abdeckung für das Lösungs-Repository aufgezeichnet. Diese Option ist nur für Java/Kotlin-Aufgaben mit nicht-sequentiellen Testläufen verfügbar.", "customizeBuildPlansWithAeolusTooltip": "Aktiviere diese Option, um das Docker Image und das Build Skript anzupassen, das auf Artemis für jede Einreichung in einem Docker Container ausgeführt wird.", "customizeBuildPlansTooltip": "Aktiviere diese Option, um das Docker Image und das Build Skript anzupassen, das auf Artemis für jede Einreichung in einem Docker Container ausgeführt wird.", "coveredLineRatio": "Anteil Test-Abgedeckter Zeilen", - "testwiseCoverageReport": { - "button": "Testbasierte Code-Abdeckung anzeigen", - "tooltip": "Zeigt die testbasierte Code-Abdeckung für das Lösungs-Repository.", - "title": "Testbasierte Code-Abdeckung", - "404": "Testbasierte Code-Abdeckung wurde noch nicht generiert. Bitte pushe etwas in die Lösungs- oder Test-Repositories um den Report zu erstellen.", - "selectTests": "Wähle die Tests aus, für die die Test-abgedeckten Zeilen markiert werden:", - "filePath": "Datei-Pfad", - "coveredLines": "Abgedeckte Zeilen" - }, "buildLogStatistics": { "title": "Durchschnittliche Statistiken der Build-Logs", "numberOfBuilds": "# Builds", diff --git a/src/main/webapp/i18n/en/codeHint.json b/src/main/webapp/i18n/en/codeHint.json deleted file mode 100644 index e0c014bbc557..000000000000 --- a/src/main/webapp/i18n/en/codeHint.json +++ /dev/null @@ -1,105 +0,0 @@ -{ - "artemisApp": { - "codeHint": { - "type": "Code Hint", - "entry": "Entry", - "entryType": "Type (S/B)", - "lineOfCode": "1 line of code", - "linesOfCode": "{{ lines }} lines of code", - "entryTypeTooltip": "Defines whether this entry has been generated for a structural (S) or behavioral (B) test case.", - "entryAbbreviation": { - "structural": "S", - "behavioral": "B" - }, - "entries": "Entries", - "removeEntryQuestion": "Are you sure you want to remove this entry?", - "noEntries": "This code hint has no Solution Code Snippets.", - "deleted": "Deleted Code Hint with identifier {{ param }}", - "management": { - "title": "Code Hint Management", - "previousButton": { - "label": "Back" - }, - "nextButton": { - "label": "Next", - "tooltip": "The current step has to be performed to proceed." - }, - "step1": { - "name": "Git-Diff Report", - "description": "The git-diff report is created automatically for the template and solution repositories for every submission to either of the repositories.", - "notGenerated": "The Git-Diff report has not been generated yet." - }, - "step2": { - "name": "Coverage Report", - "description": "The coverage report is created automatically for every submission to the solution repository.", - "notGenerated": "The coverage report has not been generated yet." - }, - "step3": { - "name": "Solution Code Snippets", - "description": "This steps allows editing the individual Solution Code Snippets from which code hints are generated.", - "notGenerated": "No Solution Code Snippets have been generated yet.", - "createManualFragmentButton": { - "label": "Create manual fragment", - "tooltip": "Allows to create a manual Solution Code Snippets for a file and test case." - }, - "structuralEntriesButton": { - "label": "Generate structural snippets", - "tooltip": "Generates fragments for structural test cases. Already existing structural fragments will be overwritten." - }, - "behavioralEntriesButton": { - "label": "Generate behavioral snippets", - "tooltip": "Generates fragments for behavioral test cases. Already existing behavioral fragments will be overwritten." - }, - "deleteAllEntriesButton": { - "label": "Delete all snippets", - "tooltip": "Deletes all Solution Code Snippets for this exercise.", - "question": "Are you sure to delete all Solution Code Snippets for this exercise?", - "title": "Delete all Solution Code Snippets", - "success": "Successfully deleted all Solution Code Snippets" - }, - "deleteIndividualEntryButton": { - "question": "Are you sure to delete the Solution Code Snippets with ID {{ title }}?" - } - }, - "step4": { - "name": "Code Hints", - "description": "This step allows generating the code hints.", - "title": "Title", - "task": "Task", - "notGenerated": "No code hints have been generated yet.", - "updateHintsButton": { - "label": "Update code hints", - "tooltip": "Create new code hints by re-assigning the existing Solution Code Snippets. The title, description, and content of the existing hint retain.", - "success": "Code hints have been updated successfully. The hints will be visible immediately to participating students." - }, - "createHintsButton": { - "label": "Create code hints", - "tooltip": "Creates code hints from the existing Solution Code Snippets.", - "success": "Code hints have been created successfully. The hints will be visible immediately to participating students." - }, - "recreateHintsButton": { - "label": "Recreate code hints", - "tooltip": "Recreate code hints from the existing Solution Code Snippets. Existing hints will be overwritten.", - "success": "Code hints have been recreated successfully. The hints will be visible immediately to participating students." - }, - "iris": { - "generateDescription": { - "label": "Generate description & content", - "tooltip": "Generates a short and long description and for the code hint based on the Solution Code Snippets using Iris." - } - } - } - } - }, - "programmingExerciseSolutionEntry": { - "simplifiedName": "Solution Code Snippets", - "created": "Created Solution Code Snippets with ID {{ param }}", - "updated": "Updated Solution Code Snippets with ID {{ param }}", - "deleted": "Deleted Solution Code Snippets with ID {{ param }}", - "test": "Test", - "testOnlyForCurrentTask": "You can only select a test case that the currently selected task contains.", - "file": "File", - "changeContentNote": "Note: Changes apply to the fragments in existing code hints that contain this fragment." - } - } -} diff --git a/src/main/webapp/i18n/en/error.json b/src/main/webapp/i18n/en/error.json index 900d1332c417..67386b7c53b4 100644 --- a/src/main/webapp/i18n/en/error.json +++ b/src/main/webapp/i18n/en/error.json @@ -83,7 +83,6 @@ }, "idNull": "Invalid ID", "unexpectedError": "An unexpected error occurred: {{error}}", - "manualCodeHintOperation": "This endpoint does not allow changing code hints manually.", "exerciseNotDefined": "The exercise is not defined.", "exerciseIdMismatch": "The exercise IDs do not match.", "solutionEntryError": "An error occurred while retrieving the extracted tasks from the problem statement.", diff --git a/src/main/webapp/i18n/en/iris.json b/src/main/webapp/i18n/en/iris.json index c36553db40fa..fd12226bb148 100644 --- a/src/main/webapp/i18n/en/iris.json +++ b/src/main/webapp/i18n/en/iris.json @@ -31,7 +31,6 @@ "title": "Lecture Ingestion Settings", "autoIngestOnAttachmentUpload": "Send Lectures To Pyris Automatically" }, - "hestiaSettings": "Hestia Settings", "competencyGenerationSettings": "Competency Generation Settings", "proactivityBuildFailedEventEnabled": { "label": "Monitor submission build failures", @@ -70,8 +69,7 @@ "enabled": { "on": "Enabled", "off": "Disabled", - "chat": "Iris Chat", - "hestia": "Hestia Integration" + "chat": "Iris Chat" }, "disabled": { "course": "IRIS is disabled in course IRIS settings", diff --git a/src/main/webapp/i18n/en/programmingExercise.json b/src/main/webapp/i18n/en/programmingExercise.json index e1340d958a4c..4982bf654361 100644 --- a/src/main/webapp/i18n/en/programmingExercise.json +++ b/src/main/webapp/i18n/en/programmingExercise.json @@ -218,21 +218,6 @@ "combineTemplateCommitsError": "Template repository commits could not be combined.", "diffReportError": "Diff report could not be generated.", "combineTemplateCommitsSuccess": "Template repository commits have been successfully combined.", - "extractTasksFromProblemStatementWarning": "Only for testing purpose: Extract all tasks and test cases from problems statement.", - "extractTasksFromProblemStatementSuccess": "Loaded {{numberTasks}} extracted tasks with {{numberTestCases}} tests from the problem statement:\n{{detailedResult}}", - "extractTasksFromProblemStatementTitle": "Get Extracted Tasks", - "deleteTasksAndSolutionEntriesWarning": "Only for testing purpose: Delete all tasks and Solution Code Snippets.", - "deleteTasksAndSolutionEntriesSuccess": "Successfully deleted all tasks and Solution Code Snippets.", - "deleteTasksAndSolutionEntriesTitle": "Delete Tasks", - "generateCodeHintsTooltip": "Generates code hints for all tasks in this exercise using the already created Solution Code Snippets.", - "generateCodeHintsTitle": "Generate code hints", - "generateCodeHintsSuccess": "Code hints have been generated", - "createStructuralSolutionEntriesTooltip": "Creates Solution Code Snippets for all structural test cases of this exercise.", - "createStructuralSolutionEntriesTitle": "Create structural Solution Code Snippets", - "createStructuralSolutionEntriesSuccess": "Structural Solution Code Snippets have been created successfully", - "createBehavioralSolutionEntriesTooltip": "Creates Solution Code Snippets for all behavioral test cases of this exercise.", - "createBehavioralSolutionEntriesTitle": "Create behavioral Solution Code Snippets", - "createBehavioralSolutionEntriesSuccess": "Behavioral Solution Code Snippets have been created successfully", "editable": { "unsaved": "Unsaved.", "unsavedTooltip": "There are unsaved changes in the problem statement.", @@ -705,20 +690,9 @@ }, "withDependencies": "With exemplary dependency", "withDependenciesTooltip": "Adds an external Apache commons-lang dependency to the generated project as an example how Maven dependencies should be used with Artemis exercises.", - "recordTestwiseCoverage": "Record Testwise Coverage", - "recordTestwiseCoverageTooltip": "Activate this option to record the testwise coverage for the solution repository. This option is only available for Java/Kotlin-exercises with non-sequential test runs.", "customizeBuildPlansWithAeolusTooltip": "Activate this option to customize the docker image and build plan which is executed on Artemis for each submission in a Docker container.", "customizeBuildPlansTooltip": "Activate this option to customize the docker image and build plan which is executed on Artemis for each submission in a Docker container.", "coveredLineRatio": "Ratio Test-Covered Lines", - "testwiseCoverageReport": { - "button": "Show testwise coverage", - "tooltip": "Shows the testwise coverage for the solution repository.", - "title": "Testwise Coverage", - "404": "Testwise Coverage has not been generated yet. Please do a push to the solution or test repository to generate the report.", - "selectTests": "Select the tests for which the covered lines are marked:", - "filePath": "File Path", - "coveredLines": "Covered Lines" - }, "buildLogStatistics": { "title": "Average build log statistics", "numberOfBuilds": "# Builds", diff --git a/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildAgentDockerServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildAgentDockerServiceTest.java index c0c289c01bb8..cdc2af3ec1b1 100644 --- a/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildAgentDockerServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildAgentDockerServiceTest.java @@ -92,8 +92,7 @@ void testPullDockerImage() { InspectImageCmd inspectImageCmd = mock(InspectImageCmd.class); doReturn(inspectImageCmd).when(dockerClient).inspectImageCmd(anyString()); doThrow(new NotFoundException("")).when(inspectImageCmd).exec(); - BuildConfig buildConfig = new BuildConfig("echo 'test'", "test-image-name", "test", "test", "test", "test", null, null, false, false, false, null, 0, null, null, null, - null); + BuildConfig buildConfig = new BuildConfig("echo 'test'", "test-image-name", "test", "test", "test", "test", null, null, false, false, null, 0, null, null, null, null); BuildAgentDTO buildAgent = new BuildAgentDTO("buildagent1", "address1", "buildagent1"); var build = new BuildJobQueueItem("1", "job1", buildAgent, 1, 1, 1, 1, 1, BuildStatus.SUCCESSFUL, null, null, buildConfig, null); // Pull image diff --git a/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildResultTest.java b/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildResultTest.java index b7ad99348575..32c64c25d0fe 100644 --- a/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildResultTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildResultTest.java @@ -11,9 +11,7 @@ class BuildResultTest extends AbstractSpringIntegrationLocalCILocalVCTest { @Test void testUnsupportedMethods() { - BuildResult buildResult = new BuildResult(null, null, null, true, null, null, null); - + BuildResult buildResult = new BuildResult(null, null, null, true, null, null, null, null, false); assertThat(buildResult.extractBuildLogs()).isEmpty(); - assertThat(buildResult.getTestwiseCoverageReports()).isEmpty(); } } diff --git a/src/test/java/de/tum/cit/aet/artemis/buildagent/service/TestResultXmlParserTest.java b/src/test/java/de/tum/cit/aet/artemis/buildagent/service/TestResultXmlParserTest.java index 0a8931701c04..458dfc011191 100644 --- a/src/test/java/de/tum/cit/aet/artemis/buildagent/service/TestResultXmlParserTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/buildagent/service/TestResultXmlParserTest.java @@ -8,13 +8,13 @@ import org.junit.jupiter.api.Test; -import de.tum.cit.aet.artemis.buildagent.dto.BuildResult; +import de.tum.cit.aet.artemis.buildagent.dto.LocalCITestJobDTO; class TestResultXmlParserTest { - private final List failedTests = new ArrayList<>(); + private final List failedTests = new ArrayList<>(); - private final List successfulTests = new ArrayList<>(); + private final List successfulTests = new ArrayList<>(); @Test void testParseResultXmlInnerText() throws IOException { @@ -31,8 +31,8 @@ void testParseResultXmlInnerText() throws IOException { TestResultXmlParser.processTestResultFile(exampleXml, failedTests, successfulTests); assertThat(failedTests).hasSize(1); var test = failedTests.getFirst(); - assertThat(test.getName()).isEqualTo("testBubbleSort()"); - assertThat(test.getTestMessages()).containsExactly(""" + assertThat(test.name()).isEqualTo("testBubbleSort()"); + assertThat(test.testMessages()).containsExactly(""" test `add` failed on ≥ 1 cases: (0, 0) Your submission raised an error Failure("TODO add")"""); @@ -52,8 +52,8 @@ void testParseResultXmlMessageAttribute() throws IOException { TestResultXmlParser.processTestResultFile(exampleXml, failedTests, successfulTests); assertThat(failedTests).hasSize(1); var test = failedTests.getFirst(); - assertThat(test.getName()).isEqualTo("testBubbleSort()"); - assertThat(test.getTestMessages()).containsExactly("test `add` failed"); + assertThat(test.name()).isEqualTo("testBubbleSort()"); + assertThat(test.testMessages()).containsExactly("test `add` failed"); } @Test @@ -69,8 +69,8 @@ void testParseResultXmlCData() throws IOException { TestResultXmlParser.processTestResultFile(exampleXml, failedTests, successfulTests); assertThat(failedTests).hasSize(1); var test = failedTests.getFirst(); - assertThat(test.getName()).isEqualTo("testMergeSort()"); - assertThat(failedTests.getFirst().getTestMessages()).containsExactly("org.opentest4j.AssertionFailedError: Deine Einreichung enthält keine Ausgabe. (67cac2)"); + assertThat(test.name()).isEqualTo("testMergeSort()"); + assertThat(failedTests.getFirst().testMessages()).containsExactly("org.opentest4j.AssertionFailedError: Deine Einreichung enthält keine Ausgabe. (67cac2)"); } @Test @@ -88,7 +88,7 @@ void testSuccessfulTests() throws IOException { TestResultXmlParser.processTestResultFile(exampleXml, failedTests, successfulTests); assertThat(failedTests).isEmpty(); assertThat(successfulTests).hasSize(4); - assertThat(successfulTests).map(BuildResult.LocalCITestJobDTO::getName).containsExactlyInAnyOrder("testMergeSort()", "testUseBubbleSortForSmallList()", "testBubbleSort()", + assertThat(successfulTests).map(LocalCITestJobDTO::name).containsExactlyInAnyOrder("testMergeSort()", "testUseBubbleSortForSmallList()", "testBubbleSort()", "testUseMergeSortForBigList()"); } @@ -184,8 +184,8 @@ void testOutputInvalidXMLCharacters() throws IOException { assertThat(successfulTests).isEmpty(); assertThat(failedTests).hasSize(1); var test = failedTests.getFirst(); - assertThat(test.getName()).isEqualTo("CompileLinkedList"); - assertThat(test.getTestMessages().getFirst()).isEqualTo("Build for directory ../assignment/build failed. Returncode is 2."); + assertThat(test.name()).isEqualTo("CompileLinkedList"); + assertThat(test.testMessages().getFirst()).isEqualTo("Build for directory ../assignment/build failed. Returncode is 2."); } @Test @@ -205,8 +205,8 @@ void testEmptyTestMessage() throws IOException { TestResultXmlParser.processTestResultFile(input, failedTests, successfulTests); assertThat(failedTests).hasSize(1); var test = failedTests.getFirst(); - assertThat(test.getName()).isEqualTo("mwe-name"); - assertThat(test.getTestMessages()).hasSize(1).contains(""); + assertThat(test.name()).isEqualTo("mwe-name"); + assertThat(test.testMessages()).hasSize(1).contains(""); } @Test @@ -242,7 +242,7 @@ void testNestedTestsuite() throws IOException { TestResultXmlParser.processTestResultFile(input, failedTests, successfulTests); // @formatter:off - assertThat(successfulTests).extracting(BuildResult.LocalCITestJobDTO::getName).containsExactlyInAnyOrder( + assertThat(successfulTests).extracting(LocalCITestJobDTO::name).containsExactlyInAnyOrder( "Properties.Checked by SmallCheck.Testing filtering in A", "Properties.Checked by SmallCheck.Testing mapping in A", "Properties.Checked by SmallCheck.Testing filtering in B", @@ -280,7 +280,7 @@ void testMultipleTestsuite() throws IOException { TestResultXmlParser.processTestResultFile(input, failedTests, successfulTests); // @formatter:off - assertThat(successfulTests).extracting(BuildResult.LocalCITestJobDTO::getName).containsExactlyInAnyOrder( + assertThat(successfulTests).extracting(LocalCITestJobDTO::name).containsExactlyInAnyOrder( "SuiteA.Test1", "SuiteA.Test2", "SuiteA.Test3", @@ -310,7 +310,7 @@ void testNestedTestsuiteMissingNames() throws IOException { TestResultXmlParser.processTestResultFile(input, failedTests, successfulTests); - assertThat(successfulTests).extracting(BuildResult.LocalCITestJobDTO::getName).containsExactlyInAnyOrder("Test1", "Test2", "Test3"); + assertThat(successfulTests).extracting(LocalCITestJobDTO::name).containsExactlyInAnyOrder("Test1", "Test2", "Test3"); assertThat(failedTests).isEmpty(); } @@ -329,7 +329,7 @@ void testXmlProlog() throws IOException { TestResultXmlParser.processTestResultFile(input, failedTests, successfulTests); - assertThat(successfulTests).singleElement().extracting(BuildResult.LocalCITestJobDTO::getName).isEqualTo("Test"); + assertThat(successfulTests).singleElement().extracting(LocalCITestJobDTO::name).isEqualTo("Test"); assertThat(failedTests).isEmpty(); } @@ -346,7 +346,7 @@ void testRootTestsuiteNameIgnored() throws IOException { TestResultXmlParser.processTestResultFile(input, failedTests, successfulTests); - assertThat(successfulTests).singleElement().extracting(BuildResult.LocalCITestJobDTO::getName).isEqualTo("Suite.Test"); + assertThat(successfulTests).singleElement().extracting(LocalCITestJobDTO::name).isEqualTo("Suite.Test"); assertThat(failedTests).isEmpty(); } @@ -365,7 +365,7 @@ void testSingleTopLevelTestsuiteNameIgnored() throws IOException { TestResultXmlParser.processTestResultFile(input, failedTests, successfulTests); - assertThat(successfulTests).singleElement().extracting(BuildResult.LocalCITestJobDTO::getName).isEqualTo("Suite.Test"); + assertThat(successfulTests).singleElement().extracting(LocalCITestJobDTO::name).isEqualTo("Suite.Test"); assertThat(failedTests).isEmpty(); } @@ -385,7 +385,7 @@ void testMixedNestedTestsuiteTestcase() throws IOException { TestResultXmlParser.processTestResultFile(input, failedTests, successfulTests); - assertThat(successfulTests).extracting(BuildResult.LocalCITestJobDTO::getName).containsExactlyInAnyOrder("Test", "Suite.Test"); + assertThat(successfulTests).extracting(LocalCITestJobDTO::name).containsExactlyInAnyOrder("Test", "Suite.Test"); assertThat(failedTests).isEmpty(); } } diff --git a/src/test/java/de/tum/cit/aet/artemis/core/service/TitleCacheEvictionServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/core/service/TitleCacheEvictionServiceTest.java index 30f880543785..9f969fb29328 100644 --- a/src/test/java/de/tum/cit/aet/artemis/core/service/TitleCacheEvictionServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/core/service/TitleCacheEvictionServiceTest.java @@ -20,8 +20,6 @@ import de.tum.cit.aet.artemis.modeling.domain.DiagramType; import de.tum.cit.aet.artemis.modeling.repository.ApollonDiagramRepository; import de.tum.cit.aet.artemis.modeling.util.ModelingExerciseFactory; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.repository.hestia.ExerciseHintRepository; import de.tum.cit.aet.artemis.programming.util.ProgrammingExerciseUtilService; import de.tum.cit.aet.artemis.shared.base.AbstractSpringIntegrationIndependentTest; import de.tum.cit.aet.artemis.text.util.TextExerciseUtilService; @@ -50,9 +48,6 @@ class TitleCacheEvictionServiceTest extends AbstractSpringIntegrationIndependent @Autowired private ExamRepository examRepository; - @Autowired - private ExerciseHintRepository exerciseHintRepository; - @Autowired private CourseUtilService courseUtilService; @@ -211,40 +206,6 @@ void testEvictsTitleOnUpdateTitleOrDeleteExam() { })); } - @Test - void testEvictsTitleOnUpdateTitleOrDeleteExerciseHint() { - var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(); - var exercise = (ProgrammingExercise) course.getExercises().stream().findAny().orElseThrow(); - programmingExerciseUtilService.addHintsToExercise(exercise); - var hint = exercise.getExerciseHints().stream().findFirst().orElseThrow(); - testCacheEvicted("exerciseHintTitle", () -> new Tuple<>(exercise.getId() + "-" + hint.getId(), hint.getTitle()), List.of( - // Should evict as we change the title - () -> { - hint.setTitle("testEvictsTitleOnUpdateTitleOrDeleteExerciseHint"); - exerciseHintRepository.save(hint); - return true; - }, - // Should not evict as title remains the same - () -> { - hint.setDescription("testEvictsTitleOnUpdateTitleOrDeleteExerciseHint"); // Change some other values - exerciseHintRepository.save(hint); - return false; - }, - // Should not do something if the exercise is missing - () -> { - hint.setExercise(null); - hint.setTitle("testEvictsTitleOnUpdateTitleOrDeleteExerciseHint"); - exerciseHintRepository.save(hint); - return false; - }, - // Should evict after deletion - () -> { - hint.setExercise(exercise); - exerciseHintRepository.delete(hint); - return true; - })); - } - private void testCacheEvicted(String cacheName, Supplier> idTitleSupplier, List> entityModifiers) { var cache = cacheManager.getCache(cacheName); assertThat(cache).isNotNull(); diff --git a/src/test/java/de/tum/cit/aet/artemis/core/util/CourseTestService.java b/src/test/java/de/tum/cit/aet/artemis/core/util/CourseTestService.java index 690a781a5848..82696055a742 100644 --- a/src/test/java/de/tum/cit/aet/artemis/core/util/CourseTestService.java +++ b/src/test/java/de/tum/cit/aet/artemis/core/util/CourseTestService.java @@ -2488,7 +2488,7 @@ public void testCleanupCourseAsInstructor_no_Archive() throws Exception { // Test public void testCleanupCourseAsInstructor() throws Exception { // Generate a course that has an archive - var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, false, ProgrammingLanguage.JAVA); + var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, ProgrammingLanguage.JAVA); course.setCourseArchivePath("some-archive-path"); course = courseRepo.save(course); diff --git a/src/test/java/de/tum/cit/aet/artemis/plagiarism/PlagiarismDetectionServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/plagiarism/PlagiarismDetectionServiceTest.java index 2c2f971a37a9..560a74a893c7 100644 --- a/src/test/java/de/tum/cit/aet/artemis/plagiarism/PlagiarismDetectionServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/plagiarism/PlagiarismDetectionServiceTest.java @@ -93,7 +93,7 @@ void shouldExecuteChecksForProgrammingExercise() throws IOException, ExitExcepti .thenReturn(programmingPlagiarismResult); // and - var programmingLanguageFeature = new ProgrammingLanguageFeature(null, false, false, true, false, false, emptyList(), false, false); + var programmingLanguageFeature = new ProgrammingLanguageFeature(null, false, false, true, false, false, emptyList(), false); when(programmingLanguageFeatureService.getProgrammingLanguageFeatures(any())).thenReturn(programmingLanguageFeature); // when @@ -107,7 +107,7 @@ void shouldExecuteChecksForProgrammingExercise() throws IOException, ExitExcepti void shouldThrowExceptionOnUnsupportedProgrammingLanguage() { // given var programmingExercise = new ProgrammingExercise(); - var programmingLanguageFeature = new ProgrammingLanguageFeature(null, false, false, false, false, false, emptyList(), false, false); + var programmingLanguageFeature = new ProgrammingLanguageFeature(null, false, false, false, false, false, emptyList(), false); when(programmingLanguageFeatureService.getProgrammingLanguageFeatures(any())).thenReturn(programmingLanguageFeature); // expect @@ -125,7 +125,7 @@ void shouldExecuteChecksWithJplagReportForProgrammingExercise() throws Programmi .thenReturn(zipFile); // and - var programmingLanguageFeature = new ProgrammingLanguageFeature(null, false, false, true, false, false, emptyList(), false, false); + var programmingLanguageFeature = new ProgrammingLanguageFeature(null, false, false, true, false, false, emptyList(), false); when(programmingLanguageFeatureService.getProgrammingLanguageFeatures(any())).thenReturn(programmingLanguageFeature); // when @@ -139,7 +139,7 @@ void shouldExecuteChecksWithJplagReportForProgrammingExercise() throws Programmi void shouldThrowExceptionOnUnsupportedProgrammingLanguageForChecksWithJplagReport() { // given var programmingExercise = new ProgrammingExercise(); - var programmingLanguageFeature = new ProgrammingLanguageFeature(null, false, false, false, false, false, emptyList(), false, false); + var programmingLanguageFeature = new ProgrammingLanguageFeature(null, false, false, false, false, false, emptyList(), false); when(programmingLanguageFeatureService.getProgrammingLanguageFeatures(any())).thenReturn(programmingLanguageFeature); // expect diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/AbstractProgrammingIntegrationIndependentTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/AbstractProgrammingIntegrationIndependentTest.java index 21791e5d24f7..3bacbd850f01 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/AbstractProgrammingIntegrationIndependentTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/AbstractProgrammingIntegrationIndependentTest.java @@ -19,13 +19,6 @@ import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseBuildConfigRepository; import de.tum.cit.aet.artemis.programming.repository.SolutionProgrammingExerciseParticipationRepository; import de.tum.cit.aet.artemis.programming.repository.StaticCodeAnalysisCategoryRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CodeHintRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CoverageFileReportRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CoverageReportRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ExerciseHintActivationRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ExerciseHintRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseSolutionEntryRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.TestwiseCoverageReportEntryRepository; import de.tum.cit.aet.artemis.programming.repository.settings.IdeRepository; import de.tum.cit.aet.artemis.programming.repository.settings.UserIdeMappingRepository; import de.tum.cit.aet.artemis.programming.service.AuxiliaryRepositoryService; @@ -33,9 +26,7 @@ import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseFeedbackCreationService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseGradingService; import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseRepositoryService; -import de.tum.cit.aet.artemis.programming.service.hestia.CodeHintService; -import de.tum.cit.aet.artemis.programming.service.hestia.ExerciseHintService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseTaskService; +import de.tum.cit.aet.artemis.programming.service.ProgrammingExerciseTaskService; import de.tum.cit.aet.artemis.programming.test_repository.ProgrammingExerciseStudentParticipationTestRepository; import de.tum.cit.aet.artemis.programming.test_repository.ProgrammingExerciseTaskTestRepository; import de.tum.cit.aet.artemis.programming.test_repository.ProgrammingExerciseTestCaseTestRepository; @@ -51,30 +42,12 @@ public abstract class AbstractProgrammingIntegrationIndependentTest extends Abst @Autowired protected AuxiliaryRepositoryRepository auxiliaryRepositoryRepository; - @Autowired - protected CodeHintRepository codeHintRepository; - - @Autowired - protected CoverageFileReportRepository coverageFileReportRepository; - - @Autowired - protected CoverageReportRepository coverageReportRepository; - - @Autowired - protected ExerciseHintActivationRepository exerciseHintActivationRepository; - - @Autowired - protected ExerciseHintRepository exerciseHintRepository; - @Autowired protected IdeRepository ideRepository; @Autowired protected ProgrammingExerciseBuildConfigRepository programmingExerciseBuildConfigRepository; - @Autowired - protected ProgrammingExerciseSolutionEntryRepository programmingExerciseSolutionEntryRepository; - @Autowired protected ProgrammingExerciseStudentParticipationTestRepository programmingExerciseStudentParticipationRepository; @@ -96,9 +69,6 @@ public abstract class AbstractProgrammingIntegrationIndependentTest extends Abst @Autowired protected StaticCodeAnalysisCategoryRepository staticCodeAnalysisCategoryRepository; - @Autowired - protected TestwiseCoverageReportEntryRepository testwiseCoverageReportEntryRepository; - @Autowired protected UserIdeMappingRepository userIdeMappingRepository; @@ -131,12 +101,6 @@ public abstract class AbstractProgrammingIntegrationIndependentTest extends Abst @Autowired protected BuildLogEntryService buildLogEntryService; - @Autowired - protected CodeHintService codeHintService; - - @Autowired - protected ExerciseHintService exerciseHintService; - @Autowired protected GitUtilService gitUtilService; diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/AbstractProgrammingIntegrationLocalCILocalVCTestBase.java b/src/test/java/de/tum/cit/aet/artemis/programming/AbstractProgrammingIntegrationLocalCILocalVCTestBase.java index 56a168f7e776..5112f51af58a 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/AbstractProgrammingIntegrationLocalCILocalVCTestBase.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/AbstractProgrammingIntegrationLocalCILocalVCTestBase.java @@ -25,19 +25,11 @@ import de.tum.cit.aet.artemis.programming.domain.ProjectType; import de.tum.cit.aet.artemis.programming.domain.SolutionProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.TemplateProgrammingExerciseParticipation; -import de.tum.cit.aet.artemis.programming.hestia.util.HestiaUtilTestService; import de.tum.cit.aet.artemis.programming.repository.AuxiliaryRepositoryRepository; +import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseGitDiffReportRepository; import de.tum.cit.aet.artemis.programming.repository.SolutionProgrammingExerciseParticipationRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CoverageFileReportRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CoverageReportRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseGitDiffReportRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.TestwiseCoverageReportEntryRepository; import de.tum.cit.aet.artemis.programming.service.BuildLogEntryService; import de.tum.cit.aet.artemis.programming.service.ParticipationVcsAccessTokenService; -import de.tum.cit.aet.artemis.programming.service.hestia.ProgrammingExerciseGitDiffReportService; -import de.tum.cit.aet.artemis.programming.service.hestia.TestwiseCoverageService; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralTestCaseService; -import de.tum.cit.aet.artemis.programming.service.hestia.structural.StructuralTestCaseService; import de.tum.cit.aet.artemis.programming.service.localci.LocalCIResultService; import de.tum.cit.aet.artemis.programming.service.localci.LocalCITriggerService; import de.tum.cit.aet.artemis.programming.service.localvc.LocalVCServletService; @@ -67,12 +59,6 @@ public abstract class AbstractProgrammingIntegrationLocalCILocalVCTestBase exten @Autowired protected AuxiliaryRepositoryRepository auxiliaryRepositoryRepository; - @Autowired - protected CoverageFileReportRepository coverageFileReportRepository; - - @Autowired - protected CoverageReportRepository coverageReportRepository; - @Autowired protected ProgrammingExerciseGitDiffReportRepository reportRepository; @@ -85,9 +71,6 @@ public abstract class AbstractProgrammingIntegrationLocalCILocalVCTestBase exten @Autowired protected SolutionProgrammingExerciseParticipationRepository solutionProgrammingExerciseRepository; - @Autowired - protected TestwiseCoverageReportEntryRepository testwiseCoverageReportEntryRepository; - // External Repositories @Autowired protected ExamRepository examRepository; @@ -99,15 +82,9 @@ public abstract class AbstractProgrammingIntegrationLocalCILocalVCTestBase exten protected TeamRepository teamRepository; // Services - @Autowired - protected BehavioralTestCaseService behavioralTestCaseService; - @Autowired protected BuildLogEntryService buildLogEntryService; - @Autowired - protected HestiaUtilTestService hestiaUtilTestService; - @Autowired protected LocalCIResultService localCIResultService; @@ -120,15 +97,6 @@ public abstract class AbstractProgrammingIntegrationLocalCILocalVCTestBase exten @Autowired protected ParticipationVcsAccessTokenService participationVcsAccessTokenService; - @Autowired - protected ProgrammingExerciseGitDiffReportService reportService; - - @Autowired - protected StructuralTestCaseService structuralTestCaseService; - - @Autowired - protected TestwiseCoverageService testwiseCoverageService; - // External Services // Util Services diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/BuildPlanIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/BuildPlanIntegrationTest.java index 82809c5875e2..e62a95dd68aa 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/BuildPlanIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/BuildPlanIntegrationTest.java @@ -31,7 +31,6 @@ void init() { programmingExercise.setProjectType(ProjectType.MAVEN_MAVEN); programmingExercise.setStaticCodeAnalysisEnabled(true); buildConfig.setSequentialTestRuns(false); - buildConfig.setTestwiseCoverageEnabled(false); var savedBuildConfig = programmingExerciseBuildConfigRepository.save(buildConfig); programmingExercise.setBuildConfig(savedBuildConfig); diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseIntegrationJenkinsGitlabTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseIntegrationJenkinsGitlabTest.java index f2b10fd57caf..521e9f7784db 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseIntegrationJenkinsGitlabTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseIntegrationJenkinsGitlabTest.java @@ -234,7 +234,7 @@ void testGetProgrammingExerciseWithJustTemplateAndSolutionParticipation(boolean @ValueSource(booleans = { true, false }) @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") void testGetProgrammingExerciseWithTemplateAndSolutionParticipationAndAuxiliaryRepositories(boolean withSubmissionResults) throws Exception { - programmingExerciseIntegrationTestService.testGetProgrammingExerciseWithTemplateAndSolutionParticipationAndAuxiliaryRepositories(withSubmissionResults, false); + programmingExerciseIntegrationTestService.testGetProgrammingExerciseWithTemplateAndSolutionParticipationAndAuxiliaryRepositories(withSubmissionResults); } @Test @@ -345,12 +345,6 @@ void updateProgrammingExercise_updatingSCAEnabledOption_badRequest() throws Exce programmingExerciseIntegrationTestService.updateProgrammingExerciseShouldFailWithBadRequestWhenUpdatingSCAOption(); } - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateProgrammingExercise_updatingCoverageOption_badRequest() throws Exception { - programmingExerciseIntegrationTestService.updateProgrammingExerciseShouldFailWithBadRequestWhenUpdatingCoverageOption(); - } - @Test @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") void updateExerciseDueDateWithIndividualDueDateUpdate() throws Exception { @@ -612,21 +606,6 @@ void createProgrammingExercise_notIncluded_invalidBonusPoints_badRequest() throw programmingExerciseIntegrationTestService.createProgrammingExercise_notIncluded_invalidBonusPoints_badRequest(); } - private static Set generateSupportedLanguagesWithoutJavaAndKotlin() { - Set supportedLanguages = ArgumentSources.generateJenkinsSupportedLanguages(); - supportedLanguages.remove(ProgrammingLanguage.JAVA); - supportedLanguages.remove(ProgrammingLanguage.KOTLIN); - return supportedLanguages; - } - - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - @ParameterizedTest(name = "{displayName} [{index}] {argumentsWithNames}") - // It should return a bad request error for all ProgrammingExercises except Java and Kotlin. - @MethodSource("generateSupportedLanguagesWithoutJavaAndKotlin") - void createProgrammingExercise_testwiseCoverageAnalysisNotSupported_badRequest(ProgrammingLanguage programmingLanguage) throws Exception { - programmingExerciseIntegrationTestService.createProgrammingExercise_testwiseCoverageAnalysisNotSupported_badRequest(programmingLanguage); - } - @Test @WithMockUser(username = TEST_PREFIX + "instructoralt1", roles = "INSTRUCTOR") void importProgrammingExercise_sourceExerciseIdNegative_badRequest() throws Exception { diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseIntegrationTestService.java b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseIntegrationTestService.java index b7885c881155..415101303e07 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseIntegrationTestService.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseIntegrationTestService.java @@ -109,7 +109,6 @@ import de.tum.cit.aet.artemis.programming.dto.ProgrammingExerciseResetOptionsDTO; import de.tum.cit.aet.artemis.programming.dto.ProgrammingExerciseTestCaseDTO; import de.tum.cit.aet.artemis.programming.dto.ProgrammingExerciseTestCaseStateDTO; -import de.tum.cit.aet.artemis.programming.hestia.util.HestiaUtilTestService; import de.tum.cit.aet.artemis.programming.repository.AuxiliaryRepositoryRepository; import de.tum.cit.aet.artemis.programming.service.GitService; import de.tum.cit.aet.artemis.programming.service.UriService; @@ -124,11 +123,13 @@ import de.tum.cit.aet.artemis.programming.util.MockDelegate; import de.tum.cit.aet.artemis.programming.util.ProgrammingExerciseFactory; import de.tum.cit.aet.artemis.programming.util.ProgrammingExerciseUtilService; +import de.tum.cit.aet.artemis.programming.util.ProgrammingUtilTestService; import de.tum.cit.aet.artemis.text.util.TextExerciseUtilService; /** * Note: this class should be independent of the actual VCS and CIS and contains common test logic for scenarios: * 1) Jenkins + Gitlab + * 2) LocalVC + LocalCI */ @Service public class ProgrammingExerciseIntegrationTestService { @@ -197,9 +198,6 @@ public class ProgrammingExerciseIntegrationTestService { @Autowired private TextExerciseUtilService textExerciseUtilService; - @Autowired - protected HestiaUtilTestService hestiaUtilTestService; - @Autowired private ProgrammingExerciseTestRepository programmingExerciseTestRepository; @@ -212,6 +210,9 @@ public class ProgrammingExerciseIntegrationTestService { @Autowired private GradingCriterionRepository gradingCriterionRepository; + @Autowired + private ProgrammingUtilTestService programmingUtilTestService; + private Course course; public ProgrammingExercise programmingExercise; @@ -579,19 +580,19 @@ void testExportSubmissionsByParticipationIds_instructorNotInCourse_forbidden() t } void testExportSubmissionsByStudentLogins() throws Exception { - File downloadedFile = exportSubmissionsByStudentLogins(HttpStatus.OK); + File downloadedFile = exportSubmissionsByStudentLogins(); assertThat(downloadedFile).exists(); // TODO: unzip the files and add some checks } - private File exportSubmissionsByStudentLogins(HttpStatus expectedStatus) throws Exception { + private File exportSubmissionsByStudentLogins() throws Exception { var repository1 = gitService.getExistingCheckedOutRepositoryByLocalPath(localRepoFile.toPath(), null); var repository2 = gitService.getExistingCheckedOutRepositoryByLocalPath(localRepoFile2.toPath(), null); doReturn(repository1).when(gitService).getOrCheckoutRepository(eq(participation1.getVcsRepositoryUri()), anyString(), anyBoolean()); doReturn(repository2).when(gitService).getOrCheckoutRepository(eq(participation2.getVcsRepositoryUri()), anyString(), anyBoolean()); final var path = "/api/programming-exercises/" + programmingExercise.getId() + "/export-repos-by-participant-identifiers/" + userPrefix + "student1," + userPrefix + "student2"; - return request.postWithResponseBodyFile(path, getOptions(), expectedStatus); + return request.postWithResponseBodyFile(path, getOptions(), HttpStatus.OK); } private RepositoryExportOptionsDTO getOptions() { @@ -763,24 +764,19 @@ void testGetProgrammingExerciseWithJustTemplateAndSolutionParticipation(boolean assertThat(programmingExerciseServer.getStudentParticipations()).isEmpty(); } - void testGetProgrammingExerciseWithTemplateAndSolutionParticipationAndAuxiliaryRepositories(boolean withSubmissionResults, boolean withGradingCriteria) throws Exception { + void testGetProgrammingExerciseWithTemplateAndSolutionParticipationAndAuxiliaryRepositories(boolean withSubmissionResults) throws Exception { AuxiliaryRepository auxiliaryRepository = programmingExerciseUtilService.addAuxiliaryRepositoryToExercise(programmingExercise); Set gradingCriteria = exerciseUtilService.addGradingInstructionsToExercise(programmingExercise); - gradingCriteria = Set.copyOf(gradingCriterionRepository.saveAll(gradingCriteria)); + gradingCriterionRepository.saveAll(gradingCriteria); programmingExercise = programmingExerciseRepository.save(programmingExercise); var path = "/api/programming-exercises/" + programmingExercise.getId() + "/with-template-and-solution-participation" + "?withSubmissionResults=" + withSubmissionResults - + "&withGradingCriteria=" + withGradingCriteria; + + "&withGradingCriteria=" + false; var programmingExerciseServer = request.get(path, HttpStatus.OK, ProgrammingExercise.class); checkTemplateAndSolutionParticipationsFromServer(programmingExerciseServer); assertThat(programmingExerciseServer.getAuxiliaryRepositories()).hasSize(1).containsExactly(auxiliaryRepository); - if (withGradingCriteria) { - assertThat(programmingExerciseServer.getGradingCriteria()).containsAll(gradingCriteria); - } - else { - assertThat(programmingExerciseServer.getGradingCriteria()).isEmpty(); - } + assertThat(programmingExerciseServer.getGradingCriteria()).isEmpty(); } private void checkTemplateAndSolutionParticipationsFromServer(ProgrammingExercise programmingExerciseServer) { @@ -962,18 +958,6 @@ void updateProgrammingExerciseShouldFailWithBadRequestWhenUpdatingSCAOption() th request.put("/api/programming-exercises", updatedExercise, HttpStatus.BAD_REQUEST); } - /** - * This test checks that it is not allowed to change coverage enabled option - */ - void updateProgrammingExerciseShouldFailWithBadRequestWhenUpdatingCoverageOption() throws Exception { - mockBuildPlanAndRepositoryCheck(programmingExercise); - - ProgrammingExercise updatedExercise = programmingExercise; - updatedExercise.getBuildConfig().setTestwiseCoverageEnabled(true); - - request.put("/api/programming-exercises", updatedExercise, HttpStatus.BAD_REQUEST); - } - void updateExerciseDueDateWithIndividualDueDateUpdate() throws Exception { mockBuildPlanAndRepositoryCheck(programmingExercise); mockConfigureRepository(programmingExercise); @@ -988,6 +972,7 @@ void updateExerciseDueDateWithIndividualDueDateUpdate() throws Exception { } programmingExercise.setDueDate(ZonedDateTime.now().plusHours(12)); + assertThat(programmingExercise.getDueDate()).isNotNull(); programmingExercise.setReleaseDate(programmingExercise.getDueDate().minusDays(1)); request.put("/api/programming-exercises", programmingExercise, HttpStatus.OK); @@ -1308,16 +1293,6 @@ void createProgrammingExercise_notIncluded_invalidBonusPoints_badRequest() throw request.post("/api/programming-exercises/setup", programmingExercise, HttpStatus.BAD_REQUEST); } - void createProgrammingExercise_testwiseCoverageAnalysisNotSupported_badRequest(ProgrammingLanguage programmingLanguage) throws Exception { - programmingExercise.setId(null); - programmingExercise.setProjectType(null); - programmingExercise.setTitle("New title"); - programmingExercise.setShortName("NewShortname"); - programmingExercise.setProgrammingLanguage(programmingLanguage); - programmingExercise.getBuildConfig().setTestwiseCoverageEnabled(true); - request.post("/api/programming-exercises/setup", programmingExercise, HttpStatus.BAD_REQUEST); - } - void importProgrammingExercise_sourceExerciseIdNegative_badRequest() throws Exception { programmingExercise.setId(-1L); request.post("/api/programming-exercises/import/" + programmingExercise.getId(), programmingExercise, HttpStatus.BAD_REQUEST); @@ -1544,7 +1519,7 @@ void updateTestCases_asInstrutor() throws Exception { testCasesResponse.forEach(testCase -> testCase.setExercise(programmingExercise)); final var testCasesInDB = programmingExerciseTestCaseRepository.findByExerciseId(programmingExercise.getId()); - assertThat(new HashSet<>(testCasesResponse)).usingRecursiveFieldByFieldElementComparatorIgnoringFields("exercise", "tasks", "solutionEntries", "coverageEntries") + assertThat(new HashSet<>(testCasesResponse)).usingRecursiveFieldByFieldElementComparatorIgnoringFields("exercise", "tasks") .containsExactlyInAnyOrderElementsOf(testCasesInDB); assertThat(testCasesResponse).allSatisfy(testCase -> { assertThat(testCase.isAfterDueDate()).isTrue(); @@ -2286,34 +2261,11 @@ void testReEvaluateAndUpdateProgrammingExercise_isNotSameGivenExerciseIdInReques request.put("/api/programming-exercises/" + programmingExercise.getId() + "/re-evaluate", programmingExerciseToBeConflicted, HttpStatus.CONFLICT); } - void test_redirectGetSolutionRepositoryFilesWithoutContent() throws Exception { - test_redirectGetSolutionRepositoryFilesWithoutContent((exercise, files) -> { - LocalRepository localRepository = new LocalRepository("main"); - try { - hestiaUtilTestService.setupSolution(files, exercise, localRepository); - } - catch (Exception e) { - fail("Setup solution threw unexpected exception: " + e.getMessage()); - } - return localRepository; - }); - } - - private void test_redirectGetSolutionRepositoryFilesWithoutContent(BiFunction, LocalRepository> setupRepositoryMock) throws Exception { - setupRepositoryMock.apply(programmingExercise, Map.ofEntries(Map.entry("A.java", "abc"), Map.entry("B.java", "cde"), Map.entry("C.java", "efg"))); - - var savedExercise = programmingExerciseRepository.findByIdWithTemplateAndSolutionParticipationElseThrow(programmingExercise.getId()); - - // We expect an URL which is the endpoint, with which the file contents can be retrieved - request.getWithForwardedUrl("/api/programming-exercises/" + programmingExercise.getId() + "/file-names", HttpStatus.OK, - "/api/repository/" + savedExercise.getSolutionParticipation().getId() + "/file-names"); - } - void test_redirectGetTemplateRepositoryFilesWithContent() throws Exception { test_redirectGetTemplateRepositoryFilesWithContent((exercise, files) -> { LocalRepository localRepository = new LocalRepository("main"); try { - hestiaUtilTestService.setupTemplate(files, exercise, localRepository); + programmingUtilTestService.setupTemplate(files, exercise, localRepository); } catch (Exception e) { fail("Setup template threw unexpected exception: " + e.getMessage()); @@ -2337,7 +2289,7 @@ void testRedirectGetParticipationRepositoryFilesWithContentAtCommit(String testP var studentLogin = testPrefix + "student1"; try { localRepository.configureRepos("testLocalRepo", "testOriginRepo"); - return hestiaUtilTestService.setupSubmission(files, exercise, localRepository, studentLogin); + return programmingUtilTestService.setupSubmission(files, exercise, localRepository, studentLogin); } catch (Exception e) { fail("Test setup failed"); @@ -2348,8 +2300,13 @@ void testRedirectGetParticipationRepositoryFilesWithContentAtCommit(String testP private void testRedirectGetParticipationRepositoryFilesWithContentAtCommit(BiFunction, ProgrammingSubmission> setupRepositoryMock) throws Exception { - var submission = setupRepositoryMock.apply(programmingExercise, Map.ofEntries(Map.entry("A.java", "abc"), Map.entry("B.java", "cde"), Map.entry("C.java", "efg"))); - String filesWithContentsAsJson = "{\n" + " \"C.java\" : \"efg\",\n" + " \"B.java\" : \"cde\",\n" + " \"A.java\" : \"abc\"\n" + "}"; + var submission = setupRepositoryMock.apply(programmingExercise, Map.of("A.java", "abc", "B.java", "cde", "C.java", "efg")); + String filesWithContentsAsJson = """ + { + "C.java" : "efg", + "B.java" : "cde", + "A.java" : "abc" + }"""; request.getWithFileContents("/api/programming-exercise-participations/" + participation1.getId() + "/files-content/" + submission.getCommitHash(), HttpStatus.OK, filesWithContentsAsJson); @@ -2362,7 +2319,7 @@ void testRedirectGetParticipationRepositoryFilesWithContentAtCommitForbidden(Str var studentLogin = testPrefix + "student1"; try { localRepository.configureRepos("testLocalRepo", "testOriginRepo"); - return hestiaUtilTestService.setupSubmission(files, exercise, localRepository, studentLogin); + return programmingUtilTestService.setupSubmission(files, exercise, localRepository, studentLogin); } catch (Exception e) { fail("Test setup failed"); @@ -2373,7 +2330,7 @@ void testRedirectGetParticipationRepositoryFilesWithContentAtCommitForbidden(Str private void testRedirectGetParticipationRepositoryFilesWithContentAtCommitForbidden( BiFunction, ProgrammingSubmission> setupRepositoryMock) throws Exception { - var submission = setupRepositoryMock.apply(programmingExercise, Map.ofEntries(Map.entry("A.java", "abc"), Map.entry("B.java", "cde"), Map.entry("C.java", "efg"))); + var submission = setupRepositoryMock.apply(programmingExercise, Map.of("A.java", "abc", "B.java", "cde", "C.java", "efg")); request.get("/api/programming-exercise-participations/" + participation1.getId() + "/files-content/" + submission.getCommitHash(), HttpStatus.FORBIDDEN, Map.class); } diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseLocalVCIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseLocalVCIntegrationTest.java index c395a37b3095..1a4022a4a7f6 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseLocalVCIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseLocalVCIntegrationTest.java @@ -175,12 +175,6 @@ void testReEvaluateAndUpdateProgrammingExercise_isNotSameGivenExerciseIdInReques programmingExerciseIntegrationTestService.testReEvaluateAndUpdateProgrammingExercise_isNotSameGivenExerciseIdInRequestBody_conflict(); } - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void test_redirectGetSolutionRepositoryFilesWithoutContent() throws Exception { - programmingExerciseIntegrationTestService.test_redirectGetSolutionRepositoryFilesWithoutContent(); - } - @Test @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") void test_redirectGetTemplateRepositoryFilesWithContent() throws Exception { diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseResultJenkinsIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseResultJenkinsIntegrationTest.java index 9e0650403412..8340f501af41 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseResultJenkinsIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseResultJenkinsIntegrationTest.java @@ -117,7 +117,7 @@ void shouldRejectCommitsWithoutCommitHash() { CommitDTO dummy = new CommitDTO("", "", ""); var notification = ProgrammingExerciseFactory.generateTestResultDTO(null, Constants.ASSIGNMENT_REPO_NAME, null, ProgrammingLanguage.JAVA, true, List.of("test1", "test2", "test4"), List.of(), new ArrayList<>(), new ArrayList<>(List.of(dummy)), null); - notification.getCommits().clear(); + notification.commits().clear(); programmingExerciseResultTestService.shouldRejectNotificationWithoutCommitHash(notification); } diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseServiceIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseServiceIntegrationTest.java index e32956ac6bd8..1f0f094f10b2 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseServiceIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseServiceIntegrationTest.java @@ -19,8 +19,6 @@ import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; import de.tum.cit.aet.artemis.programming.domain.StaticCodeAnalysisCategory; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; import de.tum.cit.aet.artemis.programming.domain.submissionpolicy.LockRepositoryPolicy; import de.tum.cit.aet.artemis.programming.domain.submissionpolicy.SubmissionPenaltyPolicy; import de.tum.cit.aet.artemis.programming.domain.submissionpolicy.SubmissionPolicy; @@ -45,10 +43,7 @@ void setUp() { programmingExercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); // Needed, as we need the test cases for the next steps programmingExercise = programmingExerciseUtilService.loadProgrammingExerciseWithEagerReferences(programmingExercise); - programmingExerciseUtilService.addHintsToExercise(programmingExercise); programmingExerciseUtilService.addTasksToProgrammingExercise(programmingExercise); - programmingExerciseUtilService.addSolutionEntriesToProgrammingExercise(programmingExercise); - programmingExerciseUtilService.addCodeHintsToProgrammingExercise(programmingExercise); programmingExerciseUtilService.addStaticCodeAnalysisCategoriesToProgrammingExercise(programmingExercise); // Load again to fetch changes to statement and hints while keeping eager refs @@ -82,36 +77,13 @@ void importProgrammingExerciseBasis_baseReferencesGotCloned() { final var newTestCaseIDs = newlyImported.getTestCases().stream().map(ProgrammingExerciseTestCase::getId).collect(Collectors.toSet()); assertThat(newlyImported.getTestCases()).hasSameSizeAs(programmingExercise.getTestCases()); assertThat(programmingExercise.getTestCases()).noneMatch(testCase -> newTestCaseIDs.contains(testCase.getId())); - assertThat(programmingExercise.getTestCases()).usingRecursiveFieldByFieldElementComparatorIgnoringFields("id", "exercise", "tasks", "solutionEntries", "coverageEntries") + assertThat(programmingExercise.getTestCases()).usingRecursiveFieldByFieldElementComparatorIgnoringFields("id", "exercise", "tasks") .containsExactlyInAnyOrderElementsOf(newlyImported.getTestCases()); - final var newHintIDs = newlyImported.getExerciseHints().stream().map(ExerciseHint::getId).collect(Collectors.toSet()); - assertThat(newlyImported.getExerciseHints()).hasSameSizeAs(programmingExercise.getExerciseHints()); - assertThat(programmingExercise.getExerciseHints()).noneMatch(hint -> newHintIDs.contains(hint.getId())); final var newStaticCodeAnalysisCategoriesIDs = newlyImported.getStaticCodeAnalysisCategories().stream().map(StaticCodeAnalysisCategory::getId).collect(Collectors.toSet()); assertThat(newlyImported.getStaticCodeAnalysisCategories()).hasSameSizeAs(programmingExercise.getStaticCodeAnalysisCategories()); assertThat(programmingExercise.getStaticCodeAnalysisCategories()).noneMatch(category -> newStaticCodeAnalysisCategoriesIDs.contains(category.getId())); } - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void importProgrammingExerciseBasis_testsAndHintsHoldTheSameInformation() { - final var imported = importExerciseBase(); - - // All copied hints/tests have the same content are referenced to the new exercise - assertThat(imported.getExerciseHints()).allMatch(hint -> programmingExercise.getExerciseHints().stream().anyMatch( - oldHint -> oldHint.getContent().equals(hint.getContent()) && oldHint.getTitle().equals(hint.getTitle()) && hint.getExercise().getId().equals(imported.getId()))); - assertThat(imported.getExerciseHints().stream().filter(eh -> eh instanceof CodeHint).map(eh -> (CodeHint) eh).collect(Collectors.toSet())) - .allMatch(codeHint -> programmingExercise.getExerciseHints().stream().filter(eh -> eh instanceof CodeHint).map(eh -> (CodeHint) eh) - .anyMatch(oldHint -> oldHint.getTitle().equals(codeHint.getTitle()) - && oldHint.getProgrammingExerciseTask().getTaskName().equals(codeHint.getProgrammingExerciseTask().getTaskName()) - && codeHint.getSolutionEntries().size() == 1 && oldHint.getSolutionEntries().stream().findFirst().orElseThrow().getCode() - .equals(codeHint.getSolutionEntries().stream().findFirst().orElseThrow().getCode()))); - - assertThat(imported.getTestCases()).allMatch(test -> programmingExercise.getTestCases().stream().anyMatch(oldTest -> test.getExercise().getId().equals(imported.getId()) - && oldTest.getTestName().equalsIgnoreCase(test.getTestName()) && oldTest.getWeight().equals(test.getWeight()) && test.getSolutionEntries().size() == 1 - && oldTest.getSolutionEntries().stream().findFirst().orElseThrow().getCode().equals(test.getSolutionEntries().stream().findFirst().orElseThrow().getCode()))); - } - @ParameterizedTest(name = "{displayName} [{index}] {argumentsWithNames}") @MethodSource("submissionPolicyProvider") @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseTemplateIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseTemplateIntegrationTest.java index 2d181a5e7b99..1244f782f744 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseTemplateIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingExerciseTemplateIntegrationTest.java @@ -237,18 +237,6 @@ private Stream languageTypeBuilder() { } argumentBuilder.add(Arguments.of(language, projectType, false)); } - - if (languageFeatures.testwiseCoverageAnalysisSupported()) { - if (projectTypes.isEmpty()) { - argumentBuilder.add(Arguments.of(language, null, true)); - } - for (ProjectType projectType : projectTypes) { - if (projectType == ProjectType.MAVEN_BLACKBOX) { - continue; - } - argumentBuilder.add(Arguments.of(language, projectType, true)); - } - } } return argumentBuilder.build(); } @@ -256,17 +244,17 @@ private Stream languageTypeBuilder() { @ParameterizedTest(name = "{displayName} [{index}] {argumentsWithNames}") @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") @MethodSource("languageTypeBuilder") - void testTemplateExercise(ProgrammingLanguage language, ProjectType projectType, boolean testwiseCoverageAnalysis) throws Exception { + void testTemplateExercise(ProgrammingLanguage language, ProjectType projectType) throws Exception { checkPreconditionsForJavaTemplateExecution(projectType); - runTests(language, projectType, exerciseRepo, TestResult.FAILED, testwiseCoverageAnalysis); + runTests(language, projectType, exerciseRepo, TestResult.FAILED); } @ParameterizedTest(name = "{displayName} [{index}] {argumentsWithNames}") @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") @MethodSource("languageTypeBuilder") - void testTemplateSolution(ProgrammingLanguage language, ProjectType projectType, boolean testwiseCoverageAnalysis) throws Exception { + void testTemplateSolution(ProgrammingLanguage language, ProjectType projectType) throws Exception { checkPreconditionsForJavaTemplateExecution(projectType); - runTests(language, projectType, solutionRepo, TestResult.SUCCESSFUL, testwiseCoverageAnalysis); + runTests(language, projectType, solutionRepo, TestResult.SUCCESSFUL); } private void checkPreconditionsForJavaTemplateExecution(final ProjectType projectType) { @@ -276,24 +264,20 @@ private void checkPreconditionsForJavaTemplateExecution(final ProjectType projec assumeTrue(java17Home != null, "Could not find Java 17. Skipping execution of template tests."); } - private void runTests(ProgrammingLanguage language, ProjectType projectType, LocalRepository repository, TestResult testResult, boolean testwiseCoverageAnalysis) - throws Exception { + private void runTests(ProgrammingLanguage language, ProjectType projectType, LocalRepository repository, TestResult testResult) throws Exception { exercise.setProgrammingLanguage(language); exercise.setProjectType(projectType); mockConnectorRequestsForSetup(exercise, false, true, false); exercise.setChannelName("exercise-pe"); - if (testwiseCoverageAnalysis) { - exercise.getBuildConfig().setTestwiseCoverageEnabled(true); - } request.postWithResponseBody("/api/programming-exercises/setup", exercise, ProgrammingExercise.class, HttpStatus.CREATED); moveAssignmentSourcesOf(repository); int exitCode; if (projectType != null && projectType.isGradle()) { - exitCode = invokeGradle(testwiseCoverageAnalysis); + exitCode = invokeGradle(); } else { - exitCode = invokeMaven(testwiseCoverageAnalysis); + exitCode = invokeMaven(); } if (TestResult.SUCCESSFUL.equals(testResult)) { @@ -308,14 +292,11 @@ private void runTests(ProgrammingLanguage language, ProjectType projectType, Loc assertThat(testResults).containsExactlyInAnyOrderEntriesOf(Map.of(testResult, 12 + (ProgrammingLanguage.JAVA.equals(language) ? 1 : 0))); } - private int invokeMaven(boolean testwiseCoverageAnalysis) throws MavenInvocationException { + private int invokeMaven() throws MavenInvocationException { InvocationRequest mvnRequest = new DefaultInvocationRequest(); mvnRequest.setJavaHome(java17Home); mvnRequest.setPomFile(testRepo.localRepoFile); mvnRequest.addArgs(List.of("clean", "test")); - if (testwiseCoverageAnalysis) { - mvnRequest.addArg("-Pcoverage"); - } mvnRequest.setShowVersion(true); Invoker mvnInvoker = new DefaultInvoker(); @@ -325,17 +306,11 @@ private int invokeMaven(boolean testwiseCoverageAnalysis) throws MavenInvocation return result.getExitCode(); } - private int invokeGradle(boolean recordTestwiseCoverage) { + private int invokeGradle() { try (ProjectConnection connector = GradleConnector.newConnector().forProjectDirectory(testRepo.localRepoFile).useBuildDistribution().connect()) { BuildLauncher launcher = connector.newBuild(); launcher.setJavaHome(java17Home); - String[] tasks; - if (recordTestwiseCoverage) { - tasks = new String[] { "clean", "test", "tiaTests", "--run-all-tests" }; - } - else { - tasks = new String[] { "clean", "test" }; - } + String[] tasks = new String[] { "clean", "test" }; launcher.forTasks(tasks); launcher.run(); } diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingSubmissionAndResultGitlabJenkinsIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingSubmissionAndResultGitlabJenkinsIntegrationTest.java index fe8821411c84..75fe4787b17b 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingSubmissionAndResultGitlabJenkinsIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/ProgrammingSubmissionAndResultGitlabJenkinsIntegrationTest.java @@ -72,7 +72,7 @@ void tearDown() throws Exception { void shouldReceiveBuildLogsOnNewStudentParticipationResult() throws Exception { // Precondition: Database has participation and a programming submission. String userLogin = TEST_PREFIX + "student1"; - var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, false, ProgrammingLanguage.JAVA); + var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, ProgrammingLanguage.JAVA); var exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); exercise = programmingExerciseRepository.findWithEagerStudentParticipationsById(exercise.getId()).orElseThrow(); @@ -102,7 +102,7 @@ void shouldReceiveBuildLogsOnNewStudentParticipationResult() throws Exception { void shouldExtractBuildLogAnalytics_noSca() throws Exception { // Precondition: Database has participation and a programming submission. String userLogin = TEST_PREFIX + "student1"; - Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, false, ProgrammingLanguage.JAVA); + Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, ProgrammingLanguage.JAVA); ProgrammingExercise exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); exercise = programmingExerciseRepository.findWithEagerStudentParticipationsStudentAndLegalSubmissionsById(exercise.getId()).orElseThrow(); var participation = participationUtilService.addStudentParticipationForProgrammingExercise(exercise, userLogin); @@ -138,7 +138,7 @@ private static List getLogs(String dependencyDownloaded, String jobFinis void shouldExtractBuildLogAnalytics_noSca_gradle() throws Exception { // Precondition: Database has participation and a programming submission. String userLogin = TEST_PREFIX + "student1"; - Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, false, ProgrammingLanguage.JAVA); + Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, ProgrammingLanguage.JAVA); ProgrammingExercise exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); exercise = programmingExerciseRepository.findWithEagerStudentParticipationsStudentAndLegalSubmissionsById(exercise.getId()).orElseThrow(); exercise.setProjectType(ProjectType.GRADLE_GRADLE); @@ -168,7 +168,7 @@ void shouldExtractBuildLogAnalytics_noSca_gradle() throws Exception { void shouldExtractBuildLogAnalytics_sca() throws Exception { // Precondition: Database has participation and a programming submission. String userLogin = TEST_PREFIX + "student1"; - Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, false, ProgrammingLanguage.JAVA); + Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, ProgrammingLanguage.JAVA); ProgrammingExercise exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); exercise = programmingExerciseRepository.findWithEagerStudentParticipationsStudentAndLegalSubmissionsById(exercise.getId()).orElseThrow(); var participation = participationUtilService.addStudentParticipationForProgrammingExercise(exercise, userLogin); @@ -196,7 +196,7 @@ void shouldExtractBuildLogAnalytics_sca() throws Exception { void shouldExtractBuildLogAnalytics_unsupportedProgrammingLanguage() throws Exception { // Precondition: Database has participation and a programming submission. String userLogin = TEST_PREFIX + "student1"; - var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, false, ProgrammingLanguage.PYTHON); + var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, ProgrammingLanguage.PYTHON); var exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); var participation = participationUtilService.addStudentParticipationForProgrammingExercise(exercise, userLogin); programmingExerciseUtilService.createProgrammingSubmission(participation, false); @@ -227,7 +227,7 @@ private static Stream shouldSaveBuildLogsOnStudentParticipationArgume void shouldReturnBadRequestWhenPlanKeyDoesntExist(ProgrammingLanguage programmingLanguage, boolean enableStaticCodeAnalysis) throws Exception { // Precondition: Database has participation and a programming submission. String userLogin = TEST_PREFIX + "student1"; - var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, false, programmingLanguage); + var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, programmingLanguage); exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); exercise = programmingExerciseRepository.findWithEagerStudentParticipationsById(exercise.getId()).orElseThrow(); @@ -297,7 +297,7 @@ private static Stream shouldSaveBuildLogsOnStudentParticipationWithou void shouldSaveBuildLogsOnStudentParticipationWithoutResult(ProgrammingLanguage programmingLanguage, boolean enableStaticCodeAnalysis, boolean buildFailed) throws Exception { // Precondition: Database has participation and a programming submission. String userLogin = TEST_PREFIX + "student1"; - var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, false, programmingLanguage); + var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, programmingLanguage); exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); exercise = programmingExerciseRepository.findWithEagerStudentParticipationsById(exercise.getId()).orElseThrow(); @@ -324,7 +324,7 @@ void shouldSaveBuildLogsOnStudentParticipationWithoutResult(ProgrammingLanguage void shouldSaveBuildLogsOnStudentParticipationWithoutSubmissionNorResult(ProgrammingLanguage programmingLanguage, boolean enableStaticCodeAnalysis) throws Exception { // Precondition: Database has participation without result and a programming String userLogin = TEST_PREFIX + "student1"; - var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, false, programmingLanguage); + var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, programmingLanguage); exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); exercise = programmingExerciseRepository.findWithEagerStudentParticipationsById(exercise.getId()).orElseThrow(); @@ -341,7 +341,7 @@ void shouldSaveBuildLogsOnStudentParticipationWithoutSubmissionNorResult(Program @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") void shouldCreateGradleFeedback() throws Exception { String userLogin = TEST_PREFIX + "student1"; - var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, false, JAVA); + var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, JAVA); exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); exercise.setProjectType(ProjectType.GRADLE_GRADLE); exercise = programmingExerciseRepository.save(exercise); @@ -352,7 +352,7 @@ void shouldCreateGradleFeedback() throws Exception { var longErrorMessage = new TestCaseDetailMessageDTO("abc\nmultiline\nfeedback"); var testCase = new TestCaseDTO("test1", "Class", 0d, new ArrayList<>(), List.of(longErrorMessage), new ArrayList<>()); - notification.getResults().getFirst().testCases().set(0, testCase); + notification.results().getFirst().testCases().set(0, testCase); postResult(notification, HttpStatus.OK); diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/RepositoryIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/RepositoryIntegrationTest.java index 64307de24282..1fc69fb5eedb 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/RepositoryIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/RepositoryIntegrationTest.java @@ -1201,13 +1201,6 @@ void testLockStudentRepository_beforeStateRepoConfigured() { assertThat(logsList.getFirst().getArgumentArray()).containsExactly(participation.getId()); } - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testGetSolutionFileNames() throws Exception { - var fileNames = request.get(studentRepoBaseUrl + participation.getId() + "/file-names", HttpStatus.OK, String[].class); - assertThat(fileNames).containsExactly("currentFileName"); - } - private List getFileSubmissions(String fileContent) { List fileSubmissions = new ArrayList<>(); FileSubmission fileSubmission = new FileSubmission(); diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/StaticCodeAnalysisIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/StaticCodeAnalysisIntegrationTest.java index 2a6cccd7d15e..0c2220769f0f 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/StaticCodeAnalysisIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/StaticCodeAnalysisIntegrationTest.java @@ -183,7 +183,7 @@ void testResetCategories(ProgrammingLanguage programmingLanguage) throws Excepti doReturn(ObjectId.fromString(dummyHash)).when(gitService).getLastCommitHash(any()); // Create a programming exercise with real categories - var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(true, false, programmingLanguage); + var course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(true, programmingLanguage); ProgrammingExercise exercise = programmingExerciseRepository .findWithTemplateAndSolutionParticipationTeamAssignmentConfigCategoriesById(course.getExercises().iterator().next().getId()).orElseThrow(); staticCodeAnalysisService.createDefaultCategories(exercise); @@ -369,7 +369,7 @@ void testImportCategories_asTutor() throws Exception { @Test @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") void testImportCategories_differentLanguages() throws Exception { - ProgrammingExercise sourceExercise = programmingExerciseUtilService.addProgrammingExerciseToCourse(course, true, false, ProgrammingLanguage.SWIFT); + ProgrammingExercise sourceExercise = programmingExerciseUtilService.addProgrammingExerciseToCourse(course, true, ProgrammingLanguage.SWIFT); var endpoint = parameterizeEndpoint("/api/programming-exercises/{exerciseId}/static-code-analysis-categories/import", programmingExerciseSCAEnabled); request.patch(endpoint + "?sourceExerciseId=" + sourceExercise.getId(), null, HttpStatus.CONFLICT); diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusBuildScriptGenerationServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusBuildScriptGenerationServiceTest.java index 7bcfb7d3a9aa..73cc266c6a4b 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusBuildScriptGenerationServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusBuildScriptGenerationServiceTest.java @@ -22,11 +22,11 @@ import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseBuildConfig; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.ProjectType; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; +import de.tum.cit.aet.artemis.programming.dto.aeolus.WindfileMetadata; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusBuildPlanService; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusBuildScriptGenerationService; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusTemplateService; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; -import de.tum.cit.aet.artemis.programming.service.aeolus.WindfileMetadata; import de.tum.cit.aet.artemis.shared.base.AbstractSpringIntegrationLocalCILocalVCTest; class AeolusBuildScriptGenerationServiceTest extends AbstractSpringIntegrationLocalCILocalVCTest { @@ -76,10 +76,7 @@ void testBuildScriptGeneration() throws JsonProcessingException { } private Windfile getWindfile() { - Windfile windfile = new Windfile(); - windfile.setApi("v0.0.1"); - windfile.setMetadata(new WindfileMetadata("test", "test", "test", null, null, null, null, null)); - return windfile; + return new Windfile("v0.0.1", new WindfileMetadata("test", "test", "test", null, null, null, null, null), null, null); } private String getSerializedWindfile() throws JsonProcessingException { @@ -97,7 +94,6 @@ void testBuildScriptGenerationUsingBuildPlanGenerationService() throws JsonProce programmingExercise.setProjectType(ProjectType.PLAIN_GRADLE); programmingExercise.setStaticCodeAnalysisEnabled(true); programmingExercise.getBuildConfig().setSequentialTestRuns(true); - programmingExercise.getBuildConfig().setTestwiseCoverageEnabled(true); String script = aeolusBuildScriptGenerationService.getScript(programmingExercise); assertThat(script).isNotNull(); assertThat(script).isEqualTo("imagine a result here"); @@ -105,7 +101,6 @@ void testBuildScriptGenerationUsingBuildPlanGenerationService() throws JsonProce programmingExercise.setProjectType(null); programmingExercise.setStaticCodeAnalysisEnabled(false); programmingExercise.getBuildConfig().setSequentialTestRuns(false); - programmingExercise.getBuildConfig().setTestwiseCoverageEnabled(false); programmingExercise.getBuildConfig().setBuildPlanConfiguration(null); script = aeolusBuildScriptGenerationService.getScript(programmingExercise); assertThat(script).isNotNull(); diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusServiceTest.java index 957f33a9cfb8..32520d1db703 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusServiceTest.java @@ -31,13 +31,13 @@ import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.ProjectType; import de.tum.cit.aet.artemis.programming.domain.VcsRepositoryUri; +import de.tum.cit.aet.artemis.programming.dto.aeolus.AeolusRepository; +import de.tum.cit.aet.artemis.programming.dto.aeolus.ScriptAction; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; +import de.tum.cit.aet.artemis.programming.dto.aeolus.WindfileMetadata; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusBuildPlanService; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusBuildScriptGenerationService; -import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusRepository; import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusTemplateService; -import de.tum.cit.aet.artemis.programming.service.aeolus.ScriptAction; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; -import de.tum.cit.aet.artemis.programming.service.aeolus.WindfileMetadata; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationService; import de.tum.cit.aet.artemis.shared.base.AbstractSpringIntegrationIndependentTest; @@ -84,9 +84,9 @@ void tearDown() throws Exception { */ @Test void testSuccessfulPublishBuildPlan() throws JsonProcessingException { - Windfile mockWindfile = new Windfile(); var expectedPlanKey = "PLAN"; - mockWindfile.setPreProcessingMetadata("PROJECT-" + expectedPlanKey, null, null, null, null, null, null); + var metadata = new WindfileMetadata(null, "PROJECT-" + expectedPlanKey, null, null, null, null, null, null); + Windfile mockWindfile = new Windfile(null, metadata, null, null); aeolusRequestMockProvider.mockSuccessfulPublishBuildPlan(AeolusTarget.JENKINS, expectedPlanKey); String key = aeolusBuildPlanService.publishBuildPlan(mockWindfile, AeolusTarget.JENKINS); @@ -98,9 +98,9 @@ void testSuccessfulPublishBuildPlan() throws JsonProcessingException { */ @Test void testFailedPublishBuildPlan() throws JsonProcessingException { - Windfile mockWindfile = new Windfile(); var expectedPlanKey = "PLAN"; - mockWindfile.setPreProcessingMetadata("PROJECT-" + expectedPlanKey, null, null, null, null, null, null); + var metadata = new WindfileMetadata(null, "PROJECT-" + expectedPlanKey, null, null, null, null, null, null); + Windfile mockWindfile = new Windfile(null, metadata, null, null); aeolusRequestMockProvider.mockFailedPublishBuildPlan(AeolusTarget.JENKINS); String key = aeolusBuildPlanService.publishBuildPlan(mockWindfile, AeolusTarget.JENKINS); @@ -172,7 +172,7 @@ void testRepositoryMapForHaskellWindfileCreation() throws URISyntaxException { @Test void testReturnsNullonUrlNull() throws JsonProcessingException { ReflectionTestUtils.setField(aeolusBuildPlanService, "ciUrl", null); - assertThat(aeolusBuildPlanService.publishBuildPlan(new Windfile(), AeolusTarget.JENKINS)).isNull(); + assertThat(aeolusBuildPlanService.publishBuildPlan(new Windfile(null, null, null, null), AeolusTarget.JENKINS)).isNull(); } @Test @@ -184,11 +184,8 @@ void testBuildScriptGeneration() throws JsonProcessingException { } private Windfile getWindfile() { - Windfile windfile = new Windfile(); - windfile.setApi("v0.0.1"); - windfile.setMetadata(new WindfileMetadata("test", "test", "test", null, null, null, null, null)); - windfile.setActions(List.of(new ScriptAction())); - return windfile; + var action = new ScriptAction(null, null, null, null, null, false, null, null); + return new Windfile("v0.0.1", new WindfileMetadata("test", "test", "test", null, null, null, null, null), List.of(action), null); } private String getSerializedWindfile() throws JsonProcessingException { @@ -204,17 +201,16 @@ void testShouldNotGenerateAnything() throws JsonProcessingException { programmingExercise.setProjectType(ProjectType.PLAIN_GRADLE); programmingExercise.setStaticCodeAnalysisEnabled(true); programmingExercise.getBuildConfig().setSequentialTestRuns(true); - programmingExercise.getBuildConfig().setTestwiseCoverageEnabled(true); String script = aeolusBuildScriptGenerationService.getScript(programmingExercise); assertThat(script).isNull(); } @Test void testGetWindfileFor() throws IOException { - Windfile windfile = aeolusTemplateService.getWindfileFor(ProgrammingLanguage.JAVA, Optional.empty(), false, false, false); + Windfile windfile = aeolusTemplateService.getWindfileFor(ProgrammingLanguage.JAVA, Optional.empty(), false, false); assertThat(windfile).isNotNull(); - assertThat(windfile.getActions()).isNotNull(); - assertThat(windfile.getActions()).hasSize(1); + assertThat(windfile.actions()).isNotNull(); + assertThat(windfile.actions()).hasSize(1); } @Test @@ -224,7 +220,6 @@ void testGetDefaultWindfileFor() { programmingExercise.setProgrammingLanguage(ProgrammingLanguage.HASKELL); programmingExercise.setStaticCodeAnalysisEnabled(true); programmingExercise.getBuildConfig().setSequentialTestRuns(true); - programmingExercise.getBuildConfig().setTestwiseCoverageEnabled(true); Windfile windfile = aeolusTemplateService.getDefaultWindfileFor(programmingExercise); assertThat(windfile).isNull(); } diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusTemplateResourceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusTemplateResourceTest.java index 3c82d8858409..ef5c18e18012 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusTemplateResourceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusTemplateResourceTest.java @@ -18,8 +18,8 @@ import de.tum.cit.aet.artemis.core.user.util.UserUtilService; import de.tum.cit.aet.artemis.core.util.RequestUtilService; -import de.tum.cit.aet.artemis.programming.service.aeolus.ScriptAction; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; +import de.tum.cit.aet.artemis.programming.dto.aeolus.ScriptAction; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; import de.tum.cit.aet.artemis.shared.base.AbstractSpringIntegrationLocalCILocalVCTest; class AeolusTemplateResourceTest extends AbstractSpringIntegrationLocalCILocalVCTest { @@ -39,11 +39,10 @@ void initTestCase() { private static Stream templateProvider() { return Stream.of(new Object[][] { { "JAVA/PLAIN_GRADLE", 1 }, { "JAVA/PLAIN_GRADLE?sequentialRuns=true", 2 }, { "JAVA/PLAIN_GRADLE?staticAnalysis=true", 2 }, - { "JAVA/PLAIN_GRADLE?staticAnalysis=true&testCoverage=true", 2 }, { "JAVA/PLAIN_MAVEN", 1 }, { "JAVA/PLAIN_MAVEN?sequentialRuns=true", 2 }, - { "JAVA/PLAIN_MAVEN?staticAnalysis=true", 2 }, { "JAVA/PLAIN_MAVEN?staticAnalysis=true&testCoverage=true", 3 }, { "JAVA/MAVEN_BLACKBOX", 5 }, + { "JAVA/PLAIN_MAVEN", 1 }, { "JAVA/PLAIN_MAVEN?sequentialRuns=true", 2 }, { "JAVA/PLAIN_MAVEN?staticAnalysis=true", 2 }, { "JAVA/MAVEN_BLACKBOX", 5 }, { "JAVA/MAVEN_BLACKBOX?staticAnalysis=true", 6 }, { "ASSEMBLER", 4 }, { "C/FACT", 2 }, { "C/GCC", 3 }, { "C/GCC?staticAnalysis=true", 3 }, { "KOTLIN", 1 }, - { "KOTLIN?testCoverage=true", 2 }, { "KOTLIN?sequentialRuns=true", 3 }, { "VHDL", 4 }, { "HASKELL", 1 }, { "HASKELL?sequentialRuns=true", 2 }, { "OCAML", 2 }, - { "SWIFT/PLAIN", 1 }, { "SWIFT/PLAIN?staticAnalysis=true", 2 } }).map(params -> Arguments.of(params[0], params[1])); + { "KOTLIN?sequentialRuns=true", 3 }, { "VHDL", 4 }, { "HASKELL", 1 }, { "HASKELL?sequentialRuns=true", 2 }, { "OCAML", 2 }, { "SWIFT/PLAIN", 1 }, + { "SWIFT/PLAIN?staticAnalysis=true", 2 } }).map(params -> Arguments.of(params[0], params[1])); } @ParameterizedTest @@ -114,7 +113,7 @@ void testValidWindfileDeserializationWithClass() throws JsonProcessingException Windfile windfile = Windfile.deserialize(validWindfile); assertThat(windfile).isNotNull(); - assertThat(windfile.getActions().getFirst()).isInstanceOf(ScriptAction.class); + assertThat(windfile.actions().getFirst()).isInstanceOf(ScriptAction.class); } @Test @@ -150,13 +149,13 @@ void testValidWindfileWithInvalidAction() { @Test @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") void testGetNonExistingAeolusTemplateFile() throws Exception { - request.get("/api/aeolus/templates/JAVA/PLAIN_GRADLE?staticAnalysis=true&sequentialRuns=true&testCoverage=true", HttpStatus.NOT_FOUND, String.class); + request.get("/api/aeolus/templates/JAVA/PLAIN_GRADLE?staticAnalysis=true&sequentialRuns=true", HttpStatus.NOT_FOUND, String.class); } void assertWindfileIsCorrect(Windfile windfile, long expectedScriptActions) { - assertThat(windfile.getApi()).isEqualTo("v0.0.1"); - assertThat(windfile.getMetadata().gitCredentials()).isNull(); - assertThat(windfile.getMetadata().docker()).isNotNull(); - assertThat(windfile.getScriptActions().size()).isEqualTo(expectedScriptActions); + assertThat(windfile.api()).isEqualTo("v0.0.1"); + assertThat(windfile.metadata().gitCredentials()).isNull(); + assertThat(windfile.metadata().docker()).isNotNull(); + assertThat(windfile.scriptActions().size()).isEqualTo(expectedScriptActions); } } diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusTest.java index 4010816396cd..816586b6fc60 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/aelous/AeolusTest.java @@ -8,13 +8,13 @@ import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; -import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusRepository; -import de.tum.cit.aet.artemis.programming.service.aeolus.AeolusResult; -import de.tum.cit.aet.artemis.programming.service.aeolus.DockerConfig; -import de.tum.cit.aet.artemis.programming.service.aeolus.PlatformAction; -import de.tum.cit.aet.artemis.programming.service.aeolus.ScriptAction; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; -import de.tum.cit.aet.artemis.programming.service.aeolus.WindfileMetadata; +import de.tum.cit.aet.artemis.programming.dto.aeolus.AeolusRepository; +import de.tum.cit.aet.artemis.programming.dto.aeolus.AeolusResult; +import de.tum.cit.aet.artemis.programming.dto.aeolus.DockerConfig; +import de.tum.cit.aet.artemis.programming.dto.aeolus.PlatformAction; +import de.tum.cit.aet.artemis.programming.dto.aeolus.ScriptAction; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; +import de.tum.cit.aet.artemis.programming.dto.aeolus.WindfileMetadata; class AeolusTest { @@ -25,45 +25,29 @@ void setup() { DockerConfig dockerConfig = new DockerConfig("image", "tag", List.of("host:container"), List.of("--param1", "--param2")); WindfileMetadata metadata = new WindfileMetadata("name", "id", "description", "author", "gitCredentials", dockerConfig, null, null); - windfile = new Windfile(); - windfile.setApi("v0.0.1"); - windfile.setMetadata(metadata); - - ScriptAction scriptAction = new ScriptAction(); - scriptAction.setName("scriptAction"); - scriptAction.setRunAlways(true); - scriptAction.setScript("script"); - scriptAction.setEnvironment(Map.of("key", "value")); - scriptAction.setParameters(Map.of("key", "value")); AeolusResult scriptResult = new AeolusResult("junit", "text.xml", "ignore", "junit", false); - scriptAction.setResults(List.of(scriptResult)); - assertThat(scriptAction.getResults().getFirst().name()).isEqualTo("junit"); - assertThat(scriptAction.getResults().getFirst().path()).isEqualTo("text.xml"); - assertThat(scriptAction.getResults().getFirst().ignore()).isEqualTo("ignore"); - assertThat(scriptAction.getResults().getFirst().type()).isEqualTo("junit"); - assertThat(scriptAction.getResults().getFirst().before()).isEqualTo(false); - - PlatformAction platformAction = new PlatformAction(); - platformAction.setName("platformAction"); - platformAction.setWorkdir("workdir"); - platformAction.setRunAlways(true); - platformAction.setPlatform("jenkins"); - platformAction.setPlatform("jenkins"); - platformAction.setKind("junit"); + ScriptAction scriptAction = new ScriptAction("scriptAction", Map.of("key", "value"), Map.of("key", "value"), List.of(scriptResult), null, true, null, "script"); + assertThat(scriptAction.results().getFirst().name()).isEqualTo("junit"); + assertThat(scriptAction.results().getFirst().path()).isEqualTo("text.xml"); + assertThat(scriptAction.results().getFirst().ignore()).isEqualTo("ignore"); + assertThat(scriptAction.results().getFirst().type()).isEqualTo("junit"); + assertThat(scriptAction.results().getFirst().before()).isEqualTo(false); + AeolusResult result = new AeolusResult("name", "path", "ignore", "type", true); - platformAction.setResults(List.of(result)); - assertThat(platformAction.getResults().getFirst().name()).isEqualTo("name"); - assertThat(platformAction.getResults().getFirst().path()).isEqualTo("path"); - assertThat(platformAction.getResults().getFirst().ignore()).isEqualTo("ignore"); - assertThat(platformAction.getResults().getFirst().type()).isEqualTo("type"); - assertThat(platformAction.getResults().getFirst().before()).isEqualTo(true); - - windfile.setActions(List.of(scriptAction, platformAction)); + PlatformAction platformAction = new PlatformAction("platformAction", Map.of("key", "value"), Map.of("key", "value"), List.of(result), "workdir", true, "jenkins", "junit", + "type"); + assertThat(platformAction.results().getFirst().name()).isEqualTo("name"); + assertThat(platformAction.results().getFirst().path()).isEqualTo("path"); + assertThat(platformAction.results().getFirst().ignore()).isEqualTo("ignore"); + assertThat(platformAction.results().getFirst().type()).isEqualTo("type"); + assertThat(platformAction.results().getFirst().before()).isEqualTo(true); + + windfile = new Windfile("v0.0.1", metadata, List.of(scriptAction, platformAction), null); } @Test void testGetResults() { - var results = windfile.getResults(); + var results = windfile.results(); assertThat(results.size()).isEqualTo(2); assertThat(results.getFirst().name()).isEqualTo("junit"); assertThat(results.getFirst().path()).isEqualTo("text.xml"); @@ -79,43 +63,25 @@ void testGetResults() { @Test void testWindfileGetterAndSetter() { - assertThat(windfile.getApi()).isEqualTo("v0.0.1"); - assertThat(windfile.getMetadata().author()).isEqualTo("author"); - assertThat(windfile.getMetadata().id()).isEqualTo("id"); - assertThat(windfile.getMetadata().description()).isEqualTo("description"); + assertThat(windfile.api()).isEqualTo("v0.0.1"); + assertThat(windfile.metadata().author()).isEqualTo("author"); + assertThat(windfile.metadata().id()).isEqualTo("id"); + assertThat(windfile.metadata().description()).isEqualTo("description"); DockerConfig dockerConfig = new DockerConfig("image", "tag", List.of("host:container"), List.of("--param1", "--param2")); - assertThat(windfile.getMetadata().docker()).isEqualTo(dockerConfig); - assertThat(windfile.getMetadata().name()).isEqualTo("name"); - assertThat(windfile.getMetadata().gitCredentials()).isEqualTo("gitCredentials"); - assertThat(windfile.getActions().getFirst().getName()).isEqualTo("scriptAction"); - assertThat(windfile.getActions().getFirst().isRunAlways()).isEqualTo(true); - ScriptAction scriptAction = (ScriptAction) windfile.getActions().getFirst(); - assertThat(scriptAction.getScript()).isEqualTo("script"); - assertThat(scriptAction.getEnvironment()).isEqualTo(Map.of("key", "value")); - assertThat(scriptAction.getParameters()).isEqualTo(Map.of("key", "value")); - PlatformAction platformAction = (PlatformAction) windfile.getActions().get(1); - platformAction.setKind("junit"); - assertThat(platformAction.getKind()).isEqualTo("junit"); - platformAction.setType("type"); - assertThat(platformAction.getType()).isEqualTo("type"); - assertThat(platformAction.getWorkdir()).isEqualTo("workdir"); - assertThat(platformAction.getName()).isEqualTo("platformAction"); - assertThat(platformAction.isRunAlways()).isEqualTo(true); - assertThat(platformAction.getPlatform()).isEqualTo("jenkins"); - } - - @Test - void testSettersWithoutMetadata() { - windfile.setMetadata(null); - AeolusRepository aeolusRepository = new AeolusRepository("url", "branch", "path"); - windfile.setPreProcessingMetadata("id", "name", "gitCredentials", "resultHook", "description", Map.of("key", aeolusRepository), "resultHookCredentials"); - assertThat(windfile.getMetadata().id()).isEqualTo("id"); - assertThat(windfile.getMetadata().description()).isEqualTo("description"); - assertThat(windfile.getMetadata().name()).isEqualTo("name"); - assertThat(windfile.getRepositories().get("key")).isEqualTo(aeolusRepository); - assertThat(windfile.getMetadata().gitCredentials()).isEqualTo("gitCredentials"); - assertThat(windfile.getMetadata().resultHook()).isEqualTo("resultHook"); - assertThat(windfile.getMetadata().resultHookCredentials()).isEqualTo("resultHookCredentials"); + assertThat(windfile.metadata().docker()).isEqualTo(dockerConfig); + assertThat(windfile.metadata().name()).isEqualTo("name"); + assertThat(windfile.metadata().gitCredentials()).isEqualTo("gitCredentials"); + assertThat(windfile.actions().getFirst().name()).isEqualTo("scriptAction"); + assertThat(windfile.actions().getFirst().runAlways()).isEqualTo(true); + ScriptAction scriptAction = (ScriptAction) windfile.actions().getFirst(); + assertThat(scriptAction.script()).isEqualTo("script"); + assertThat(scriptAction.environment()).isEqualTo(Map.of("key", "value")); + assertThat(scriptAction.parameters()).isEqualTo(Map.of("key", "value")); + PlatformAction platformAction = (PlatformAction) windfile.actions().get(1); + assertThat(platformAction.workdir()).isEqualTo("workdir"); + assertThat(platformAction.name()).isEqualTo("platformAction"); + assertThat(platformAction.runAlways()).isEqualTo(true); + assertThat(platformAction.platform()).isEqualTo("jenkins"); } @Test diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/CodeHintIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/CodeHintIntegrationTest.java deleted file mode 100644 index a49fe18c881c..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/CodeHintIntegrationTest.java +++ /dev/null @@ -1,147 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.HashSet; -import java.util.Map; -import java.util.Set; -import java.util.stream.Collectors; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.http.HttpStatus; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationIndependentTest; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; - -class CodeHintIntegrationTest extends AbstractProgrammingIntegrationIndependentTest { - - private static final String TEST_PREFIX = "codehint"; - - private ProgrammingExercise exercise; - - private CodeHint codeHint; - - private ProgrammingExerciseSolutionEntry solutionEntry; - - @BeforeEach - void initTestCase() { - userUtilService.addUsers(TEST_PREFIX, 1, 1, 1, 1); - - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExerciseAndTestCases(); - exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void generateCodeHintsForAnExerciseAsAStudent() throws Exception { - request.postListWithResponseBody("/api/programming-exercises/" + exercise.getId() + "/code-hints", null, CodeHint.class, HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void generateCodeHintsForAnExerciseAsATutor() throws Exception { - request.postListWithResponseBody("/api/programming-exercises/" + exercise.getId() + "/code-hints", null, CodeHint.class, HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void generateCodeHintsForAnExerciseAsAnEditor() throws Exception { - request.postListWithResponseBody("/api/programming-exercises/" + exercise.getId() + "/code-hints", null, CodeHint.class, HttpStatus.OK); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void generateCodeHintsForAnExerciseAsAnInstructor() throws Exception { - request.postListWithResponseBody("/api/programming-exercises/" + exercise.getId() + "/code-hints", null, CodeHint.class, HttpStatus.OK); - } - - private void addCodeHints() { - exercise = programmingExerciseUtilService.loadProgrammingExerciseWithEagerReferences(exercise); - programmingExerciseUtilService.addHintsToExercise(exercise); - programmingExerciseUtilService.addTasksToProgrammingExercise(exercise); - programmingExerciseUtilService.addSolutionEntriesToProgrammingExercise(exercise); - programmingExerciseUtilService.addCodeHintsToProgrammingExercise(exercise); - Set hints = codeHintRepository.findByExerciseId(exercise.getId()); - codeHint = hints.stream().filter(hint -> "Task for test1".equals(hint.getProgrammingExerciseTask().getTaskName())).findFirst().orElseThrow(); - codeHint = codeHintRepository.findByIdWithSolutionEntriesElseThrow(codeHint.getId()); - solutionEntry = codeHint.getSolutionEntries().stream().findFirst().orElseThrow(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void removeSolutionEntryFromCodeHintAsAStudent() throws Exception { - addCodeHints(); - request.delete("/api/programming-exercises/" + exercise.getId() + "/code-hints/" + codeHint.getId() + "/solution-entries/" + solutionEntry.getId(), HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void removeSolutionEntryFromCodeHintAsATutor() throws Exception { - addCodeHints(); - request.delete("/api/programming-exercises/" + exercise.getId() + "/code-hints/" + codeHint.getId() + "/solution-entries/" + solutionEntry.getId(), HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void removeSolutionEntryFromCodeHintAsAnEditor() throws Exception { - addCodeHints(); - request.delete("/api/programming-exercises/" + exercise.getId() + "/code-hints/" + codeHint.getId() + "/solution-entries/" + solutionEntry.getId(), HttpStatus.NO_CONTENT); - assertThat(codeHintRepository.findByIdWithSolutionEntriesElseThrow(codeHint.getId()).getSolutionEntries()).isEmpty(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void removeSolutionEntryFromCodeHintAsAnInstructor() throws Exception { - addCodeHints(); - request.delete("/api/programming-exercises/" + exercise.getId() + "/code-hints/" + codeHint.getId() + "/solution-entries/" + solutionEntry.getId(), HttpStatus.NO_CONTENT); - assertThat(codeHintRepository.findByIdWithSolutionEntriesElseThrow(codeHint.getId()).getSolutionEntries()).isEmpty(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateSolutionEntriesOnSaving() throws Exception { - addCodeHints(); - var solutionEntries = codeHint.getSolutionEntries().stream().toList(); - - Map testCases = testCaseRepository.findByExerciseId(exercise.getId()).stream() - .collect(Collectors.toMap(ProgrammingExerciseTestCase::getTestName, test -> test)); - - var changedEntry = solutionEntries.getFirst(); - changedEntry.setLine(100); - changedEntry.setPreviousLine(200); - changedEntry.setCode("Changed code"); - changedEntry.setPreviousCode("Changed previous code"); - changedEntry.setTestCase(testCases.get("test3")); - - var newEntry = new ProgrammingExerciseSolutionEntry(); - newEntry.setLine(200); - newEntry.setPreviousLine(300); - newEntry.setCode("New code"); - newEntry.setPreviousCode("New previous code"); - var testCase = testCases.get("test1"); - newEntry.setTestCase(testCase); - var savedNewEntry = programmingExerciseSolutionEntryRepository.save(newEntry); - savedNewEntry.setTestCase(testCase); - codeHint.setSolutionEntries(new HashSet<>(Set.of(changedEntry, savedNewEntry))); - - request.put("/api/programming-exercises/" + exercise.getId() + "/exercise-hints/" + codeHint.getId(), codeHint, HttpStatus.OK); - - var updatedHint = codeHintRepository.findByIdWithSolutionEntriesElseThrow(codeHint.getId()); - assertThat(updatedHint.getSolutionEntries()).containsExactlyInAnyOrder(changedEntry, savedNewEntry); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGetAllCodeHints() throws Exception { - addCodeHints(); - - var actualCodeHints = request.getList("/api/programming-exercises/" + exercise.getId() + "/code-hints", HttpStatus.OK, CodeHint.class); - assertThat(actualCodeHints).hasSize(3); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/CodeHintServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/CodeHintServiceTest.java deleted file mode 100644 index 05a72ac9a78d..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/CodeHintServiceTest.java +++ /dev/null @@ -1,249 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.assertj.core.api.Assertions.assertThatExceptionOfType; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashSet; -import java.util.List; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.assessment.domain.Visibility; -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.core.exception.BadRequestAlertException; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationIndependentTest; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTestCaseType; - -@SuppressWarnings("ArraysAsListWithZeroOrOneArgument") -class CodeHintServiceTest extends AbstractProgrammingIntegrationIndependentTest { - - private static final String TEST_PREFIX = "codehintservice"; - - private ProgrammingExercise exercise; - - @BeforeEach - void initTestCase() { - userUtilService.addUsers(TEST_PREFIX, 0, 0, 0, 1); - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(); - exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - } - - private ProgrammingExerciseTestCase addTestCaseToExercise(String name) { - var testCase = new ProgrammingExerciseTestCase(); - testCase.setTestName(name); - testCase.setExercise(exercise); - testCase.setVisibility(Visibility.ALWAYS); - testCase.setActive(true); - testCase.setWeight(1D); - testCase.setType(ProgrammingExerciseTestCaseType.BEHAVIORAL); - return testCaseRepository.save(testCase); - } - - private ProgrammingExerciseSolutionEntry addSolutionEntryToTestCase(ProgrammingExerciseTestCase testCase) { - var solutionEntry = new ProgrammingExerciseSolutionEntry(); - solutionEntry.setTestCase(testCase); - solutionEntry.setLine(1); - solutionEntry.setCode("code"); - return programmingExerciseSolutionEntryRepository.save(solutionEntry); - } - - private ProgrammingExerciseTask addTaskToExercise(String name, List testCases) { - var task = new ProgrammingExerciseTask(); - task.setExercise(exercise); - task.setTaskName(name); - task = taskRepository.save(task); - for (int i = 0; i < testCases.size(); i++) { - ProgrammingExerciseTestCase testCase = testCases.get(i); - testCase.getTasks().add(task); - testCases.set(i, testCaseRepository.save(testCase)); - } - task.setTestCases(new HashSet<>(testCases)); - return task; - } - - private CodeHint addCodeHintToTask(String name, ProgrammingExerciseTask task, Set solutionEntries) { - var codeHint = new CodeHint(); - codeHint.setTitle(name); - codeHint.setProgrammingExerciseTask(task); - codeHint.setExercise(exercise); - codeHint.setSolutionEntries(solutionEntries); - - solutionEntries.forEach(entry -> entry.setCodeHint(codeHint)); - var createdHint = codeHintRepository.save(codeHint); - programmingExerciseSolutionEntryRepository.saveAll(solutionEntries); - return createdHint; - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationWithNoSolutionEntry() { - var testCase = addTestCaseToExercise("TestCase1"); - addTaskToExercise("Task1", Arrays.asList(testCase)); - - var codeHints = codeHintService.generateCodeHintsForExercise(exercise, true); - assertThat(codeHints).isEmpty(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationWithOneSolutionEntry() { - var testCase = addTestCaseToExercise("TestCase1"); - var solutionEntry = addSolutionEntryToTestCase(testCase); - var task = addTaskToExercise("Task1", Arrays.asList(testCase)); - - var codeHints = codeHintService.generateCodeHintsForExercise(exercise, true); - assertThat(codeHints).hasSize(1); - assertThat(codeHints.getFirst().getProgrammingExerciseTask()).isEqualTo(task); - assertThat(codeHints.getFirst().getSolutionEntries()).containsExactly(solutionEntry); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationTwiceShouldDeleteOldHint() { - var testCase = addTestCaseToExercise("TestCase1"); - var solutionEntry = addSolutionEntryToTestCase(testCase); - var task = addTaskToExercise("Task1", Arrays.asList(testCase)); - - var codeHints = codeHintService.generateCodeHintsForExercise(exercise, true); - assertThat(codeHints).hasSize(1); - var codeHint = codeHints.getFirst(); - - codeHints = codeHintService.generateCodeHintsForExercise(exercise, true); - assertThat(codeHints).hasSize(1); - assertThat(codeHints.getFirst()).isNotEqualTo(codeHint); - assertThat(codeHints.getFirst().getProgrammingExerciseTask()).isEqualTo(task); - assertThat(codeHints.getFirst().getSolutionEntries()).containsExactly(solutionEntry); - - final Set codeHintsAfterSaving = codeHintRepository.findByExerciseId(exercise.getId()); - assertThat(codeHintsAfterSaving).hasSize(1); - assertThat(codeHintsAfterSaving.stream().findAny().orElseThrow()).isNotEqualTo(codeHint).isEqualTo(codeHints.getFirst()); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationTwiceShouldNotDeleteOldHint() { - var testCase = addTestCaseToExercise("TestCase1"); - var solutionEntry = addSolutionEntryToTestCase(testCase); - var task = addTaskToExercise("Task1", Arrays.asList(testCase)); - - var codeHints = codeHintService.generateCodeHintsForExercise(exercise, false); - assertThat(codeHints).hasSize(1); - var codeHint = codeHints.getFirst(); - - codeHints = codeHintService.generateCodeHintsForExercise(exercise, false); - assertThat(codeHints).hasSize(1); - assertThat(codeHints.getFirst()).isNotEqualTo(codeHint); - assertThat(codeHints.getFirst().getProgrammingExerciseTask()).isEqualTo(task); - assertThat(codeHints.getFirst().getSolutionEntries()).containsExactly(solutionEntry); - assertThat(codeHintRepository.findByExerciseId(exercise.getId())).containsExactlyInAnyOrder(codeHint, codeHints.getFirst()); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testUpdateTestCaseOfSolutionEntry() { - var testCase1 = addTestCaseToExercise("testCase1"); - var testCase2 = addTestCaseToExercise("testCase2"); - var entry = addSolutionEntryToTestCase(testCase1); - var task = addTaskToExercise("task", new ArrayList<>(List.of(testCase1, testCase2))); - var codeHint = addCodeHintToTask("codeHint1", task, new HashSet<>(Set.of(entry))); - - var entryToUpdate = codeHint.getSolutionEntries().stream().findFirst().orElseThrow(); - entryToUpdate.setTestCase(testCase2); - codeHintService.updateSolutionEntriesForCodeHint(codeHint); - - var allEntries = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(exercise.getId()); - assertThat(allEntries).hasSize(1); - assertThat(allEntries.stream().findAny().orElseThrow().getTestCase().getId()).isEqualTo(testCase2.getId()); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testUpdatedContentOfSolutionEntry() { - var testCase1 = addTestCaseToExercise("testCase"); - var entry = addSolutionEntryToTestCase(testCase1); - var task = addTaskToExercise("task", new ArrayList<>(List.of(testCase1))); - var codeHint = addCodeHintToTask("codeHint", task, new HashSet<>(Set.of(entry))); - - var entryToUpdate = codeHint.getSolutionEntries().stream().findFirst().orElseThrow(); - entryToUpdate.setLine(120); - entry.setPreviousLine(130); - entryToUpdate.setCode("Updated code"); - entry.setPreviousCode("Updated previous code"); - entry.setFilePath("Updated file path"); - codeHintService.updateSolutionEntriesForCodeHint(codeHint); - - var allEntries = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(exercise.getId()); - assertThat(allEntries).hasSize(1); - assertThat(allEntries.stream().findAny().orElseThrow()).isEqualTo(entryToUpdate); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testSaveWithNewSolutionEntry() { - // the entry has been created and persisted, but not assigned to the hint yet - var testCase = addTestCaseToExercise("testCase"); - var manuallyCreatedEntry = addSolutionEntryToTestCase(testCase); - var task = addTaskToExercise("task", new ArrayList<>(List.of(testCase))); - var codeHint = addCodeHintToTask("codeHint", task, new HashSet<>(Collections.emptySet())); - - codeHint.setSolutionEntries(new HashSet<>(Set.of(manuallyCreatedEntry))); - codeHintService.updateSolutionEntriesForCodeHint(codeHint); - - var allEntries = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(exercise.getId()); - assertThat(allEntries).containsExactly(manuallyCreatedEntry); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testSaveWithRemovedSolutionEntry() { - // the entry has been created and persisted, but not assigned to the hint yet - var testCase = addTestCaseToExercise("testCase"); - var entryToRemove = addSolutionEntryToTestCase(testCase); - var task = addTaskToExercise("task", new ArrayList<>(List.of(testCase))); - var codeHint = addCodeHintToTask("codeHint", task, new HashSet<>(Set.of(entryToRemove))); - - codeHint.setSolutionEntries(new HashSet<>(Collections.emptySet())); - codeHintService.updateSolutionEntriesForCodeHint(codeHint); - - var entriesForHint = programmingExerciseSolutionEntryRepository.findByCodeHintId(codeHint.getId()); - assertThat(entriesForHint).isEmpty(); - - var allEntries = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(exercise.getId()); - assertThat(allEntries).containsExactly(entryToRemove); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void testSaveEntryWithTestCaseUnrelatedToHintTask() { - // the test case of an entry belongs to a task unequal to the task of the hint that is updated - var unrelatedTestCase = addTestCaseToExercise("unrelatedTaskTestCase"); - addTaskToExercise("unrelatedTask", new ArrayList<>(List.of(unrelatedTestCase))); - var invalidSolutionEntry = new ProgrammingExerciseSolutionEntry(); - invalidSolutionEntry.setTestCase(unrelatedTestCase); - invalidSolutionEntry.setCode("abc"); - - var relatedTestCase = addTestCaseToExercise("relatedTaskTestCase"); - var relatedTask = addTaskToExercise("relatedTask", new ArrayList<>(List.of(relatedTestCase))); - var codeHint = addCodeHintToTask("codeHint", relatedTask, new HashSet<>(Collections.emptySet())); - - codeHint.setSolutionEntries(new HashSet<>(Set.of(invalidSolutionEntry))); - assertThatExceptionOfType(BadRequestAlertException.class).isThrownBy(() -> codeHintService.updateSolutionEntriesForCodeHint(codeHint)); - - var entriesForHint = programmingExerciseSolutionEntryRepository.findByCodeHintId(codeHint.getId()); - assertThat(entriesForHint).isEmpty(); - - var allEntries = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(exercise.getId()); - assertThat(allEntries).isEmpty(); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ExerciseHintIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ExerciseHintIntegrationTest.java deleted file mode 100644 index 6a44f54820fe..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ExerciseHintIntegrationTest.java +++ /dev/null @@ -1,457 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.time.ZonedDateTime; -import java.util.ArrayList; -import java.util.Collection; -import java.util.List; -import java.util.Optional; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.http.HttpStatus; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.assessment.domain.AssessmentType; -import de.tum.cit.aet.artemis.assessment.domain.Feedback; -import de.tum.cit.aet.artemis.assessment.domain.FeedbackType; -import de.tum.cit.aet.artemis.assessment.domain.Result; -import de.tum.cit.aet.artemis.assessment.domain.Visibility; -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationIndependentTest; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseStudentParticipation; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHintActivation; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; - -class ExerciseHintIntegrationTest extends AbstractProgrammingIntegrationIndependentTest { - - private static final String TEST_PREFIX = "exercisehintintegration"; - - private ProgrammingExercise exercise; - - private ProgrammingExercise exerciseLite; - - private List hints; - - private ExerciseHint exerciseHint; - - private ProgrammingExerciseStudentParticipation studentParticipation; - - private int timeOffset = 0; - - @BeforeEach - void initTestCase() { - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExerciseAndTestCases(); - final ProgrammingExercise programmingExercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - - userUtilService.addUsers(TEST_PREFIX, 2, 2, 1, 2); - - testCaseRepository.saveAll(testCaseRepository.findByExerciseId(programmingExercise.getId()).stream().peek(testCase -> testCase.setActive(true)).toList()); - exerciseLite = programmingExerciseRepository.findByIdElseThrow(programmingExercise.getId()); - exercise = programmingExerciseUtilService.loadProgrammingExerciseWithEagerReferences(exerciseLite); - programmingExerciseUtilService.addHintsToExercise(exercise); - programmingExerciseUtilService.addTasksToProgrammingExercise(exercise); - - List sortedTasks = programmingExerciseTaskService.getSortedTasks(exercise); - - hints = new ArrayList<>(exerciseHintRepository.findByExerciseId(exerciseLite.getId())); - exerciseHint = hints.getFirst(); - exerciseHint.setProgrammingExerciseTask(sortedTasks.getFirst()); - hints.get(1).setProgrammingExerciseTask(sortedTasks.get(1)); - hints.get(2).setProgrammingExerciseTask(sortedTasks.get(2)); - exerciseHintRepository.saveAll(hints); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void queryAllAvailableHintsForAnExercise() throws Exception { - var user = userTestRepository.getUserWithGroupsAndAuthorities(); - studentParticipation = participationUtilService.addStudentParticipationForProgrammingExercise(exercise, user.getLogin()); - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - - var availableHints = request.getList("/api/programming-exercises/" + exercise.getId() + "/exercise-hints/available", HttpStatus.OK, ExerciseHint.class); - assertThat(availableHints).hasSize(1); - assertThat(availableHints.getFirst().getContent()).isNullOrEmpty(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void queryAllActivatedHintsForAnExercise() throws Exception { - var user = userTestRepository.getUserWithGroupsAndAuthorities(); - var ueha = new ExerciseHintActivation(); - ueha.setExerciseHint(exerciseHint); - ueha.setUser(user); - ueha.setActivationDate(ZonedDateTime.now()); - ueha.setRating(4); - exerciseHintActivationRepository.save(ueha); - - var availableHints = request.getList("/api/programming-exercises/" + exercise.getId() + "/exercise-hints/activated", HttpStatus.OK, ExerciseHint.class); - assertThat(availableHints).hasSize(1); - assertThat(availableHints.getFirst().getId()).isEqualTo(exerciseHint.getId()); - assertThat(availableHints.getFirst().getContent()).isEqualTo(exerciseHint.getContent()); - assertThat(availableHints.getFirst().getCurrentUserRating()).isEqualTo(4); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void activateHintForAnExercise() throws Exception { - var user = userTestRepository.getUserWithGroupsAndAuthorities(); - studentParticipation = participationUtilService.addStudentParticipationForProgrammingExercise(exercise, user.getLogin()); - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - - var activatedHint = request.postWithResponseBody("/api/programming-exercises/" + exercise.getId() + "/exercise-hints/" + exerciseHint.getId() + "/activate", null, - ExerciseHint.class, HttpStatus.OK); - assertThat(activatedHint.getId()).isEqualTo(exerciseHint.getId()); - assertThat(activatedHint.getContent()).isEqualTo(exerciseHint.getContent()); - - var uehas = exerciseHintActivationRepository.findByExerciseAndUserWithExerciseHintRelations(exercise.getId(), user.getId()); - assertThat(uehas).hasSize(1); - assertThat(uehas.stream().findFirst().get().getExerciseHint().getId()).isEqualTo(exerciseHint.getId()); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void rateActivatedHintForAnExercise() throws Exception { - var user = userTestRepository.getUserWithGroupsAndAuthorities(); - var ueha = new ExerciseHintActivation(); - ueha.setExerciseHint(exerciseHint); - ueha.setUser(user); - ueha.setActivationDate(ZonedDateTime.now()); - exerciseHintActivationRepository.save(ueha); - - request.postWithoutLocation("/api/programming-exercises/" + exercise.getId() + "/exercise-hints/" + exerciseHint.getId() + "/rating/" + 4, null, HttpStatus.OK, null); - - ueha = exerciseHintActivationRepository.findById(ueha.getId()).orElseThrow(); - assertThat(ueha.getRating()).isEqualTo(4); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void rateActivatedHintForAnExerciseBadRequest() throws Exception { - var user = userTestRepository.getUserWithGroupsAndAuthorities(); - var ueha = new ExerciseHintActivation(); - ueha.setExerciseHint(exerciseHint); - ueha.setUser(user); - ueha.setActivationDate(ZonedDateTime.now()); - exerciseHintActivationRepository.save(ueha); - - request.postWithoutLocation("/api/programming-exercises/" + exercise.getId() + "/exercise-hints/" + exerciseHint.getId() + "/rating/" + 100, null, HttpStatus.BAD_REQUEST, - null); - - ueha = exerciseHintActivationRepository.findById(ueha.getId()).orElseThrow(); - assertThat(ueha.getRating()).isNull(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void rateNotActivatedHintForAnExerciseForbidden() throws Exception { - long sizeBefore = exerciseHintActivationRepository.count(); - request.postWithoutLocation("/api/programming-exercises/" + exercise.getId() + "/exercise-hints/" + exerciseHint.getId() + "/rating/" + 4, null, HttpStatus.NOT_FOUND, - null); - assertThat(exerciseHintActivationRepository.count()).isEqualTo(sizeBefore); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void queryAllHintsForAnExerciseAsAStudent() throws Exception { - request.getList("/api/programming-exercises/" + exercise.getId() + "/exercise-hints", HttpStatus.FORBIDDEN, ExerciseHint.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void queryAllHintsForAnExerciseAsATutor() throws Exception { - request.getList("/api/programming-exercises/" + exercise.getId() + "/exercise-hints", HttpStatus.OK, ExerciseHint.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void queryAllHintsForAnExerciseAsAnInstructor() throws Exception { - request.getList("/api/programming-exercises/" + exercise.getId() + "/exercise-hints", HttpStatus.OK, ExerciseHint.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void getHintForAnExerciseAsStudentShouldReturnForbidden() throws Exception { - request.get("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + exerciseHint.getId(), HttpStatus.FORBIDDEN, ExerciseHint.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void getHintForAnExerciseAsTutor() throws Exception { - request.get("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + exerciseHint.getId(), HttpStatus.OK, ExerciseHint.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void getHintForAnExerciseAsEditor() throws Exception { - request.get("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + exerciseHint.getId(), HttpStatus.OK, ExerciseHint.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void getHintForAnExerciseAsAnInstructor() throws Exception { - request.get("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + exerciseHint.getId(), HttpStatus.OK, ExerciseHint.class); - request.get("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + 0L, HttpStatus.NOT_FOUND, ExerciseHint.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void createHintAsAStudentShouldReturnForbidden() throws Exception { - ExerciseHint exerciseHint = new ExerciseHint().content("content 4").title("title 4").exercise(exerciseLite); - request.post("/api/programming-exercises/" + exercise.getId() + "/exercise-hints", exerciseHint, HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void createHintAsTutorForbidden() throws Exception { - ExerciseHint exerciseHint = new ExerciseHint().content("content 4").title("title 4").exercise(exerciseLite); - request.post("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints", exerciseHint, HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void createHintAsEditor() throws Exception { - ExerciseHint exerciseHint = new ExerciseHint().content("content 4").title("title 4").exercise(exerciseLite); - request.post("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints", exerciseHint, HttpStatus.CREATED); - - Set exerciseHints = exerciseHintRepository.findByExerciseId(exerciseLite.getId()); - assertThat(exerciseHints).hasSize(4); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void createHintAsInstructor() throws Exception { - ExerciseHint exerciseHint = new ExerciseHint().content("content 4").title("title 4").exercise(exerciseLite); - - request.post("/api/programming-exercises/" + exercise.getId() + "/exercise-hints", exerciseHint, HttpStatus.CREATED); - Set exerciseHints = exerciseHintRepository.findByExerciseId(exerciseLite.getId()); - assertThat(exerciseHints).hasSize(4); - - exerciseHint.setExercise(null); - request.post("/api/programming-exercises/" + exercise.getId() + "/exercise-hints", exerciseHint, HttpStatus.CONFLICT); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void createCodeHintShouldFail() throws Exception { - CodeHint codeHint = new CodeHint(); - codeHint.setTitle("Hint 1"); - codeHint.setExercise(exercise); - - long sizeBefore = exerciseHintRepository.count(); - request.post("/api/programming-exercises/" + codeHint.getExercise().getId() + "/exercise-hints", codeHint, HttpStatus.BAD_REQUEST); - long sizeAfter = exerciseHintRepository.count(); - assertThat(sizeAfter).isEqualTo(sizeBefore); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void updateHintAsAStudentShouldReturnForbidden() throws Exception { - updateHintForbidden(); - } - - private void updateHintForbidden() throws Exception { - String newContent = "new content value!"; - String contentBefore = exerciseHint.getContent(); - exerciseHint.setContent(newContent); - request.put("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + exerciseHint.getId(), exerciseHint, HttpStatus.FORBIDDEN); - - Optional hintAfterSave = exerciseHintRepository.findById(exerciseHint.getId()); - assertThat(hintAfterSave).isPresent(); - assertThat(hintAfterSave.get()).isInstanceOf(ExerciseHint.class); - assertThat((hintAfterSave.get()).getContent()).isEqualTo(contentBefore); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void updateHintAsTutorForbidden() throws Exception { - updateHintForbidden(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void updateHintAsEditor() throws Exception { - String newContent = "new content value!"; - exerciseHint.setContent(newContent); - request.put("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + exerciseHint.getId(), exerciseHint, HttpStatus.OK); - Optional hintAfterSave = exerciseHintRepository.findById(exerciseHint.getId()); - assertThat(hintAfterSave).isPresent(); - assertThat(hintAfterSave.get()).isInstanceOf(ExerciseHint.class); - assertThat((hintAfterSave.get()).getContent()).isEqualTo(newContent); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateHintAsInstructor() throws Exception { - String newContent = "new content value!"; - - exerciseHint.setContent(newContent); - request.put("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + exerciseHint.getId(), exerciseHint, HttpStatus.OK); - - Optional hintAfterSave = exerciseHintRepository.findById(exerciseHint.getId()); - assertThat(hintAfterSave).isPresent(); - assertThat(hintAfterSave.get()).isInstanceOf(ExerciseHint.class); - assertThat((hintAfterSave.get()).getContent()).isEqualTo(newContent); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateCodeHintTitle() throws Exception { - CodeHint codeHint = new CodeHint(); - codeHint.setTitle("Hint 1"); - codeHint.setExercise(exerciseLite); - codeHint.setProgrammingExerciseTask(programmingExerciseTaskService.getSortedTasks(exercise).getFirst()); - - exerciseHintRepository.save(codeHint); - - codeHint.setTitle("New Title"); - - request.put("/api/programming-exercises/" + codeHint.getExercise().getId() + "/exercise-hints/" + codeHint.getId(), codeHint, HttpStatus.OK); - Optional hintAfterSave = exerciseHintRepository.findById(codeHint.getId()); - assertThat(hintAfterSave).isPresent(); - assertThat(hintAfterSave.get()).isInstanceOf(CodeHint.class); - assertThat((hintAfterSave.get()).getTitle()).isEqualTo("New Title"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void deleteHintAsInstructor() throws Exception { - final ExerciseHint exerciseHint = new ExerciseHint().content("content 4").title("title 4").exercise(exerciseLite); - request.delete("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + 0L, HttpStatus.NOT_FOUND); - request.post("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints", exerciseHint, HttpStatus.CREATED); - final ExerciseHint exerciseHintAfterCreation = exerciseHintRepository.findByExerciseId(exerciseLite.getId()).stream().findAny().orElseThrow(); - request.delete("/api/programming-exercises/" + exerciseHint.getExercise().getId() + "/exercise-hints/" + exerciseHintAfterCreation.getId(), HttpStatus.NO_CONTENT); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void deleteCodeHintAsInstructor() throws Exception { - CodeHint codeHint = new CodeHint(); - codeHint.setTitle("Hint 1"); - codeHint.setExercise(exercise); - exerciseHintRepository.save(codeHint); - - request.delete("/api/programming-exercises/" + codeHint.getExercise().getId() + "/exercise-hints/" + codeHint.getId(), HttpStatus.NO_CONTENT); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGetHintTitleAsInstructor() throws Exception { - // Only user and role matter, so we can re-use the logic - testGetHintTitle(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testGetHintTitleAsTeachingAssistant() throws Exception { - // Only user and role matter, so we can re-use the logic - testGetHintTitle(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "user1", roles = "USER") - void testGetHintTitleAsUser() throws Exception { - // Only user and role matter, so we can re-use the logic - testGetHintTitle(); - } - - private void testGetHintTitle() throws Exception { - final var hint = new ExerciseHint().title("Test Hint").exercise(exerciseLite); - exerciseHintRepository.save(hint); - - final var title = request.get("/api/programming-exercises/" + hint.getExercise().getId() + "/exercise-hints/" + hint.getId() + "/title", HttpStatus.OK, String.class); - assertThat(title).isEqualTo(hint.getTitle()); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "user1", roles = "USER") - void testGetHintTitleForNonExistingHint() throws Exception { - request.get("/api/programming-exercises/" + exercise.getId() + "/exercise-hints/12312312321/title", HttpStatus.NOT_FOUND, String.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void createHintWithInvalidExerciseIds() throws Exception { - Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(); - var unrelatedExercise = course.getExercises().stream().findFirst().orElseThrow(); - - ExerciseHint exerciseHint = new ExerciseHint(); - exerciseHint.setTitle("Test Title"); - exerciseHint.setExercise(exerciseLite); - - request.post("/api/programming-exercises/" + unrelatedExercise.getId() + "/exercise-hints", exerciseHint, HttpStatus.CONFLICT); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateHintWithInvalidExerciseIds() throws Exception { - Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(); - var unrelatedExercise = course.getExercises().stream().findFirst().orElseThrow(); - - exerciseHint.setTitle("New Title"); - - request.put("/api/programming-exercises/" + unrelatedExercise.getId() + "/exercise-hints/" + exerciseHint.getId(), exerciseHint, HttpStatus.CONFLICT); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void getHintTitleWithInvalidExerciseIds() throws Exception { - Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(); - var unrelatedExercise = course.getExercises().stream().findFirst().orElseThrow(); - - request.get("/api/programming-exercises/" + unrelatedExercise.getId() + "/exercise-hints/" + exerciseHint.getId(), HttpStatus.CONFLICT, String.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void getExerciseHintWithInvalidExerciseIds() throws Exception { - Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(); - var unrelatedExercise = course.getExercises().stream().findFirst().orElseThrow(); - - request.get("/api/programming-exercises/" + unrelatedExercise.getId() + "/exercise-hints/" + exerciseHint.getId(), HttpStatus.CONFLICT, String.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void deleteHintWithInvalidExerciseIds() throws Exception { - Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(); - var unrelatedExercise = course.getExercises().stream().findFirst().orElseThrow(); - - request.delete("/api/programming-exercises/" + unrelatedExercise.getId() + "/exercise-hints/" + exerciseHint.getId(), HttpStatus.CONFLICT); - } - - private void addResultWithFailedTestCases(Collection failedTestCases) { - var successfulTestCases = new ArrayList<>(exercise.getTestCases()); - successfulTestCases.removeAll(failedTestCases); - addResultWithSuccessfulTestCases(successfulTestCases); - } - - private void addResultWithSuccessfulTestCases(Collection successfulTestCases) { - var submission = programmingExerciseUtilService.createProgrammingSubmission(studentParticipation, false); - Result result = new Result().participation(submission.getParticipation()).assessmentType(AssessmentType.AUTOMATIC).score(0D).rated(true) - .completionDate(ZonedDateTime.now().plusSeconds(timeOffset++)); - result = resultRepository.save(result); - result.setSubmission(submission); - submission.addResult(result); - programmingSubmissionRepository.save(submission); - - for (ProgrammingExerciseTestCase testCase : exercise.getTestCases()) { - var feedback = new Feedback(); - feedback.setPositive(successfulTestCases.contains(testCase)); - feedback.setTestCase(testCase); - feedback.setVisibility(Visibility.ALWAYS); - feedback.setType(FeedbackType.AUTOMATIC); - participationUtilService.addFeedbackToResult(feedback, result); - } - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ExerciseHintServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ExerciseHintServiceTest.java deleted file mode 100644 index 99b3d357f73f..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ExerciseHintServiceTest.java +++ /dev/null @@ -1,304 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.time.ZonedDateTime; -import java.util.ArrayList; -import java.util.Collection; -import java.util.List; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.assessment.domain.AssessmentType; -import de.tum.cit.aet.artemis.assessment.domain.Feedback; -import de.tum.cit.aet.artemis.assessment.domain.FeedbackType; -import de.tum.cit.aet.artemis.assessment.domain.Result; -import de.tum.cit.aet.artemis.assessment.domain.Visibility; -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.core.domain.User; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationIndependentTest; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseStudentParticipation; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHintActivation; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; - -class ExerciseHintServiceTest extends AbstractProgrammingIntegrationIndependentTest { - - private static final String TEST_PREFIX = "exercisehintservice"; - - private ProgrammingExercise exercise; - - private List sortedTasks; - - private List hints; - - private ExerciseHint exerciseHint; - - private User student; - - private ProgrammingExerciseStudentParticipation studentParticipation; - - private int timeOffset = 0; - - @BeforeEach - void initTestCase() { - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExerciseAndTestCases(); - final ProgrammingExercise programmingExercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - - userUtilService.addUsers(TEST_PREFIX, 2, 2, 1, 2); - - student = userRepository.getUserWithGroupsAndAuthorities(TEST_PREFIX + "student1"); - userUtilService.changeUser(TEST_PREFIX + "student1"); - - var activatedTestCases = testCaseRepository.findByExerciseId(programmingExercise.getId()).stream().peek(testCase -> testCase.setActive(true)).toList(); - testCaseRepository.saveAll(activatedTestCases); - exercise = programmingExerciseRepository.findByIdElseThrow(programmingExercise.getId()); - exercise = programmingExerciseUtilService.loadProgrammingExerciseWithEagerReferences(exercise); - programmingExerciseUtilService.addHintsToExercise(exercise); - programmingExerciseUtilService.addTasksToProgrammingExercise(exercise); - - sortedTasks = programmingExerciseTaskService.getSortedTasks(exercise); - - hints = new ArrayList<>(exerciseHintRepository.findByExerciseId(exercise.getId())); - exerciseHint = hints.getFirst(); - exerciseHint.setProgrammingExerciseTask(sortedTasks.getFirst()); - hints.get(1).setProgrammingExerciseTask(sortedTasks.get(1)); - hints.get(2).setProgrammingExerciseTask(sortedTasks.get(2)); - exerciseHintRepository.saveAll(hints); - - studentParticipation = participationUtilService.addStudentParticipationForProgrammingExercise(exercise, student.getLogin()); - } - - @Test - void testGetAvailableExerciseHintsTasksWithoutTestCases() { - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - for (ProgrammingExerciseTask sortedTask : sortedTasks) { - sortedTask.getTestCases().clear(); - taskRepository.save(sortedTask); - } - exercise.setProblemStatement(exercise.getProblemStatement().replaceAll("\\([^()]+\\)", "()")); - programmingExerciseRepository.save(exercise); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).isEmpty(); - } - - @Test - void testGetAvailableExerciseHintsEmpty1() { - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).isEmpty(); - } - - @Test - void testGetAvailableExerciseHintsEmpty2() { - addResultWithFailedTestCases(exercise.getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).isEmpty(); - } - - @Test - void testGetAvailableExerciseHintsEmpty3() { - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithSuccessfulTestCases(exercise.getTestCases()); - addResultWithSuccessfulTestCases(exercise.getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).isEmpty(); - } - - @Test - void testGetAvailableExerciseHintsEmpty4() { - addResultWithSuccessfulTestCases(sortedTasks.getFirst().getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.getFirst().getTestCases()); - addResultWithFailedTestCases(sortedTasks.get(2).getTestCases()); - addResultWithFailedTestCases(sortedTasks.get(2).getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).isEmpty(); - } - - @Test - void testGetAvailableExerciseHints1() { - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).containsExactly(exerciseHint); - } - - @Test - void testGetAvailableExerciseHints2() { - addResultWithSuccessfulTestCases(sortedTasks.getFirst().getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.getFirst().getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.getFirst().getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).containsExactly(hints.get(1)); - } - - @Test - void testGetAvailableExerciseHints3() { - addResultWithSuccessfulTestCases(sortedTasks.get(1).getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.get(1).getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.get(1).getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).containsExactly(exerciseHint); - } - - @Test - void testGetAvailableExerciseHints4() { - addResultWithSuccessfulTestCases(sortedTasks.get(2).getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.get(2).getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.get(2).getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).containsExactly(exerciseHint); - } - - @Test - void testGetAvailableExerciseHints5() { - addResultWithSuccessfulTestCases(sortedTasks.get(1).getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.get(2).getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.get(1).getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).containsExactly(exerciseHint); - } - - @Test - void testGetAvailableExerciseHintsWithZeroThreshold1() { - exerciseHint.setDisplayThreshold((short) 0); - exerciseHintRepository.save(exerciseHint); - addResultWithFailedTestCases(exercise.getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).containsExactly(exerciseHint); - } - - @Test - void testGetAvailableExerciseHintsWithZeroThreshold2() { - exerciseHint.setDisplayThreshold((short) 0); - exerciseHintRepository.save(exerciseHint); - addResultWithSuccessfulTestCases(exercise.getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).containsExactly(exerciseHint); - } - - @Test - void testGetAvailableExerciseHintsWithZeroThreshold3() { - exerciseHint.setDisplayThreshold((short) 0); - exerciseHintRepository.save(exerciseHint); - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.getFirst().getTestCases()); - var availableExerciseHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableExerciseHints).containsExactly(exerciseHint); - } - - @Test - void testGetAvailableExerciseHints_skippedTestsConsideredAsNegative() { - // create result with feedbacks with "null" for attribute "positive" - addResultWithSuccessfulTestCases(exercise.getTestCases()); - var results = resultRepository.findAllByParticipationExerciseId(exercise.getId()); - var optionalResult = resultRepository.findWithBidirectionalSubmissionAndFeedbackAndAssessorAndAssessmentNoteAndTeamStudentsById(results.iterator().next().getId()); - assertThat(optionalResult).isPresent(); - - var result = optionalResult.get(); - result.getFeedbacks().forEach(feedback -> feedback.setPositive(null)); - resultRepository.save(result); - - // create results with feedbacks with "false" for attribute "positive" - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - - var availableHints = exerciseHintService.getAvailableExerciseHints(exercise, student); - assertThat(availableHints).containsExactly(exerciseHint); - } - - @Test - void testActivateExerciseHint1() { - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - - assertThat(exerciseHintService.activateHint(hints.getFirst(), student)).isTrue(); - assertThat(exerciseHintService.activateHint(hints.get(1), student)).isFalse(); - assertThat(exerciseHintService.activateHint(hints.get(2), student)).isFalse(); - Set exerciseHintActivations = exerciseHintActivationRepository.findByExerciseAndUserWithExerciseHintRelations(exercise.getId(), student.getId()); - assertThat(exerciseHintActivations).hasSize(1).allMatch(activation -> activation.getExerciseHint().getId().equals(exerciseHint.getId())); - } - - @Test - void testActivateExerciseHint2() { - addResultWithSuccessfulTestCases(sortedTasks.getFirst().getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.getFirst().getTestCases()); - addResultWithSuccessfulTestCases(sortedTasks.getFirst().getTestCases()); - - assertThat(exerciseHintService.activateHint(hints.getFirst(), student)).isFalse(); - assertThat(exerciseHintService.activateHint(hints.get(1), student)).isTrue(); - assertThat(exerciseHintService.activateHint(hints.get(2), student)).isFalse(); - Set exerciseHintActivations = exerciseHintActivationRepository.findByExerciseAndUserWithExerciseHintRelations(exercise.getId(), student.getId()); - assertThat(exerciseHintActivations).hasSize(1).allMatch(activation -> activation.getExerciseHint().getId().equals(hints.get(1).getId())); - } - - @Test - void testActivateExerciseHint3() { - addResultWithFailedTestCases(sortedTasks.get(2).getTestCases()); - addResultWithFailedTestCases(sortedTasks.get(2).getTestCases()); - addResultWithFailedTestCases(sortedTasks.get(2).getTestCases()); - - assertThat(exerciseHintService.activateHint(hints.getFirst(), student)).isFalse(); - assertThat(exerciseHintService.activateHint(hints.get(1), student)).isFalse(); - assertThat(exerciseHintService.activateHint(hints.get(2), student)).isTrue(); - Set exerciseHintActivations = exerciseHintActivationRepository.findByExerciseAndUserWithExerciseHintRelations(exercise.getId(), student.getId()); - assertThat(exerciseHintActivations).hasSize(1).allMatch(activation -> activation.getExerciseHint().getId().equals(hints.get(2).getId())); - } - - @Test - void testActivateExerciseHint4() { - addResultWithSuccessfulTestCases(exercise.getTestCases()); - addResultWithSuccessfulTestCases(exercise.getTestCases()); - addResultWithSuccessfulTestCases(exercise.getTestCases()); - - assertThat(exerciseHintService.activateHint(hints.getFirst(), student)).isFalse(); - assertThat(exerciseHintService.activateHint(hints.get(1), student)).isFalse(); - assertThat(exerciseHintService.activateHint(hints.get(2), student)).isFalse(); - Set exerciseHintActivations = exerciseHintActivationRepository.findByExerciseAndUserWithExerciseHintRelations(exercise.getId(), student.getId()); - assertThat(exerciseHintActivations).isEmpty(); - } - - @Test - void testActivateExerciseHintTwiceFails() { - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - addResultWithFailedTestCases(exercise.getTestCases()); - - assertThat(exerciseHintService.activateHint(exerciseHint, student)).isTrue(); - assertThat(exerciseHintService.activateHint(exerciseHint, student)).isFalse(); - Set exerciseHintActivations = exerciseHintActivationRepository.findByExerciseAndUserWithExerciseHintRelations(exercise.getId(), student.getId()); - assertThat(exerciseHintActivations).hasSize(1).allMatch(activation -> activation.getExerciseHint().getId().equals(exerciseHint.getId())); - } - - private void addResultWithFailedTestCases(Collection failedTestCases) { - var successfulTestCases = new ArrayList<>(exercise.getTestCases()); - successfulTestCases.removeAll(failedTestCases); - addResultWithSuccessfulTestCases(successfulTestCases); - } - - private void addResultWithSuccessfulTestCases(Collection successfulTestCases) { - var submission = programmingExerciseUtilService.createProgrammingSubmission(studentParticipation, false); - Result result = new Result().participation(submission.getParticipation()).assessmentType(AssessmentType.AUTOMATIC).score(0D).rated(true) - .completionDate(ZonedDateTime.now().plusSeconds(timeOffset++)); - result = resultRepository.save(result); - result.setSubmission(submission); - submission.addResult(result); - programmingSubmissionRepository.save(submission); - - for (ProgrammingExerciseTestCase testCase : exercise.getTestCases()) { - var feedback = new Feedback(); - feedback.setPositive(successfulTestCases.contains(testCase)); - feedback.setTestCase(testCase); - feedback.setVisibility(Visibility.ALWAYS); - feedback.setType(FeedbackType.AUTOMATIC); - participationUtilService.addFeedbackToResult(feedback, result); - } - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/HestiaDatabaseTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/HestiaDatabaseTest.java deleted file mode 100644 index 040547ae9cf9..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/HestiaDatabaseTest.java +++ /dev/null @@ -1,165 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.HashSet; -import java.util.List; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationIndependentTest; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; - -/** - * This class tests the database relations of the Hestia domain models. - * This currently includes ProgrammingExerciseTask, ProgrammingExerciseSolutionEntry and CodeHint. - * It tests if the addition and deletion of these models works as expected. - */ -class HestiaDatabaseTest extends AbstractProgrammingIntegrationIndependentTest { - - private static final String TEST_PREFIX = "hestiadatabase"; - - private Long programmingExerciseId; - - @BeforeEach - void init() { - userUtilService.addUsers(TEST_PREFIX, 2, 2, 0, 2); - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(); - programmingExerciseId = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class).getId(); - } - - ProgrammingExerciseTask addTaskToProgrammingExercise(String taskName) { - var task = new ProgrammingExerciseTask(); - task.setTaskName(taskName); - task.setExercise(programmingExerciseRepository.getReferenceById(programmingExerciseId)); - task = taskRepository.save(task); - return task; - } - - ProgrammingExerciseSolutionEntry[] addSolutionEntriesToTestCase(int count, ProgrammingExerciseTestCase testCase) { - var solutionEntries = new ProgrammingExerciseSolutionEntry[count]; - for (int i = 0; i < count; i++) { - var solutionEntry = new ProgrammingExerciseSolutionEntry(); - solutionEntry.setTestCase(testCase); - solutionEntry.setCode("Code block 1"); - solutionEntry.setLine(i); - solutionEntry = programmingExerciseSolutionEntryRepository.save(solutionEntry); - solutionEntries[i] = solutionEntry; - } - return solutionEntries; - } - - @Test - void addOneTaskToProgrammingExercise() { - var task = addTaskToProgrammingExercise("Task 1"); - assertThat(taskRepository.findByExerciseIdWithTestCases(programmingExerciseId)).containsExactly(task); - } - - @Test - void deleteProgrammingExerciseWithTask() { - addOneTaskToProgrammingExercise(); - programmingExerciseRepository.deleteById(programmingExerciseId); - assertThat(taskRepository.findByExerciseId(programmingExerciseId)).isEmpty(); - } - - @Test - void addTestCasesWithSolutionEntriesToProgrammingExercise() { - var programmingExercise = programmingExerciseRepository.findByIdElseThrow(programmingExerciseId); - programmingExerciseUtilService.addTestCasesToProgrammingExercise(programmingExercise); - var testCases = testCaseRepository.findByExerciseId(programmingExerciseId); - assertThat(testCases).isNotEmpty(); - for (ProgrammingExerciseTestCase testCase : testCases) { - var solutionEntries = addSolutionEntriesToTestCase(2, testCase); - assertThat(programmingExerciseSolutionEntryRepository.findByTestCaseId(testCase.getId())).containsExactly(solutionEntries); - } - } - - @Test - void deleteProgrammingExerciseWithTestCasesAndSolutionEntries() { - addTestCasesWithSolutionEntriesToProgrammingExercise(); - programmingExerciseRepository.deleteById(programmingExerciseId); - assertThat(testCaseRepository.findByExerciseId(programmingExerciseId)).isEmpty(); - assertThat(programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(programmingExerciseId)).isEmpty(); - } - - @Test - void deleteTaskWithTestCases() { - var programmingExercise = programmingExerciseRepository.findByIdElseThrow(programmingExerciseId); - programmingExerciseUtilService.addTestCasesToProgrammingExercise(programmingExercise); - var testCases = testCaseRepository.findByExerciseId(programmingExerciseId); - assertThat(testCases).isNotEmpty(); - var task = addTaskToProgrammingExercise("Task 1"); - task.setTestCases(testCases); - task = taskRepository.save(task); - taskRepository.delete(task); - assertThat(testCaseRepository.findByExerciseId(programmingExerciseId)).isEqualTo(testCases); - } - - @Test - void addCodeHintToProgrammingExercise() { - var programmingExercise = programmingExerciseRepository.findByIdElseThrow(programmingExerciseId); - programmingExerciseUtilService.addTestCasesToProgrammingExercise(programmingExercise); - var testCases = testCaseRepository.findByExerciseId(programmingExerciseId); - assertThat(testCases).isNotEmpty(); - var task = addTaskToProgrammingExercise("Task 1"); - task.setTestCases(testCases); - task = taskRepository.save(task); - Set allSolutionEntries = new HashSet<>(); - for (ProgrammingExerciseTestCase testCase : testCases) { - var solutionEntries = addSolutionEntriesToTestCase(2, testCase); - assertThat(programmingExerciseSolutionEntryRepository.findByTestCaseId(testCase.getId())).containsExactly(solutionEntries); - allSolutionEntries.addAll(List.of(solutionEntries)); - } - var codeHint = (CodeHint) new CodeHint(); - codeHint.setProgrammingExerciseTask(task); - codeHint.setExercise(programmingExercise); - codeHint.setTitle("Code Hint 1"); - codeHint = codeHintRepository.save(codeHint); - for (ProgrammingExerciseSolutionEntry solutionEntry : allSolutionEntries) { - solutionEntry.setCodeHint(codeHint); - programmingExerciseSolutionEntryRepository.save(solutionEntry); - } - codeHint.setSolutionEntries(allSolutionEntries); - codeHint = codeHintRepository.save(codeHint); - task.setExerciseHints(Set.of(codeHint)); - taskRepository.save(task); - assertThat(programmingExerciseSolutionEntryRepository.findByCodeHintId(codeHint.getId())).isEqualTo(allSolutionEntries); - assertThat(codeHintRepository.findByExerciseId(programmingExerciseId)).containsExactly(codeHint); - } - - @Test - void deleteCodeHint() { - addCodeHintToProgrammingExercise(); - var codeHint = codeHintRepository.findByExerciseId(programmingExerciseId).stream().findAny().orElseThrow(); - codeHintRepository.delete(codeHint); - assertThat(taskRepository.findByExerciseId(programmingExerciseId)).hasSize(1); - assertThat(programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(programmingExerciseId)).hasSize(6); - } - - @Test - void deleteProgrammingExerciseWithCodeHint() { - addCodeHintToProgrammingExercise(); - programmingExerciseRepository.deleteById(programmingExerciseId); - assertThat(taskRepository.findByExerciseId(programmingExerciseId)).isEmpty(); - assertThat(programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(programmingExerciseId)).isEmpty(); - assertThat(codeHintRepository.findByExerciseId(programmingExerciseId)).isEmpty(); - assertThat(testCaseRepository.findByExerciseId(programmingExerciseId)).isEmpty(); - } - - @Test - void deleteTaskWithCodeHint() { - addCodeHintToProgrammingExercise(); - var task = taskRepository.findByExerciseId(programmingExerciseId).stream().findAny().orElseThrow(); - taskRepository.delete(task); - assertThat(codeHintRepository.findByExerciseId(programmingExerciseId)).isEmpty(); - assertThat(testCaseRepository.findByExerciseId(programmingExerciseId)).hasSize(3); - assertThat(programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(programmingExerciseId)).hasSize(6); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseGitDiffReportIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseGitDiffReportIntegrationTest.java deleted file mode 100644 index b896af7ac582..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseGitDiffReportIntegrationTest.java +++ /dev/null @@ -1,223 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.time.ZonedDateTime; - -import org.junit.jupiter.api.AfterEach; -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.http.HttpStatus; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationLocalCILocalVCTestBase; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; -import de.tum.cit.aet.artemis.programming.util.LocalRepository; -import de.tum.cit.aet.artemis.programming.util.ProgrammingExerciseFactory; - -/** - * Tests for the ProgrammingExerciseGitDiffReportResource - */ -class ProgrammingExerciseGitDiffReportIntegrationTest extends AbstractProgrammingIntegrationLocalCILocalVCTestBase { - - private static final String TEST_PREFIX = "progexgitdiffreport"; - - private static final String FILE_NAME = "test.java"; - - private static final String FILE_NAME2 = "test2.java"; - - private final LocalRepository solutionRepo = new LocalRepository("main"); - - private final LocalRepository templateRepo = new LocalRepository("main"); - - private final LocalRepository participationRepo = new LocalRepository("main"); - - private ProgrammingExercise exercise; - - @BeforeEach - void initTestCase() { - Course course = courseUtilService.addEmptyCourse(); - userUtilService.addUsers(TEST_PREFIX, 1, 1, 1, 1); - exercise = ProgrammingExerciseFactory.generateProgrammingExercise(ZonedDateTime.now().minusDays(1), ZonedDateTime.now().plusDays(7), course); - } - - @Override - protected String getTestPrefix() { - return TEST_PREFIX; - } - - @AfterEach - void cleanup() throws Exception { - solutionRepo.resetLocalRepo(); - templateRepo.resetLocalRepo(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void getGitDiffAsAStudent() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "TEST", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "TEST", exercise, solutionRepo); - reportService.updateReport(exercise); - request.get("/api/programming-exercises/" + exercise.getId() + "/diff-report", HttpStatus.FORBIDDEN, ProgrammingExerciseGitDiffReport.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void getGitDiffAsATutor() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "TEST", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "TEST", exercise, solutionRepo); - reportService.updateReport(exercise); - var report = request.get("/api/programming-exercises/" + exercise.getId() + "/diff-report", HttpStatus.OK, ProgrammingExerciseGitDiffReport.class); - assertThat(report).isNotNull(); - assertThat(report.getEntries()).isNull(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void getGitDiffAsAnEditor() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "TEST", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "TEST", exercise, solutionRepo); - reportService.updateReport(exercise); - var report = request.get("/api/programming-exercises/" + exercise.getId() + "/diff-report", HttpStatus.OK, ProgrammingExerciseGitDiffReport.class); - assertThat(report).isNotNull(); - assertThat(report.getEntries()).isNull(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void getGitDiffAsAnInstructor() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "TEST", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "TEST", exercise, solutionRepo); - reportService.updateReport(exercise); - var report = request.get("/api/programming-exercises/" + exercise.getId() + "/diff-report", HttpStatus.OK, ProgrammingExerciseGitDiffReport.class); - assertThat(report).isNotNull(); - assertThat(report.getEntries()).isNull(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void getGitDiffBetweenTemplateAndSubmission() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "ABC", exercise, templateRepo); - participationRepo.configureRepos("participationLocalRepo", "participationOriginRepo"); - var studentLogin = TEST_PREFIX + "student1"; - var submission = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST", exercise, participationRepo, studentLogin); - var report = request.get("/api/programming-exercises/" + exercise.getId() + "/submissions/" + submission.getId() + "/diff-report-with-template", HttpStatus.OK, - ProgrammingExerciseGitDiffReport.class); - assertThat(report).isNotNull(); - assertThat(report.getEntries()).isNull(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void getGitDiffBetweenTemplateAndSubmissionEditorForbidden() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "ABC", exercise, templateRepo); - participationRepo.configureRepos("participationLocalRepo", "participationOriginRepo"); - var studentLogin = TEST_PREFIX + "student1"; - var submission = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST", exercise, participationRepo, studentLogin); - request.get("/api/programming-exercises/" + exercise.getId() + "/submissions/" + submission.getId() + "/diff-report-with-template", HttpStatus.FORBIDDEN, - ProgrammingExerciseGitDiffReport.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void getGitDiffReportForCommits() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "ABC", exercise, templateRepo); - participationRepo.configureRepos("participationLocalRepo", "participationOriginRepo"); - var studentLogin = TEST_PREFIX + "student1"; - var submission = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST", exercise, participationRepo, studentLogin); - var submission2 = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST2", exercise, participationRepo, studentLogin); - var report = request.get("/api/programming-exercises/" + exercise.getId() + "/commits/" + submission.getCommitHash() + "/diff-report/" + submission2.getCommitHash() - + "?participationId=" + submission.getParticipation().getId(), HttpStatus.OK, ProgrammingExerciseGitDiffReport.class); - assertThat(report).isNotNull(); - assertThat(report.getEntries()).hasSize(1); - var entry = report.getEntries().stream().findAny().orElseThrow(); - assertThat(entry.getPreviousFilePath()).isEqualTo(FILE_NAME); - assertThat(entry.getPreviousStartLine()).isEqualTo(1); - assertThat(entry.getPreviousLineCount()).isEqualTo(1); - assertThat(entry.getFilePath()).isEqualTo(FILE_NAME); - assertThat(entry.getStartLine()).isEqualTo(1); - assertThat(entry.getLineCount()).isEqualTo(1); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void getGitDiffReportForCommitsWithRenamedFile() throws Exception { - String fileContent = "content\ncontent\ncontent\ncontent"; - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, fileContent, exercise, templateRepo); - participationRepo.configureRepos("participationLocalRepo", "participationOriginRepo"); - var studentLogin = TEST_PREFIX + "student1"; - var submission = hestiaUtilTestService.setupSubmission(FILE_NAME, fileContent, exercise, participationRepo, studentLogin); - // Simulate a renaming by deleting the file and creating a new one with the same content. Git will track this as long as the content is similar enough. - var submission2 = hestiaUtilTestService.deleteFileAndSetupSubmission(FILE_NAME, FILE_NAME2, fileContent, exercise, participationRepo, studentLogin); - var report = request.get("/api/programming-exercises/" + exercise.getId() + "/commits/" + submission.getCommitHash() + "/diff-report/" + submission2.getCommitHash() - + "?participationId=" + submission.getParticipation().getId(), HttpStatus.OK, ProgrammingExerciseGitDiffReport.class); - var entries = report.getEntries(); - assertThat(entries.size()).isEqualTo(1); - ProgrammingExerciseGitDiffEntry entry = entries.iterator().next(); - assertThat(entry.getPreviousFilePath()).isEqualTo(FILE_NAME); - assertThat(entry.getFilePath()).isEqualTo(FILE_NAME2); - assertThat(entry.isEmpty()).isTrue(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void getGitDiffReportForCommitsThrowsConflictException() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "ABC", exercise, templateRepo); - var wrongExerciseId = exercise.getId() + 1; - participationRepo.configureRepos("participationLocalRepo", "participationOriginRepo"); - var studentLogin = TEST_PREFIX + "student1"; - var submission = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST", exercise, participationRepo, studentLogin); - var submission2 = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST2", exercise, participationRepo, studentLogin); - request.get("/api/programming-exercises/" + wrongExerciseId + "/commits/" + submission.getCommitHash() + "/diff-report/" + submission2.getCommitHash() + "?participationId=" - + submission.getParticipation().getId(), HttpStatus.CONFLICT, ProgrammingExerciseGitDiffReport.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void getGitDiffReportForCommitsForbiddenIfNotParticipationOwner() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "ABC", exercise, templateRepo); - participationRepo.configureRepos("participationLocalRepo", "participationOriginRepo"); - // Create a submission for student2 and try to access it with student1 - var studentLogin = TEST_PREFIX + "instructor1"; - var submission = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST", exercise, participationRepo, studentLogin); - var submission2 = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST2", exercise, participationRepo, studentLogin); - request.get("/api/programming-exercises/" + exercise.getId() + "/commits/" + submission.getCommitHash() + "/diff-report/" + submission2.getCommitHash() - + "?participationId=" + submission.getParticipation().getId(), HttpStatus.FORBIDDEN, ProgrammingExerciseGitDiffReport.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void getGitDiffBetweenTwoSubmissions() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "ABC", exercise, templateRepo); - participationRepo.configureRepos("participationLocalRepo", "participationOriginRepo"); - var studentLogin = TEST_PREFIX + "student1"; - var submission = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST", exercise, participationRepo, studentLogin); - var submission2 = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST2", exercise, participationRepo, studentLogin); - var report = request.get("/api/programming-exercises/" + exercise.getId() + "/submissions/" + submission.getId() + "/diff-report/" + submission2.getId(), HttpStatus.OK, - ProgrammingExerciseGitDiffReport.class); - assertThat(report).isNotNull(); - assertThat(report.getEntries()).hasSize(1); - var entry = report.getEntries().stream().findAny().orElseThrow(); - assertThat(entry.getPreviousFilePath()).isEqualTo(FILE_NAME); - assertThat(entry.getPreviousStartLine()).isEqualTo(1); - assertThat(entry.getPreviousLineCount()).isEqualTo(1); - assertThat(entry.getFilePath()).isEqualTo(FILE_NAME); - assertThat(entry.getStartLine()).isEqualTo(1); - assertThat(entry.getLineCount()).isEqualTo(1); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void getGitDiffBetweenTwoSubmissionsEditorForbidden() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "ABC", exercise, templateRepo); - participationRepo.configureRepos("participationLocalRepo", "participationOriginRepo"); - var studentLogin = TEST_PREFIX + "student1"; - var submission = hestiaUtilTestService.setupSubmission(FILE_NAME, "TEST", exercise, participationRepo, studentLogin); - var submission2 = hestiaUtilTestService.setupSubmission(FILE_NAME2, "TEST2", exercise, participationRepo, studentLogin); - request.get("/api/programming-exercises/" + exercise.getId() + "/submissions/" + submission.getId() + "/diff-report/" + submission2.getId(), HttpStatus.FORBIDDEN, - ProgrammingExerciseGitDiffReport.class); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseGitDiffReportServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseGitDiffReportServiceTest.java deleted file mode 100644 index a79c8880e1fb..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseGitDiffReportServiceTest.java +++ /dev/null @@ -1,179 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.Comparator; - -import org.junit.jupiter.api.AfterEach; -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationLocalCILocalVCTestBase; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; -import de.tum.cit.aet.artemis.programming.util.LocalRepository; - -/** - * Tests for the ProgrammingExerciseGitDiffReportService - */ -class ProgrammingExerciseGitDiffReportServiceTest extends AbstractProgrammingIntegrationLocalCILocalVCTestBase { - - private static final String TEST_PREFIX = "progexgitdiffreportservice"; - - private static final String FILE_NAME = "test.java"; - - private final LocalRepository solutionRepo = new LocalRepository("main"); - - private final LocalRepository templateRepo = new LocalRepository("main"); - - private ProgrammingExercise exercise; - - @Override - protected String getTestPrefix() { - return TEST_PREFIX; - } - - @BeforeEach - void initTestCase() { - userUtilService.addUsers(TEST_PREFIX, 1, 1, 1, 1); - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(); - exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - } - - @AfterEach - void cleanup() throws IOException { - templateRepo.resetLocalRepo(); - solutionRepo.resetLocalRepo(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateGitDiffNoChanges() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "Line 1\nLine 2", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "Line 1\nLine 2", exercise, solutionRepo); - var report = reportService.updateReport(exercise); - assertThat(report.getEntries()).isNullOrEmpty(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateGitDiffAppendLine1() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "Line 1\nLine 2", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "Line 1\nLine 2\nLine 3\n", exercise, solutionRepo); - var report = reportService.updateReport(exercise); - assertThat(report.getEntries()).hasSize(1); - var entry = report.getEntries().stream().findFirst().orElseThrow(); - assertThat(entry.getPreviousStartLine()).isEqualTo(2); - assertThat(entry.getStartLine()).isEqualTo(2); - assertThat(entry.getPreviousLineCount()).isEqualTo(1); - assertThat(entry.getLineCount()).isEqualTo(2); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateGitDiffAppendLine2() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "Line 1\nLine 2\n", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "Line 1\nLine 2\nLine 3\n", exercise, solutionRepo); - var report = reportService.updateReport(exercise); - assertThat(report.getEntries()).hasSize(1); - var entry = report.getEntries().stream().findFirst().orElseThrow(); - assertThat(entry.getPreviousStartLine()).isNull(); - assertThat(entry.getStartLine()).isEqualTo(3); - assertThat(entry.getPreviousLineCount()).isNull(); - assertThat(entry.getLineCount()).isEqualTo(1); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateGitDiffAddToEmptyFile() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "Line 1\nLine 2", exercise, solutionRepo); - var report = reportService.updateReport(exercise); - assertThat(report.getEntries()).hasSize(1); - var entry = report.getEntries().stream().findFirst().orElseThrow(); - assertThat(entry.getPreviousStartLine()).isNull(); - assertThat(entry.getStartLine()).isEqualTo(1); - assertThat(entry.getPreviousLineCount()).isNull(); - assertThat(entry.getLineCount()).isEqualTo(2); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateGitDiffClearFile() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "Line 1\nLine 2", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "", exercise, solutionRepo); - var report = reportService.updateReport(exercise); - assertThat(report.getEntries()).hasSize(1); - var entry = report.getEntries().stream().findFirst().orElseThrow(); - assertThat(entry.getPreviousStartLine()).isEqualTo(1); - assertThat(entry.getStartLine()).isNull(); - assertThat(entry.getPreviousLineCount()).isEqualTo(2); - assertThat(entry.getLineCount()).isNull(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateGitDiffDoubleModify() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "L1\nL2\nL3\nL4", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "L1\nL2a\nL3\nL4a", exercise, solutionRepo); - var report = reportService.updateReport(exercise); - assertThat(report.getEntries()).hasSize(2); - var entries = new ArrayList<>(report.getEntries()); - entries.sort(Comparator.comparing(ProgrammingExerciseGitDiffEntry::getStartLine)); - assertThat(entries.getFirst().getPreviousStartLine()).isEqualTo(2); - assertThat(entries.getFirst().getStartLine()).isEqualTo(2); - assertThat(entries.getFirst().getPreviousLineCount()).isEqualTo(1); - assertThat(entries.getFirst().getLineCount()).isEqualTo(1); - - assertThat(entries.get(1).getPreviousStartLine()).isEqualTo(4); - assertThat(entries.get(1).getStartLine()).isEqualTo(4); - assertThat(entries.get(1).getPreviousLineCount()).isEqualTo(1); - assertThat(entries.get(1).getLineCount()).isEqualTo(1); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void gitDiffWhitespace() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, " ", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "\t", exercise, solutionRepo); - var report = reportService.updateReport(exercise); - assertThat(report.getEntries()).hasSize(0); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void updateGitDiffReuseExisting() throws Exception { - exercise = hestiaUtilTestService.setupTemplate(FILE_NAME, "Line 1\nLine 2", exercise, templateRepo); - exercise = hestiaUtilTestService.setupSolution(FILE_NAME, "Line 1\nLine 2\nLine 3\n", exercise, solutionRepo); - var report1 = reportService.updateReport(exercise); - assertThat(report1.getEntries()).hasSize(1); - var report2 = reportService.updateReport(exercise); - assertThat(report2.getEntries()).hasSize(1); - assertThat(report1.getId()).isEqualTo(report2.getId()); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void ensureDeletionOfDuplicateReports() { - var report1 = new ProgrammingExerciseGitDiffReport(); - report1.setProgrammingExercise(exercise); - report1.setTemplateRepositoryCommitHash("123"); - report1.setSolutionRepositoryCommitHash("456"); - reportRepository.save(report1); - var report2 = new ProgrammingExerciseGitDiffReport(); - report2.setProgrammingExercise(exercise); - report2.setTemplateRepositoryCommitHash("123"); - report2.setSolutionRepositoryCommitHash("789"); - report2 = reportRepository.save(report2); - - assertThat(reportRepository.findByProgrammingExerciseId(exercise.getId())).hasSize(2); - var returnedReport = reportService.getOrCreateReportOfExercise(exercise); - assertThat(returnedReport).isEqualTo(report2); - assertThat(reportRepository.findByProgrammingExerciseId(exercise.getId())).hasSize(1); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseSolutionEntryIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseSolutionEntryIntegrationTest.java deleted file mode 100644 index 17d326cfc221..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseSolutionEntryIntegrationTest.java +++ /dev/null @@ -1,245 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.HashSet; -import java.util.Optional; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.http.HttpStatus; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationIndependentTest; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; - -class ProgrammingExerciseSolutionEntryIntegrationTest extends AbstractProgrammingIntegrationIndependentTest { - - private static final String TEST_PREFIX = "progexsolutionentry"; - - private ProgrammingExercise programmingExercise; - - private CodeHint codeHint; - - @BeforeEach - void initTestCase() { - var course = programmingExerciseUtilService.addCourseWithOneProgrammingExerciseAndTestCases(); - userUtilService.addUsers(TEST_PREFIX, 2, 2, 1, 2); - - programmingExercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - Set testCases = testCaseRepository.findByExerciseIdWithSolutionEntries(programmingExercise.getId()); - - codeHint = new CodeHint(); - codeHint.setExercise(programmingExercise); - codeHint.setTitle("Code Hint Title"); - codeHint.setContent("Code Hint Content"); - codeHintRepository.save(codeHint); - - for (ProgrammingExerciseTestCase testCase : testCases) { - var solutionEntry = new ProgrammingExerciseSolutionEntry(); - solutionEntry.setTestCase(testCase); - solutionEntry.setPreviousCode("No code"); - solutionEntry.setCode("Some code"); - solutionEntry.setPreviousLine(1); - solutionEntry.setCodeHint(codeHint); - solutionEntry.setLine(1); - solutionEntry.setFilePath("code.java"); - programmingExerciseSolutionEntryRepository.save(solutionEntry); - } - ProgrammingExerciseTask task = new ProgrammingExerciseTask(); - task.setExercise(programmingExercise); - task.setTaskName("Task"); - task.setTestCases(new HashSet<>(testCases)); - task = taskRepository.save(task); - codeHint.setProgrammingExerciseTask(task); - codeHintRepository.save(codeHint); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testGetSolutionEntryById() throws Exception { - Long entryId = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(programmingExercise.getId()).stream().findFirst().orElseThrow().getId(); - ProgrammingExerciseSolutionEntry expectedSolutionEntry = programmingExerciseSolutionEntryRepository.findByIdWithTestCaseAndProgrammingExerciseElseThrow(entryId); - final var actualSolutionEntry = request.get("/api/programming-exercises/" + programmingExercise.getId() + "/solution-entries/" + expectedSolutionEntry.getId(), - HttpStatus.OK, ProgrammingExerciseSolutionEntry.class); - assertThat(actualSolutionEntry).isEqualTo(expectedSolutionEntry); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void testGetSolutionEntryByIdAsStudent() throws Exception { - Long entryId = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(programmingExercise.getId()).stream().findFirst().orElseThrow().getId(); - ProgrammingExerciseSolutionEntry expectedSolutionEntry = programmingExerciseSolutionEntryRepository.findByIdWithTestCaseAndProgrammingExerciseElseThrow(entryId); - request.get("/api/programming-exercises/" + programmingExercise.getId() + "/solution-entries/" + expectedSolutionEntry.getId(), HttpStatus.FORBIDDEN, - ProgrammingExerciseSolutionEntry.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void testGetSolutionEntryByIdWithInvalidExerciseIdAsStudent() throws Exception { - Long entryId = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(programmingExercise.getId()).stream().findFirst().orElseThrow().getId(); - request.get("/api/programming-exercises/" + Long.MAX_VALUE + "/solution-entries/" + entryId, HttpStatus.FORBIDDEN, ProgrammingExerciseSolutionEntry.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void testGetSolutionEntriesByCodeHintId() throws Exception { - final Set solutionEntries = new HashSet<>( - request.getList("/api/programming-exercises/" + programmingExercise.getId() + "/code-hints/" + codeHint.getId() + "/solution-entries", HttpStatus.OK, - ProgrammingExerciseSolutionEntry.class)); - assertThat(solutionEntries).isEqualTo(programmingExerciseSolutionEntryRepository.findByCodeHintId(codeHint.getId())); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void testGetSolutionEntriesByTestCaseId() throws Exception { - ProgrammingExerciseTestCase testCase = testCaseRepository.findByExerciseIdWithSolutionEntries(programmingExercise.getId()).stream().findFirst().orElseThrow(); - final var solutionEntries = new HashSet<>( - request.getList("/api/programming-exercises/" + programmingExercise.getId() + "/test-cases/" + testCase.getId() + "/solution-entries", HttpStatus.OK, - ProgrammingExerciseSolutionEntry.class)); - assertThat(solutionEntries).isEqualTo(testCase.getSolutionEntries()); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void testGetAllSolutionEntries() throws Exception { - var existingEntries = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(programmingExercise.getId()); - final var receivedEntries = request.getList("/api/programming-exercises/" + programmingExercise.getId() + "/solution-entries", HttpStatus.OK, - ProgrammingExerciseSolutionEntry.class); - assertThat(receivedEntries).containsExactlyElementsOf(existingEntries); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void testDeleteSolutionEntry() throws Exception { - ProgrammingExerciseTestCase testCase = testCaseRepository.findByExerciseIdWithSolutionEntries(programmingExercise.getId()).stream().findFirst().orElseThrow(); - Long entryId = testCase.getSolutionEntries().stream().findFirst().orElseThrow().getId(); - request.delete("/api/programming-exercises/" + programmingExercise.getId() + "/test-cases/" + testCase.getId() + "/solution-entries/" + entryId, HttpStatus.NO_CONTENT); - assertThat(programmingExerciseSolutionEntryRepository.findById(entryId)).isEmpty(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "STUDENT") - void testDeleteSolutionEntryAsStudent() throws Exception { - ProgrammingExerciseTestCase testCase = testCaseRepository.findByExerciseIdWithSolutionEntries(programmingExercise.getId()).stream().findFirst().orElseThrow(); - Long entryId = testCase.getSolutionEntries().stream().findFirst().orElseThrow().getId(); - request.delete("/api/programming-exercises/" + programmingExercise.getId() + "/test-cases/" + testCase.getId() + "/solution-entries/" + entryId, HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testDeleteSolutionEntryAsTutor() throws Exception { - ProgrammingExerciseTestCase testCase = testCaseRepository.findByExerciseIdWithSolutionEntries(programmingExercise.getId()).stream().findFirst().orElseThrow(); - Long entryId = testCase.getSolutionEntries().stream().findFirst().orElseThrow().getId(); - request.delete("/api/programming-exercises/" + programmingExercise.getId() + "/test-cases/" + testCase.getId() + "/solution-entries/" + entryId, HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void testDeleteAllSolutionEntriesForExercise() throws Exception { - request.delete("/api/programming-exercises/" + programmingExercise.getId() + "/solution-entries", HttpStatus.NO_CONTENT); - assertThat(programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(programmingExercise.getId())).hasSize(0); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void testUpdateSolutionEntry() throws Exception { - ProgrammingExerciseTestCase testCase = testCaseRepository.findByExerciseIdWithSolutionEntries(programmingExercise.getId()).stream().findFirst().orElseThrow(); - ProgrammingExerciseSolutionEntry entry = testCase.getSolutionEntries().stream().findFirst().orElseThrow(); - Long entryId = entry.getId(); - String updatedFilePath = "NewPath.java"; - entry.setFilePath(updatedFilePath); - - request.put("/api/programming-exercises/" + programmingExercise.getId() + "/test-cases/" + testCase.getId() + "/solution-entries/" + entryId, entry, HttpStatus.OK); - Optional entryAfterUpdate = programmingExerciseSolutionEntryRepository.findById(entryId); - assertThat(entryAfterUpdate).isPresent(); - assertThat(entryAfterUpdate.get().getFilePath()).isEqualTo(updatedFilePath); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void testUpdateSolutionEntryWithInvalidId() throws Exception { - ProgrammingExerciseTestCase testCase = testCaseRepository.findByExerciseIdWithSolutionEntries(programmingExercise.getId()).stream().findFirst().orElseThrow(); - ProgrammingExerciseSolutionEntry entry = testCase.getSolutionEntries().stream().findFirst().orElseThrow(); - Long entryId = entry.getId(); - String updatedFilePath = "NewPath.java"; - entry.setFilePath(updatedFilePath); - - request.put("/api/programming-exercises/" + programmingExercise.getId() + "/test-cases/" + testCase.getId() + "/solution-entries/" + entryId, entry, HttpStatus.OK); - Optional entryAfterUpdate = programmingExerciseSolutionEntryRepository.findById(entryId); - assertThat(entryAfterUpdate).isPresent(); - assertThat(entryAfterUpdate.get().getFilePath()).isEqualTo(updatedFilePath); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "STUDENT") - void testUpdateSolutionEntryAsStudent() throws Exception { - ProgrammingExerciseTestCase testCase = testCaseRepository.findByExerciseIdWithSolutionEntries(programmingExercise.getId()).stream().findFirst().orElseThrow(); - ProgrammingExerciseSolutionEntry entry = testCase.getSolutionEntries().stream().findFirst().orElseThrow(); - Long entryId = entry.getId(); - - request.put("/api/programming-exercises/" + programmingExercise.getId() + "/test-cases/" + testCase.getId() + "/solution-entries/" + entryId, entry, HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testUpdateSolutionEntryAsTutor() throws Exception { - ProgrammingExerciseTestCase testCase = testCaseRepository.findByExerciseIdWithSolutionEntries(programmingExercise.getId()).stream().findFirst().orElseThrow(); - ProgrammingExerciseSolutionEntry entry = testCase.getSolutionEntries().stream().findFirst().orElseThrow(); - Long entryId = entry.getId(); - - request.put("/api/programming-exercises/" + programmingExercise.getId() + "/test-cases/" + testCase.getId() + "/solution-entries/" + entryId, entry, HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "STUDENT") - void testCreateStructuralSolutionEntriesAsStudent() throws Exception { - request.post("/api/programming-exercises/" + programmingExercise.getId() + "/structural-solution-entries", null, HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testCreateStructuralSolutionEntriesAsTutor() throws Exception { - request.post("/api/programming-exercises/" + programmingExercise.getId() + "/structural-solution-entries", null, HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void testCreateStructuralSolutionEntriesAsEditor() throws Exception { - request.postWithoutLocation("/api/programming-exercises/" + programmingExercise.getId() + "/structural-solution-entries", null, HttpStatus.OK, null); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testCreateStructuralSolutionEntriesAsInstructor() throws Exception { - request.postWithoutLocation("/api/programming-exercises/" + programmingExercise.getId() + "/structural-solution-entries", null, HttpStatus.OK, null); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testCreateManualSolutionEntry() throws Exception { - programmingExerciseSolutionEntryRepository.deleteAll(); - - var manualEntry = new ProgrammingExerciseSolutionEntry(); - manualEntry.setCode("abc"); - manualEntry.setLine(1); - manualEntry.setFilePath("src/de/tum/in/ase/BubbleSort.java"); - - var testCase = testCaseRepository.findByExerciseId(programmingExercise.getId()).stream().findFirst().orElseThrow(); - manualEntry.setTestCase(testCase); - - request.postWithoutLocation("/api/programming-exercises/" + programmingExercise.getId() + "/test-cases/" + testCase.getId() + "/solution-entries", manualEntry, - HttpStatus.CREATED, null); - - var savedEntries = programmingExerciseSolutionEntryRepository.findByExerciseIdWithTestCases(programmingExercise.getId()); - assertThat(savedEntries).hasSize(1); - var createdEntry = savedEntries.iterator().next(); - assertThat(createdEntry).usingRecursiveComparison().ignoringActualNullFields().isEqualTo(createdEntry); - assertThat(createdEntry.getTestCase().getId()).isEqualTo(testCase.getId()); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseTaskIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseTaskIntegrationTest.java deleted file mode 100644 index 55f7d322a023..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseTaskIntegrationTest.java +++ /dev/null @@ -1,222 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.Collection; -import java.util.HashSet; -import java.util.List; -import java.util.Optional; -import java.util.Set; -import java.util.stream.Collectors; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.http.HttpStatus; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationIndependentTest; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; - -class ProgrammingExerciseTaskIntegrationTest extends AbstractProgrammingIntegrationIndependentTest { - - private static final String TEST_PREFIX = "progextask"; - - private ProgrammingExercise programmingExercise; - - private Set testCases; - - @BeforeEach - void initTestCases() { - userUtilService.addUsers(TEST_PREFIX, 1, 1, 1, 1); - - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExerciseAndSpecificTestCases(); - programmingExercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - this.testCases = testCaseRepository.findByExerciseIdWithSolutionEntries(programmingExercise.getId()); - for (ProgrammingExerciseTestCase testCase : testCases) { - var solutionEntry = new ProgrammingExerciseSolutionEntry(); - solutionEntry.setTestCase(testCase); - solutionEntry.setPreviousCode("No code"); - solutionEntry.setCode("Some code"); - solutionEntry.setPreviousLine(1); - solutionEntry.setCodeHint(null); - solutionEntry.setLine(1); - solutionEntry.setFilePath("code.java"); - programmingExerciseSolutionEntryRepository.save(solutionEntry); - } - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void testDeletionAsStudent() throws Exception { - request.delete("/api/programming-exercises/" + programmingExercise.getId() + "/tasks", HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testDeletionAsTutor() throws Exception { - request.delete("/api/programming-exercises/" + programmingExercise.getId() + "/tasks", HttpStatus.FORBIDDEN); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") - void testDeletionAsEditor() throws Exception { - request.delete("/api/programming-exercises/" + programmingExercise.getId() + "/tasks", HttpStatus.NO_CONTENT); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testDeleteAllTasksAndSolutionEntriesForProgrammingExercise() throws Exception { - Set solutionEntryIdsBeforeDeleting = testCases.stream().map(ProgrammingExerciseTestCase::getSolutionEntries).flatMap(Collection::stream).map(DomainObject::getId) - .collect(Collectors.toSet()); - - ProgrammingExerciseTask task = new ProgrammingExerciseTask(); - task.setExercise(programmingExercise); - task.setTaskName("Task"); - task.setTestCases(new HashSet<>(testCases)); - taskRepository.save(task); - - request.delete("/api/programming-exercises/" + programmingExercise.getId() + "/tasks", HttpStatus.NO_CONTENT); - assertThat(taskRepository.findByExerciseId(programmingExercise.getId())).isEmpty(); - assertThat(programmingExerciseSolutionEntryRepository.findAllById(solutionEntryIdsBeforeDeleting)).isEmpty(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void testTaskExtractionAsStudent() throws Exception { - request.get("/api/programming-exercises/" + programmingExercise.getId() + "/tasks", HttpStatus.FORBIDDEN, Set.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testTaskExtractionForProgrammingExercise() throws Exception { - String taskName1 = "Implement Bubble Sort"; - String taskName2 = "Implement Policy and Context"; - programmingExercise.setProblemStatement(""" - # Sorting with the Strategy Pattern - - In this exercise, we want to implement sorting algorithms and choose them based on runtime specific variables. - - ### Part 1: Sorting - - First, we need to implement two sorting algorithms, in this case `MergeSort` and `BubbleSort`. - - **You have the following tasks:** - - 1. [task][%s](testClass[BubbleSort]) - Implement the class `BubbleSort`. - 2. [task][%s](testMethods[Context],testMethods[Policy]) - Implement the classes `Context` and `Policy`. Make sure to follow..""".formatted(taskName1, taskName2)); - programmingExerciseRepository.save(programmingExercise); - programmingExerciseTaskService.updateTasksFromProblemStatement(programmingExercise); - - request.get("/api/programming-exercises/" + programmingExercise.getId() + "/tasks", HttpStatus.OK, Set.class); - List extractedTasks = taskRepository.findByExerciseIdWithTestCaseAndSolutionEntriesElseThrow(programmingExercise.getId()); - Optional task1Optional = extractedTasks.stream().filter(task -> task.getTaskName().equals(taskName1)).findFirst(); - Optional task2Optional = extractedTasks.stream().filter(task -> task.getTaskName().equals(taskName2)).findFirst(); - assertThat(task1Optional).isPresent(); - assertThat(task2Optional).isPresent(); - ProgrammingExerciseTask task1 = task1Optional.get(); - ProgrammingExerciseTask task2 = task2Optional.get(); - - Set expectedTestCasesForTask1 = new HashSet<>(); - expectedTestCasesForTask1.add(testCases.stream().filter(testCase -> "testClass[BubbleSort]".equals(testCase.getTestName())).findFirst().orElseThrow()); - Set expectedTestCasesForTask2 = new HashSet<>(); - expectedTestCasesForTask2.add(testCases.stream().filter(testCase -> "testMethods[Context]".equals(testCase.getTestName())).findFirst().orElseThrow()); - expectedTestCasesForTask2.add(testCases.stream().filter(testCase -> "testMethods[Policy]".equals(testCase.getTestName())).findFirst().orElseThrow()); - assertThat(task1.getTestCases()).isEqualTo(expectedTestCasesForTask1); - assertThat(task2.getTestCases()).isEqualTo(expectedTestCasesForTask2); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testTaskExtractionForEmptyProblemStatement() throws Exception { - programmingExercise.setProblemStatement(""); - programmingExercise = programmingExerciseRepository.save(programmingExercise); - - request.get("/api/programming-exercises/" + programmingExercise.getId() + "/tasks", HttpStatus.OK, Set.class); - - assertThat(taskRepository.findByExerciseId(programmingExercise.getId())).isEmpty(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testGetTasksWithUnassignedTestCases_NoTasks() throws Exception { - var response = request.getList("/api/programming-exercises/" + programmingExercise.getId() + "/tasks-with-unassigned-test-cases", HttpStatus.OK, - ProgrammingExerciseTask.class); - - // No tasks available -> all tests in one "unassigned" group - assertThat(response).hasSize(1); - var unassigned = response.getFirst(); - assertThat(unassigned.getTaskName()).isEqualTo("Not assigned to task"); - assertThat(unassigned.getTestCases()).hasSize(3); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testGetTasksWithUnassignedTestCases_AllTasksAssigned() throws Exception { - String taskName1 = "Implement Bubble Sort"; - String taskName2 = "Implement Policy and Context"; - programmingExercise.setProblemStatement(""" - # Sorting with the Strategy Pattern - - In this exercise, we want to implement sorting algorithms and choose them based on runtime specific variables. - - ### Part 1: Sorting - - First, we need to implement two sorting algorithms, in this case `MergeSort` and `BubbleSort`. - - **You have the following tasks:** - - 1. [task][%s](testClass[BubbleSort]) - Implement the class `BubbleSort`. - 2. [task][%s](testMethods[Context],testMethods[Policy]) - Implement the classes `Context` and `Policy`. Make sure to follow.. - """.formatted(taskName1, taskName2)); - programmingExerciseRepository.save(programmingExercise); - programmingExerciseTaskService.updateTasksFromProblemStatement(programmingExercise); - - var response = request.getList("/api/programming-exercises/" + programmingExercise.getId() + "/tasks-with-unassigned-test-cases", HttpStatus.OK, - ProgrammingExerciseTask.class); - - // 2 tasks, all test cases distributed across the tasks -> no unassigned - assertThat(response).hasSize(2); - var bubbleSort = response.stream().filter(task -> taskName1.equals(task.getTaskName())).findFirst().orElseThrow(); - var context = response.stream().filter(task -> taskName2.equals(task.getTaskName())).findFirst().orElseThrow(); - - assertThat(bubbleSort.getTestCases()).hasSize(1).allMatch(tc -> "testClass[BubbleSort]".equals(tc.getTestName())); - assertThat(context.getTestCases()).hasSize(2).anyMatch(tc -> "testMethods[Context]".equals(tc.getTestName())) - .anyMatch(tc -> "testMethods[Policy]".equals(tc.getTestName())); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void testGetTasksWithUnassignedTestCases_Mixed() throws Exception { - String taskName = "Implement Bubble Sort"; - programmingExercise.setProblemStatement("[task][%s](testClass[BubbleSort])".formatted(taskName)); - programmingExerciseRepository.save(programmingExercise); - programmingExerciseTaskService.updateTasksFromProblemStatement(programmingExercise); - - var response = request.getList("/api/programming-exercises/" + programmingExercise.getId() + "/tasks-with-unassigned-test-cases", HttpStatus.OK, - ProgrammingExerciseTask.class); - - // 1 task, 1 unassigned - assertThat(response).hasSize(2); - var bubbleSort = response.stream().filter(task -> taskName.equals(task.getTaskName())).findFirst().orElseThrow(); - var unassigned = response.stream().filter(task -> "Not assigned to task".equals(task.getTaskName())).findFirst().orElseThrow(); - - assertThat(bubbleSort.getTestCases()).hasSize(1).allMatch(tc -> "testClass[BubbleSort]".equals(tc.getTestName())); - assertThat(unassigned.getTestCases()).hasSize(2).anyMatch(tc -> "testMethods[Context]".equals(tc.getTestName())) - .anyMatch(tc -> "testMethods[Policy]".equals(tc.getTestName())); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void testGetTasksWithUnassignedTestCases_AsStudent() throws Exception { - request.get("/api/programming-exercises/" + programmingExercise.getId() + "/tasks-with-unassigned-test-cases", HttpStatus.FORBIDDEN, Set.class); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseTaskServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseTaskServiceTest.java deleted file mode 100644 index a322fa44ba44..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/ProgrammingExerciseTaskServiceTest.java +++ /dev/null @@ -1,402 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.List; -import java.util.Map; -import java.util.Set; -import java.util.stream.Collectors; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.core.domain.DomainObject; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationIndependentTest; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; - -class ProgrammingExerciseTaskServiceTest extends AbstractProgrammingIntegrationIndependentTest { - - private static final String TEST_PREFIX = "progextaskservice"; - - private ProgrammingExercise programmingExercise; - - @BeforeEach - void init() { - userUtilService.addUsers(TEST_PREFIX, 1, 1, 1, 1); - - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExerciseAndSpecificTestCases(); - programmingExercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - updateProblemStatement(""" - [task][Task 1](testClass[BubbleSort]) - [task][Task 2](testMethods[Context]) - """); - } - - private void updateProblemStatement(String problemStatement) { - programmingExercise.setProblemStatement(problemStatement); - programmingExercise = programmingExerciseRepository.save(programmingExercise); - programmingExerciseTaskService.updateTasksFromProblemStatement(programmingExercise); - } - - @Test - void testNewExercise() { - assertThat(taskRepository.findByExerciseId(programmingExercise.getId())).hasSize(2); - var tasks = taskRepository.findByExerciseIdWithTestCases(programmingExercise.getId()); - assertThat(tasks).hasSize(2).anyMatch(programmingExerciseTask -> checkTaskEqual(programmingExerciseTask, "Task 1", "testClass[BubbleSort]")) - .anyMatch(programmingExerciseTask -> checkTaskEqual(programmingExerciseTask, "Task 2", "testMethods[Context]")); - } - - @Test - void testAddTask() { - var previousTaskIds = taskRepository.findByExerciseIdWithTestCases(programmingExercise.getId()).stream().map(ProgrammingExerciseTask::getId).collect(Collectors.toSet()); - - updateProblemStatement(""" - [task][Task 1](testClass[BubbleSort]) - [task][Task 2](testMethods[Context]) - [task][Task 3](testMethods[Policy]) - """); - assertThat(taskRepository.findByExerciseId(programmingExercise.getId())).hasSize(3); - var tasks = taskRepository.findByExerciseIdWithTestCases(programmingExercise.getId()); - assertThat(tasks).hasSize(3).anyMatch(programmingExerciseTask -> checkTaskEqual(programmingExerciseTask, "Task 1", "testClass[BubbleSort]")) - .anyMatch(programmingExerciseTask -> checkTaskEqual(programmingExerciseTask, "Task 2", "testMethods[Context]")) - .anyMatch(programmingExerciseTask -> checkTaskEqual(programmingExerciseTask, "Task 3", "testMethods[Policy]")); - - // Test that the other tasks were not removed and re-added. - var newTaskIds = tasks.stream().map(ProgrammingExerciseTask::getId).collect(Collectors.toSet()); - assertThat(newTaskIds).containsAll(previousTaskIds); - } - - @Test - void testRemoveAllTasks() { - updateProblemStatement("Empty"); - assertThat(taskRepository.findByExerciseId(programmingExercise.getId())).isEmpty(); - } - - @Test - void testReduceToOneTask() { - updateProblemStatement("[task][Task 1](testClass[BubbleSort],testMethods[Context], testMethods[Policy])"); - assertThat(taskRepository.findByExerciseId(programmingExercise.getId())).hasSize(1); - var tasks = taskRepository.findByExerciseIdWithTestCases(programmingExercise.getId()); - assertThat(tasks).hasSize(1); - var task = tasks.stream().findFirst().orElseThrow(); - assertThat(task.getTaskName()).isEqualTo("Task 1"); - assertThat(task.getTestCases()).hasSize(3); - var expectedTestCaseNames = Set.of("testClass[BubbleSort]", "testMethods[Context]", "testMethods[Policy]"); - var actualTestCaseNames = task.getTestCases().stream().map(ProgrammingExerciseTestCase::getTestName).collect(Collectors.toSet()); - assertThat(actualTestCaseNames).isEqualTo(expectedTestCaseNames); - } - - /** - * Tests that renaming a task does not remove and read the task, but instead updates it - */ - @Test - void testRenameTask() { - var previousTaskIds = taskRepository.findByExerciseIdWithTestCases(programmingExercise.getId()).stream().map(ProgrammingExerciseTask::getId).collect(Collectors.toSet()); - - updateProblemStatement(""" - [task][Task 1a](testClass[BubbleSort]) - [task][Task 2](testMethods[Context]) - """); - - assertThat(taskRepository.findByExerciseId(programmingExercise.getId())).hasSize(2); - var tasks = taskRepository.findByExerciseIdWithTestCases(programmingExercise.getId()); - - var newTaskIds = tasks.stream().map(ProgrammingExerciseTask::getId).collect(Collectors.toSet()); - assertThat(previousTaskIds).isEqualTo(newTaskIds); - - assertThat(taskRepository.findByExerciseIdWithTestCases(programmingExercise.getId())).isEqualTo(tasks); - - assertThat(tasks).anyMatch(programmingExerciseTask -> checkTaskEqual(programmingExerciseTask, "Task 1a", "testClass[BubbleSort]")) - .anyMatch(programmingExerciseTask -> checkTaskEqual(programmingExerciseTask, "Task 2", "testMethods[Context]")); - } - - /** - * Tests that not changing any tasks in the problem statement will not update any tasks - */ - @Test - void testNoChanges() { - var previousTaskIds = taskRepository.findByExerciseIdWithTestCases(programmingExercise.getId()).stream().map(ProgrammingExerciseTask::getId).collect(Collectors.toSet()); - - updateProblemStatement(""" - Test - [task][Task 1](testClass[BubbleSort]) - [task][Task 2](testMethods[Context]) - """); - - assertThat(taskRepository.findByExerciseId(programmingExercise.getId())).hasSize(2); - - var newTaskIds = taskRepository.findByExerciseIdWithTestCases(programmingExercise.getId()).stream().map(ProgrammingExerciseTask::getId).collect(Collectors.toSet()); - assertThat(previousTaskIds).isEqualTo(newTaskIds); - } - - @Test - void testDeleteWithCodeHints() { - var task = taskRepository.findByExerciseId(programmingExercise.getId()).stream().filter(task1 -> "Task 1".equals(task1.getTaskName())).findFirst().orElse(null); - assertThat(task).isNotNull(); - - var codeHint = new CodeHint(); - codeHint.setExercise(programmingExercise); - codeHint.setProgrammingExerciseTask(task); - codeHintRepository.save(codeHint); - - programmingExerciseTaskService.delete(task); - assertThat(taskRepository.findByExerciseId(programmingExercise.getId())).hasSize(1); - assertThat(taskRepository.findById(task.getId())).isEmpty(); - assertThat(codeHintRepository.findByExerciseId(programmingExercise.getId())).isEmpty(); - } - - @Test - @WithMockUser(username = "instructor1", roles = "INSTRUCTOR") - void getTasksWithoutInactiveFiltersOutInactive() { - programmingExercise = programmingExerciseRepository - .findByIdWithEagerTestCasesStaticCodeAnalysisCategoriesHintsAndTemplateAndSolutionParticipationsAndAuxReposAndBuildConfig(programmingExercise.getId()) - .orElseThrow(); - testCaseRepository.deleteAll(programmingExercise.getTestCases()); - - String[] testCaseNames = { "testClass[BubbleSort]", "testParametrized(Parameter1, 2)[1]" }; - for (var name : testCaseNames) { - var testCase = new ProgrammingExerciseTestCase(); - testCase.setExercise(programmingExercise); - testCase.setTestName(name); - testCase.setActive(true); - testCaseRepository.save(testCase); - } - - var testCase = new ProgrammingExerciseTestCase(); - testCase.setExercise(programmingExercise); - testCase.setTestName("testWithBraces()"); - testCase.setActive(false); - testCaseRepository.save(testCase); - - programmingExercise = programmingExerciseRepository - .findByIdWithEagerTestCasesStaticCodeAnalysisCategoriesHintsAndTemplateAndSolutionParticipationsAndAuxReposAndBuildConfig(programmingExercise.getId()) - .orElseThrow(); - - updateProblemStatement(""" - [task][Task 1](testClass[BubbleSort],testWithBraces(),testParametrized(Parameter1, 2)[1]) - """); - - var actualTasks = programmingExerciseTaskService.getTasksWithoutInactiveTestCases(programmingExercise.getId()); - assertThat(actualTasks).hasSize(1); - - var actualTestCases = actualTasks.stream().findFirst().get().getTestCases(); - assertThat(actualTestCases).hasSize(2).allMatch(ProgrammingExerciseTestCase::isActive); - } - - @Test - @WithMockUser(username = "instructor1", roles = "INSTRUCTOR") - void testParseTestCaseNames() { - programmingExercise = programmingExerciseRepository - .findByIdWithEagerTestCasesStaticCodeAnalysisCategoriesHintsAndTemplateAndSolutionParticipationsAndAuxReposAndBuildConfig(programmingExercise.getId()) - .orElseThrow(); - testCaseRepository.deleteAll(programmingExercise.getTestCases()); - - String[] testCaseNames = new String[] { "testClass[BubbleSort]", "testWithBraces()", "testParametrized(Parameter1, 2)[1]" }; - for (var name : testCaseNames) { - var testCase = new ProgrammingExerciseTestCase(); - testCase.setExercise(programmingExercise); - testCase.setTestName(name); - testCase.setActive(true); - testCaseRepository.save(testCase); - } - programmingExercise = programmingExerciseRepository - .findByIdWithEagerTestCasesStaticCodeAnalysisCategoriesHintsAndTemplateAndSolutionParticipationsAndAuxReposAndBuildConfig(programmingExercise.getId()) - .orElseThrow(); - - updateProblemStatement(""" - [task][Task 1](testClass[BubbleSort],testWithBraces(),testParametrized(Parameter1, 2)[1]) - """); - - var actualTasks = taskRepository.findByExerciseId(programmingExercise.getId()); - assertThat(actualTasks).hasSize(1); - final var actualTask = actualTasks.iterator().next().getId(); - var actualTaskWithTestCases = taskRepository.findByIdWithTestCaseAndSolutionEntriesElseThrow(actualTask); - assertThat(actualTaskWithTestCases.getTaskName()).isEqualTo("Task 1"); - var actualTestCaseNames = actualTaskWithTestCases.getTestCases().stream().map(ProgrammingExerciseTestCase::getTestName).toList(); - assertThat(actualTestCaseNames).containsExactlyInAnyOrder(testCaseNames); - } - - @Test - @WithMockUser(username = "instructor1", roles = "INSTRUCTOR") - void testExtractTasksFromTestIds() { - var test1 = testCaseRepository.findByExerciseIdAndTestName(programmingExercise.getId(), "testClass[BubbleSort]").orElseThrow(); - var test2 = testCaseRepository.findByExerciseIdAndTestName(programmingExercise.getId(), "testMethods[Context]").orElseThrow(); - - updateProblemStatement("[task][Task 1](%s,%s)".formatted(test1.getId(), test2.getId())); - - var actualTasks = taskRepository.findByExerciseId(programmingExercise.getId()); - assertThat(actualTasks).hasSize(1); - final var actualTask = actualTasks.iterator().next().getId(); - var actualTaskWithTestCases = taskRepository.findByIdWithTestCaseAndSolutionEntriesElseThrow(actualTask); - assertThat(actualTaskWithTestCases.getTaskName()).isEqualTo("Task 1"); - var actualTestCaseNames = actualTaskWithTestCases.getTestCases().stream().map(ProgrammingExerciseTestCase::getTestName).toList(); - assertThat(actualTestCaseNames).containsExactlyInAnyOrder("testClass[BubbleSort]", "testMethods[Context]"); - } - - private boolean checkTaskEqual(ProgrammingExerciseTask task, String expectedName, String expectedTestName) { - var testCases = task.getTestCases(); - return expectedName.equals(task.getTaskName()) && !testCases.isEmpty() && expectedTestName.equals(testCases.stream().findFirst().get().getTestName()); - } - - @Test - void testNameReplacement() { - Map testCases = testCaseRepository.findByExerciseId(programmingExercise.getId()).stream() - .collect(Collectors.toMap(ProgrammingExerciseTestCase::getTestName, ProgrammingExerciseTestCase::getId)); - - programmingExerciseTaskService.replaceTestNamesWithIds(programmingExercise); - String problemStatement = programmingExercise.getProblemStatement(); - - // [task][Task 1](testClass[BubbleSort]) - // [task][Task 2](testMethods[Context]) - assertThat(problemStatement).as("no more test case names should be present").doesNotContain(testCases.keySet()) - .contains("[task][Task 1](%s)".formatted(testCases.get("testClass[BubbleSort]"))) - .contains("[task][Task 2](%s)".formatted(testCases.get("testMethods[Context]"))); - } - - @Test - void testNameReplacementKeepsInactiveTests() { - // Task 1 is inactive, task 2 does not exist - updateProblemStatement("[task][Task 1](testClass[BubbleSort])\n[task][Task 2](nonExistingTask)"); - var testCase = testCaseRepository.findByExerciseIdAndTestName(programmingExercise.getId(), "testClass[BubbleSort]").orElseThrow(); - testCase.setActive(false); - testCaseRepository.save(testCase); - - programmingExerciseTaskService.replaceTestNamesWithIds(programmingExercise); - String problemStatement = programmingExercise.getProblemStatement(); - - // Task 1 still contains the name since the mentioned test case is not active - assertThat(problemStatement).contains("[task][Task 1](testClass[BubbleSort])").contains("[task][Task 2](nonExistingTask)"); - } - - @Test - void testNameReplacementSpecialNames() { - var bubbleSort = testCaseRepository.findByExerciseIdAndTestName(programmingExercise.getId(), "testClass[BubbleSort]").orElseThrow(); - var braces = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "testWithBraces()"); - var parameterized = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "testParametrized(Parameter1, 2)[1]"); - updateProblemStatement(""" - [task][Task 1](testClass[BubbleSort],testWithBraces(),testParametrized(Parameter1, 2)[1]) - """); - - programmingExerciseTaskService.replaceTestNamesWithIds(programmingExercise); - String problemStatement = programmingExercise.getProblemStatement(); - - assertThat(problemStatement).as("no more test case names should be present") - .doesNotContain("testClass[BubbleSort]", "testWithBraces()", "testParametrized(Parameter1, 2)[1]") - .contains("[task][Task 1](%s,%s,%s)".formatted(bubbleSort.getId(), braces.getId(), parameterized.getId())); - } - - @Test - void replacePlantUMLTestsWithIds() { - var testConstructorsLinkedList = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "testConstructors[LinkedList]"); - var testMethodsLinkedList = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "testMethods[LinkedList]"); - var testAttributesListNode = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "testAttributes[ListNode]"); - var testConstructorsListNode = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "testConstructors[ListNode]"); - var testClassLinkedList = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "testClass[LinkedList]"); - var testAttributesLinkedList = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "testAttributes[LinkedList]"); - updateProblemStatement(""" - @startuml - - class LinkedList { - + LinkedList() - :+ toString(): String - } - - class ListNode { - - value: T - + ListNode() - + ListNode(T) - + ListNode(T, ListNode, ListNode) - } - - LinkedList -[hidden]> ListNode - - LinkedList --up-|> MyList #testsColor(testClass[LinkedList]) - LinkedList --> ListNode #testsColor(testAttributes[LinkedList]): first - LinkedList --> ListNode #testsColor(testAttributes[LinkedList]): last - ListNode -l-> ListNode #testsColor(testAttributes[ListNode]): previous - ListNode -r-> ListNode #testsColor(testAttributes[ListNode]): next - - hide empty fields - hide empty methods - hide circles - - @enduml"""); - - programmingExerciseTaskService.replaceTestNamesWithIds(programmingExercise); - String problemStatement = programmingExercise.getProblemStatement(); - - var allTests = List.of(testConstructorsLinkedList, testMethodsLinkedList, testAttributesListNode, testConstructorsListNode, testClassLinkedList, testAttributesLinkedList); - var allTestNames = allTests.stream().map(ProgrammingExerciseTestCase::getTestName).toList(); - var allExpectedReplacements = allTests.stream().map(DomainObject::getId).map(id -> "" + id + "").toList(); - - assertThat(problemStatement).as("All test names got replaced").doesNotContain(allTestNames).contains(allExpectedReplacements); - } - - @Test - void testNameReplacementOnlyWithinTasks() { - // Tests that the replacing of test names only addresses test names inside of tasks. - // If a test name gets used in the regular problem statement text it gets kept. - - var test1 = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "test"); - var test2 = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "taskTest"); - - updateProblemStatement("[task][Taskname](test, taskTest)\nThis description contains the words test and taskTest, which should not be replaced."); - - programmingExerciseTaskService.replaceTestNamesWithIds(programmingExercise); - String problemStatement = programmingExercise.getProblemStatement(); - - assertThat(problemStatement).contains("[task][Taskname](%s,%s)".formatted(test1.getId(), test2.getId())) - .contains("This description contains the words test and taskTest, which should not be replaced."); - } - - @Test - void testNameReplacementTaskNameSameAsTestName() { - var sort = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "sort"); - - updateProblemStatement(""" - [task][sort](sort) - Sort using the method sort. - @startuml - class LinkedList { - + sort() - } - @enduml"""); - - programmingExerciseTaskService.replaceTestNamesWithIds(programmingExercise); - String problemStatement = programmingExercise.getProblemStatement(); - - assertThat(problemStatement).contains("[task][sort](%s)".formatted(sort.getId())).contains("Sort using the method sort.") - .contains("%s)>+ sort()".formatted(sort.getId())); - } - - @Test - void testIdReplacementWithNames() { - var bubbleSort = testCaseRepository.findByExerciseIdAndTestName(programmingExercise.getId(), "testClass[BubbleSort]").orElseThrow(); - var inactiveTest = programmingExerciseUtilService.addTestCaseToProgrammingExercise(programmingExercise, "testName"); - inactiveTest.setActive(false); - testCaseRepository.save(inactiveTest); - - updateProblemStatement("[task][Taskname](%s,%s)".formatted(bubbleSort.getId(), inactiveTest.getId())); - - programmingExerciseTaskService.replaceTestIdsWithNames(programmingExercise); - String problemStatement = programmingExercise.getProblemStatement(); - - // inactive tests should also get replaced - assertThat(problemStatement).doesNotContain("").contains("testClass[BubbleSort]", "testName"); - } - - @Test - void testUpdateIds() { - updateProblemStatement("[task][Taskname](1,2)"); - - programmingExerciseTaskService.updateTestIds(programmingExercise, Map.of(1L, 10L, 2L, 23L)); - String problemStatement = programmingExercise.getProblemStatement(); - - assertThat(problemStatement).contains("10", "23"); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/StructuralTestCaseServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/StructuralTestCaseServiceTest.java deleted file mode 100644 index 747032881267..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/StructuralTestCaseServiceTest.java +++ /dev/null @@ -1,554 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.assertj.core.api.Assertions.assertThatExceptionOfType; - -import java.io.IOException; -import java.time.ZonedDateTime; - -import org.junit.jupiter.api.AfterEach; -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.assessment.domain.Visibility; -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationLocalCILocalVCTestBase; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTestCaseType; -import de.tum.cit.aet.artemis.programming.service.hestia.structural.StructuralSolutionEntryGenerationException; -import de.tum.cit.aet.artemis.programming.util.LocalRepository; -import de.tum.cit.aet.artemis.programming.util.ProgrammingExerciseFactory; - -/** - * Tests for the StructuralTestCaseService - * Test if solution entries are generated as expected for structural tests - */ -class StructuralTestCaseServiceTest extends AbstractProgrammingIntegrationLocalCILocalVCTestBase { - - private static final String TEST_PREFIX = "structuraltestcaseservice"; - - private final LocalRepository solutionRepo = new LocalRepository("main"); - - private final LocalRepository testRepo = new LocalRepository("main"); - - private ProgrammingExercise exercise; - - @Override - protected String getTestPrefix() { - return TEST_PREFIX; - } - - @BeforeEach - void initTestCase() { - Course course = courseUtilService.addEmptyCourse(); - userUtilService.addUsers(TEST_PREFIX, 0, 0, 0, 1); - exercise = ProgrammingExerciseFactory.generateProgrammingExercise(ZonedDateTime.now().minusDays(1), ZonedDateTime.now().plusDays(7), course); - } - - @AfterEach - void cleanup() throws IOException { - solutionRepo.resetLocalRepo(); - testRepo.resetLocalRepo(); - } - - private void addTestCaseToExercise(String name) { - var testCase = new ProgrammingExerciseTestCase(); - testCase.setTestName(name); - testCase.setExercise(exercise); - testCase.setVisibility(Visibility.ALWAYS); - testCase.setActive(true); - testCase.setWeight(1D); - testCase.setType(ProgrammingExerciseTestCaseType.STRUCTURAL); - testCaseRepository.save(testCase); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForSimpleClass() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic class Test {}", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - } - }] - """, exercise, testRepo); - addTestCaseToExercise("testClass[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("package test;\n\npublic class Test {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForSimpleClassWithoutSource() throws Exception { - exercise = hestiaUtilTestService.setupSolution("empty", "", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - } - }] - """, exercise, testRepo); - addTestCaseToExercise("testClass[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("package test;\n\npublic class Test {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForClassWithAnnotations() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n @TestA1(123) @TestA2(test=\"Test String\") public class Test {}", exercise, - solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test", - "annotations": ["TestA1", "TestA2"] - } - }] - """, exercise, testRepo); - addTestCaseToExercise("testClass[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("package test;\n\n@TestA1(value=123)\n@TestA2(test=\"Test String\")\npublic class Test {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForComplexClass() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \nprivate abstract class Test extends Test2 implements TestI1, TestI2 {}", exercise, - solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["private", "abstract"], - "package": "test", - "superclass": "Test2", - "interfaces": ["TestI1", "TestI2"] - } - }] - """, exercise, testRepo); - addTestCaseToExercise("testClass[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("package test;\n\nprivate abstract class Test extends Test2 implements TestI1, TestI2 {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForComplexClassWithoutSource() throws Exception { - exercise = hestiaUtilTestService.setupSolution("empty", "", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["private", "abstract"], - "package": "test", - "superclass": "Test2", - "interfaces": ["TestI1", "TestI2"] - } - }] - """, exercise, testRepo); - addTestCaseToExercise("testClass[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("package test;\n\nprivate abstract class Test extends Test2 implements TestI1, TestI2 {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForGenericClass() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic class Test> {}", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - } - }] - """, exercise, testRepo); - addTestCaseToExercise("testClass[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("package test;\n\npublic class Test> {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForInterface() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic interface Test {}", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test", - "isInterface": true - } - }] - """, exercise, testRepo); - addTestCaseToExercise("testClass[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("package test;\n\npublic interface Test {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForEnum() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic enum Test { CASE1, CASE2; }", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test", - "isEnum": true - }, - "enumValues": [ - "CASE1", - "CASE2" - ] - }] - """, exercise, testRepo); - addTestCaseToExercise("testClass[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("package test;\n\npublic enum Test {\n CASE1, CASE2\n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForSimpleAttribute() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic class Test {private String attributeName;}", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "attributes" : [{ - "name": "attributeName", - "modifiers": ["private"], - "type": "String" - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testAttributes[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("private String attributeName;"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForAttributeWithAnnotations() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", - "package test;\n \npublic class Test {@TestA1(123) @TestA2(test=\"Test String\") private String attributeName;}", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "attributes" : [{ - "name": "attributeName", - "modifiers": ["private"], - "type": "String", - "annotations": ["TestA1", "TestA2"] - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testAttributes[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("@TestA1(value=123)\n@TestA2(test=\"Test String\")\nprivate String attributeName;"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForSimpleAttributeWithoutSource() throws Exception { - exercise = hestiaUtilTestService.setupSolution("empty", "", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "attributes" : [{ - "name": "attributeName", - "modifiers": ["private"], - "type": "String" - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testAttributes[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("private String attributeName;"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForComplexAttribute() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic class Test {private static final List attributeName;}", exercise, - solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "attributes" : [{ - "name": "attributeName", - "modifiers": ["protected", "static", "final"], - "type": "List" - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testAttributes[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("protected static final List attributeName;"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForSimpleConstructor() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic class Test {public Test() {}}", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "constructors": [{ - "modifiers": ["public"], - "parameters": [] - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testConstructors[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("public Test() {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForConstructorWithAnnotations() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic class Test {@TestA1(123) @TestA2(test=\"Test String\") public Test() {}}", - exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "constructors": [{ - "modifiers": ["public"], - "parameters": [], - "annotations": ["TestA1", "TestA2"] - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testConstructors[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("@TestA1(value=123)\n@TestA2(test=\"Test String\")\npublic Test() {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForComplexConstructor() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic class Test {protected Test(String s1, List dates) {}}", exercise, - solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "constructors": [{ - "modifiers": ["protected"], - "parameters": ["String","List"] - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testConstructors[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("protected Test(String s1, List dates) {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForSimpleMethod() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic class Test {public void foo() {}}", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "methods": [{ - "name": "foo", - "modifiers": ["public"], - "parameters": [], - "returnType": "void" - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testMethods[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("public void foo() {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForMethodWithAnnotations() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", - "package test;\n \npublic class Test {@TestA1(123) @TestA2(test=\"Test String\") public void foo() {}}", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "methods": [{ - "name": "foo", - "modifiers": ["public"], - "parameters": [], - "returnType": "void", - "annotations": ["TestA1", "TestA2"] - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testMethods[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("@TestA1(value=123)\n@TestA2(test=\"Test String\")\npublic void foo() {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForComplexMethod() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", - "package test;\n \npublic class Test {protected static List foo(List list, String s) {}}", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "methods": [{ - "name": "foo", - "modifiers": ["protected", "static"], - "parameters": ["List", "String"], - "returnType": "List" - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testMethods[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("protected static List foo(List list, String s) {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForGenericMethod() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic class Test {public > E foo(T[] arr) {}}", exercise, - solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test.json", """ - [{ - "class": { - "name": "Test", - "modifiers": ["public"], - "package": "test" - }, - "methods": [{ - "name": "foo", - "modifiers": ["public"], - "parameters": ["Object[]"], - "returnType": "List" - }] - }] - """, exercise, testRepo); - addTestCaseToExercise("testMethods[Test]"); - - var solutionEntries = structuralTestCaseService.generateStructuralSolutionEntries(exercise); - assertThat(solutionEntries).hasSize(1); - assertThat(solutionEntries.getFirst().getFilePath()).isEqualTo("src/test/Test.java"); - assertThat(solutionEntries.getFirst().getCode()).isEqualTo("public > E foo(T[] arr) {\n \n}"); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testForMissingTestJson() throws Exception { - exercise = hestiaUtilTestService.setupSolution("src/test/Test.java", "package test;\n \npublic class Test {}", exercise, solutionRepo); - exercise = hestiaUtilTestService.setupTests("src/test/TestTest.java", "package test;\n \npublic class TestTest {}", exercise, testRepo); - addTestCaseToExercise("testClass[Test]"); - - assertThatExceptionOfType(StructuralSolutionEntryGenerationException.class).isThrownBy(() -> structuralTestCaseService.generateStructuralSolutionEntries(exercise)); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/TestwiseCoverageIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/TestwiseCoverageIntegrationTest.java deleted file mode 100644 index a9544af19d89..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/TestwiseCoverageIntegrationTest.java +++ /dev/null @@ -1,114 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.time.ZoneId; -import java.time.ZonedDateTime; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.http.HttpStatus; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationIndependentTest; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageFileReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; - -class TestwiseCoverageIntegrationTest extends AbstractProgrammingIntegrationIndependentTest { - - private static final String TEST_PREFIX = "testwisecoverageint"; - - private ProgrammingExercise programmingExercise; - - private ProgrammingSubmission latestSolutionSubmission; - - private CoverageReport latestReport; - - @BeforeEach - void setup() { - userUtilService.addUsers(TEST_PREFIX, 1, 1, 0, 0); - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, true, ProgrammingLanguage.JAVA); - programmingExercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - var solutionParticipation = solutionEntryRepository.findWithEagerResultsAndSubmissionsByProgrammingExerciseId(programmingExercise.getId()).orElseThrow(); - var unsavedPreviousSubmission = new ProgrammingSubmission(); - unsavedPreviousSubmission.setParticipation(solutionParticipation); - unsavedPreviousSubmission.setSubmissionDate(ZonedDateTime.of(2022, 4, 5, 12, 0, 0, 0, ZoneId.of("Europe/Berlin"))); - var previousSolutionSubmission = programmingSubmissionRepository.save(unsavedPreviousSubmission); - var unsavedLatestSubmission = new ProgrammingSubmission(); - unsavedLatestSubmission.setParticipation(solutionParticipation); - unsavedLatestSubmission.setSubmissionDate(ZonedDateTime.of(2022, 4, 5, 13, 0, 0, 0, ZoneId.of("Europe/Berlin"))); - latestSolutionSubmission = programmingSubmissionRepository.save(unsavedLatestSubmission); - - var testCase1 = testCaseRepository.save(new ProgrammingExerciseTestCase().exercise(programmingExercise).testName("test1()")); - var testCase2 = testCaseRepository.save(new ProgrammingExerciseTestCase().exercise(programmingExercise).testName("test2()")); - generateAndSaveSimpleReport(0.3, "src/de/tum/in/ase/BubbleSort.java", 15, 5, 1, 5, testCase1, previousSolutionSubmission); - latestReport = generateAndSaveSimpleReport(0.4, "src/de/tum/in/ase/BubbleSort.java", 20, 8, 1, 8, testCase2, latestSolutionSubmission); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void getLatestFullCoverageReportAsStudent() throws Exception { - request.get("/api/programming-exercises/" + programmingExercise.getId() + "/full-testwise-coverage-report", HttpStatus.FORBIDDEN, CoverageReport.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void getLatestFullCoverageReportAsTutor() throws Exception { - var fullReport = request.get("/api/programming-exercises/" + programmingExercise.getId() + "/full-testwise-coverage-report", HttpStatus.OK, CoverageReport.class); - assertThat(fullReport.getCoveredLineRatio()).isEqualTo(latestReport.getCoveredLineRatio()); - var fileReports = fullReport.getFileReports(); - assertThat(fileReports).hasSize(1); - var fileReport = fileReports.stream().toList().getFirst(); - assertThat(fileReport.getFilePath()).isEqualTo("src/de/tum/in/ase/BubbleSort.java"); - assertThat(fileReport.getCoveredLineCount()).isEqualTo(8); - assertThat(fileReport.getLineCount()).isEqualTo(20); - // the latest coverage report only contains one file report - assertThat(fileReport.getTestwiseCoverageEntries()) - .containsExactlyElementsOf(latestReport.getFileReports().stream().flatMap(report -> report.getTestwiseCoverageEntries().stream()).toList()); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void getLatestCoverageReportAsStudent() throws Exception { - request.get("/api/programming-exercises/" + programmingExercise.getId() + "/testwise-coverage-report", HttpStatus.FORBIDDEN, CoverageReport.class); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") - void getLatestCoverageReportAsTutor() throws Exception { - var report = request.get("/api/programming-exercises/" + programmingExercise.getId() + "/testwise-coverage-report", HttpStatus.OK, CoverageReport.class); - assertThat(report.getFileReports()).isNull(); - assertThat(report.getCoveredLineRatio()).isEqualTo(0.4); - } - - private CoverageReport generateAndSaveSimpleReport(double coveredLineRatio, String filePath, Integer fileLineCount, Integer coveredLineCount, Integer startLine, - Integer lineCount, ProgrammingExerciseTestCase testCase, ProgrammingSubmission submission) { - var unsavedLatestReport = new CoverageReport(); - unsavedLatestReport.setSubmission(latestSolutionSubmission); - unsavedLatestReport.setCoveredLineRatio(coveredLineRatio); - unsavedLatestReport.setSubmission(submission); - var resultReport = coverageReportRepository.save(unsavedLatestReport); - - var unsavedLatestFileReport = new CoverageFileReport(); - unsavedLatestFileReport.setFilePath(filePath); - unsavedLatestFileReport.setLineCount(fileLineCount); - unsavedLatestFileReport.setCoveredLineCount(coveredLineCount); - unsavedLatestFileReport.setFullReport(unsavedLatestReport); - coverageFileReportRepository.save(unsavedLatestFileReport); - - var unsavedLatestEntry = new TestwiseCoverageReportEntry(); - unsavedLatestEntry.setStartLine(startLine); - unsavedLatestEntry.setLineCount(lineCount); - unsavedLatestEntry.setTestCase(testCase); - unsavedLatestEntry.setFileReport(unsavedLatestFileReport); - testwiseCoverageReportEntryRepository.save(unsavedLatestEntry); - - return coverageReportRepository.findCoverageReportByIdWithEagerFileReportsAndEntriesElseThrow(resultReport.getId()); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/TestwiseCoverageReportServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/TestwiseCoverageReportServiceTest.java deleted file mode 100644 index 18addd11453e..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/TestwiseCoverageReportServiceTest.java +++ /dev/null @@ -1,102 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.io.IOException; -import java.util.Map; -import java.util.Set; - -import org.junit.jupiter.api.AfterEach; -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.data.domain.Pageable; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationLocalCILocalVCTestBase; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; -import de.tum.cit.aet.artemis.programming.hestia.util.TestwiseCoverageTestUtil; -import de.tum.cit.aet.artemis.programming.util.LocalRepository; - -class TestwiseCoverageReportServiceTest extends AbstractProgrammingIntegrationLocalCILocalVCTestBase { - - private static final String TEST_PREFIX = "testwisecoveragereportservice"; - - private ProgrammingExercise programmingExercise; - - private ProgrammingSubmission solutionSubmission; - - private final LocalRepository solutionRepo = new LocalRepository("main"); - - @Override - protected String getTestPrefix() { - return TEST_PREFIX; - } - - @BeforeEach - void setup() throws Exception { - userUtilService.addUsers(TEST_PREFIX, 1, 0, 0, 1); - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, true, ProgrammingLanguage.JAVA); - programmingExercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - - programmingExercise = hestiaUtilTestService.setupSolution( - Map.ofEntries(Map.entry("src/de/tum/in/ase/BubbleSort.java", "\n ".repeat(28)), Map.entry("src/de/tum/in/ase/Context.java", "\n ".repeat(18))), programmingExercise, - solutionRepo); - - var testCase1 = new ProgrammingExerciseTestCase().testName("test1()").exercise(programmingExercise).active(true).weight(1.0); - testCaseRepository.save(testCase1); - var testCase2 = new ProgrammingExerciseTestCase().testName("test2()").exercise(programmingExercise).active(true).weight(1.0); - testCaseRepository.save(testCase2); - var solutionParticipation = solutionProgrammingExerciseRepository.findWithEagerResultsAndSubmissionsByProgrammingExerciseId(programmingExercise.getId()).orElseThrow(); - solutionSubmission = programmingExerciseUtilService.createProgrammingSubmission(solutionParticipation, false); - programmingExercise = programmingExerciseRepository.findByIdElseThrow(programmingExercise.getId()); - } - - @AfterEach - void cleanup() throws IOException { - solutionRepo.resetLocalRepo(); - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void shouldCreateFullTestwiseCoverageReport() { - var fileReportsByTestName = TestwiseCoverageTestUtil.generateCoverageFileReportByTestName(); - testwiseCoverageService.createTestwiseCoverageReport(fileReportsByTestName, programmingExercise, solutionSubmission); - - var reports = coverageReportRepository.getLatestCoverageReportsWithLegalSubmissionsForProgrammingExercise(programmingExercise.getId(), Pageable.ofSize(1)); - assertThat(reports).hasSize(1); - var report = reports.getFirst(); - // 18/50 lines covered = 32% - assertThat(report.getCoveredLineRatio()).isEqualTo(0.32); - - var testCases = testCaseRepository.findByExerciseId(programmingExercise.getId()); - var testCase1 = testCases.stream().filter(testCase -> "test1()".equals(testCase.getTestName())).findFirst().orElseThrow(); - var testCase2 = testCases.stream().filter(testCase -> "test2()".equals(testCase.getTestName())).findFirst().orElseThrow(); - - var optionalFullReportWithFileReports = coverageReportRepository.findCoverageReportByIdWithEagerFileReportsAndEntries(report.getId()); - assertThat(optionalFullReportWithFileReports).isPresent(); - var fullReportWithFileReports = optionalFullReportWithFileReports.get(); - var fileReports = fullReportWithFileReports.getFileReports(); - assertThat(fileReports).hasSize(2); - - var bubbleSortFileReport = fileReports.stream().filter(fileReport -> "src/de/tum/in/ase/BubbleSort.java".equals(fileReport.getFilePath())).findFirst().orElseThrow(); - var entriesBubbleSort = bubbleSortFileReport.getTestwiseCoverageEntries(); - assertThat(entriesBubbleSort).hasSize(4); - checkIfSetContainsEntry(entriesBubbleSort, 15, 3, testCase1); - checkIfSetContainsEntry(entriesBubbleSort, 23, 1, testCase1); - checkIfSetContainsEntry(entriesBubbleSort, 2, 1, testCase2); - checkIfSetContainsEntry(entriesBubbleSort, 16, 3, testCase2); - - var contextFileReport = fileReports.stream().filter(fileReport -> "src/de/tum/in/ase/Context.java".equals(fileReport.getFilePath())).findFirst().orElseThrow(); - assertThat(contextFileReport.getTestwiseCoverageEntries()).hasSize(1); - checkIfSetContainsEntry(contextFileReport.getTestwiseCoverageEntries(), 1, 10, testCase2); - } - - private void checkIfSetContainsEntry(Set set, Integer startLine, Integer lineCount, ProgrammingExerciseTestCase testCase) { - assertThat(set).anyMatch(entry -> startLine.equals(entry.getStartLine()) && lineCount.equals(entry.getLineCount()) && testCase.equals(entry.getTestCase())); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/AddUncoveredLinesAsPotentialCodeBlocksTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/AddUncoveredLinesAsPotentialCodeBlocksTest.java deleted file mode 100644 index 30ab4619e68e..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/AddUncoveredLinesAsPotentialCodeBlocksTest.java +++ /dev/null @@ -1,122 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.ArrayList; -import java.util.List; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.AddUncoveredLinesAsPotentialCodeBlocks; - -class AddUncoveredLinesAsPotentialCodeBlocksTest { - - private AddUncoveredLinesAsPotentialCodeBlocks addPotentialCodeBlocks; - - private GroupedFile groupedFile; - - @BeforeEach - void initBlackboard() { - BehavioralBlackboard blackboard = new BehavioralBlackboard(null, null, null); - var groupedFiles = new ArrayList(); - blackboard.setGroupedFiles(groupedFiles); - groupedFile = new GroupedFile("test.java", null, null, null); - groupedFile.setCommonChanges(List.of(new GroupedFile.ChangeBlock(List.of(3, 4)))); - groupedFiles.add(groupedFile); - - addPotentialCodeBlocks = new AddUncoveredLinesAsPotentialCodeBlocks(blackboard); - } - - @Test - void testNoAction() { - assertThat(addPotentialCodeBlocks.executeCondition()).isFalse(); - } - - @Test - void testAddNoBlocks() { - groupedFile.setFileContent(""" - A - B - C - D - E - F - """); - - assertThat(addPotentialCodeBlocks.executeCondition()).isTrue(); - assertThat(addPotentialCodeBlocks.executeAction()).isFalse(); - assertThat(groupedFile.getCommonChanges()).isNotNull().containsExactly(new GroupedFile.ChangeBlock(List.of(3, 4))); - } - - @Test - void testAddPrefix() { - groupedFile.setFileContent(""" - - - C - D - E - F - """); - - assertThat(addPotentialCodeBlocks.executeCondition()).isTrue(); - assertThat(addPotentialCodeBlocks.executeAction()).isTrue(); - assertThat(groupedFile.getCommonChanges()).isNotNull().containsExactlyInAnyOrder(new GroupedFile.ChangeBlock(List.of(3, 4)), - new GroupedFile.ChangeBlock(List.of(1, 2), true)); - } - - @Test - void testAddPostfix() { - groupedFile.setFileContent(""" - A - B - C - D - - - X - """); - - assertThat(addPotentialCodeBlocks.executeCondition()).isTrue(); - assertThat(addPotentialCodeBlocks.executeAction()).isTrue(); - assertThat(groupedFile.getCommonChanges()).isNotNull().containsExactlyInAnyOrder(new GroupedFile.ChangeBlock(List.of(3, 4)), - new GroupedFile.ChangeBlock(List.of(5, 6), true)); - } - - @Test - void testIncludesCurlyBraces() { - groupedFile.setFileContent(""" - A - } - C - D - { - F - """); - - assertThat(addPotentialCodeBlocks.executeCondition()).isTrue(); - assertThat(addPotentialCodeBlocks.executeAction()).isTrue(); - assertThat(groupedFile.getCommonChanges()).isNotNull().containsExactlyInAnyOrder(new GroupedFile.ChangeBlock(List.of(3, 4)), new GroupedFile.ChangeBlock(List.of(2), true), - new GroupedFile.ChangeBlock(List.of(5), true)); - } - - @Test - void testIncludesElseStatements() { - groupedFile.setFileContent(""" - else - } else { - C - D - } else - else { - """); - - assertThat(addPotentialCodeBlocks.executeCondition()).isTrue(); - assertThat(addPotentialCodeBlocks.executeAction()).isTrue(); - assertThat(groupedFile.getCommonChanges()).isNotNull().containsExactlyInAnyOrder(new GroupedFile.ChangeBlock(List.of(3, 4)), - new GroupedFile.ChangeBlock(List.of(1, 2), true), new GroupedFile.ChangeBlock(List.of(5, 6), true)); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/BehavioralTestCaseServiceLocalCILocalVCTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/BehavioralTestCaseServiceLocalCILocalVCTest.java deleted file mode 100644 index d4de36c789e9..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/BehavioralTestCaseServiceLocalCILocalVCTest.java +++ /dev/null @@ -1,149 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.io.IOException; -import java.util.HashSet; - -import org.junit.jupiter.api.AfterEach; -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import org.springframework.security.test.context.support.WithMockUser; - -import de.tum.cit.aet.artemis.assessment.domain.Visibility; -import de.tum.cit.aet.artemis.core.domain.Course; -import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationLocalCILocalVCTestBase; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageFileReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTestCaseType; -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; -import de.tum.cit.aet.artemis.programming.util.LocalRepository; - -class BehavioralTestCaseServiceLocalCILocalVCTest extends AbstractProgrammingIntegrationLocalCILocalVCTestBase { - - private static final String TEST_PREFIX = "behavioraltestcastservice"; - - private final LocalRepository solutionRepo = new LocalRepository("main"); - - private ProgrammingExercise exercise; - - @Override - protected String getTestPrefix() { - return TEST_PREFIX; - } - - @BeforeEach - void initTestCase() { - userUtilService.addUsers(TEST_PREFIX, 0, 0, 0, 1); - final Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, true, ProgrammingLanguage.JAVA); - exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); - exercise.getBuildConfig().setTestwiseCoverageEnabled(true); - } - - @AfterEach - void cleanup() throws IOException { - solutionRepo.resetLocalRepo(); - } - - private ProgrammingExerciseTestCase addTestCaseToExercise(String name) { - var testCase = new ProgrammingExerciseTestCase(); - testCase.setTestName(name); - testCase.setExercise(exercise); - testCase.setVisibility(Visibility.ALWAYS); - testCase.setActive(true); - testCase.setWeight(1D); - testCase.setType(ProgrammingExerciseTestCaseType.BEHAVIORAL); - return testCaseRepository.save(testCase); - } - - private ProgrammingExerciseGitDiffReport newGitDiffReport() { - var gitDiffReport = new ProgrammingExerciseGitDiffReport(); - gitDiffReport.setEntries(new HashSet<>()); - gitDiffReport.setProgrammingExercise(exercise); - gitDiffReport.setSolutionRepositoryCommitHash("123a"); - gitDiffReport.setTemplateRepositoryCommitHash("123b"); - gitDiffReport = reportRepository.save(gitDiffReport); - return gitDiffReport; - } - - private ProgrammingExerciseGitDiffReport addGitDiffEntry(String filePath, int startLine, int lineCount, ProgrammingExerciseGitDiffReport gitDiffReport) { - var gitDiffEntry = new ProgrammingExerciseGitDiffEntry(); - gitDiffEntry.setFilePath(filePath); - gitDiffEntry.setStartLine(startLine); - gitDiffEntry.setLineCount(lineCount); - gitDiffEntry.setGitDiffReport(gitDiffReport); - gitDiffReport.getEntries().add(gitDiffEntry); - return reportRepository.save(gitDiffReport); - } - - private CoverageReport newCoverageReport() { - var solutionParticipation = solutionProgrammingExerciseRepository.findWithEagerResultsAndSubmissionsByProgrammingExerciseId(exercise.getId()).orElseThrow(); - var solutionSubmission = programmingExerciseUtilService.createProgrammingSubmission(solutionParticipation, false); - - var coverageReport = new CoverageReport(); - coverageReport.setFileReports(new HashSet<>()); - coverageReport.setSubmission(solutionSubmission); - coverageReport = coverageReportRepository.save(coverageReport); - return coverageReport; - } - - private CoverageFileReport newCoverageFileReport(String filePath, CoverageReport coverageReport) { - var coverageFileReport = new CoverageFileReport(); - coverageFileReport.setFilePath(filePath); - coverageFileReport.setTestwiseCoverageEntries(new HashSet<>()); - coverageFileReport.setFullReport(coverageReport); - coverageFileReport = coverageFileReportRepository.save(coverageFileReport); - coverageReport.getFileReports().add(coverageFileReport); - return coverageFileReport; - } - - private TestwiseCoverageReportEntry newCoverageReportEntry(int startLine, int lineCount, ProgrammingExerciseTestCase testCase, CoverageFileReport coverageFileReport) { - var coverageReportEntry = new TestwiseCoverageReportEntry(); - coverageReportEntry.setTestCase(testCase); - coverageReportEntry.setStartLine(startLine); - coverageReportEntry.setLineCount(lineCount); - coverageReportEntry.setFileReport(coverageFileReport); - coverageReportEntry = testwiseCoverageReportEntryRepository.save(coverageReportEntry); - coverageFileReport.getTestwiseCoverageEntries().add(coverageReportEntry); - return coverageReportEntry; - } - - @Test - @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") - void testGenerationForSimpleExample() throws Exception { - exercise = hestiaUtilTestService.setupSolution("Test.java", "A\nB\nC\nD\nE\nF\nG\nH", exercise, solutionRepo); - var testCase = addTestCaseToExercise("testCase"); - - var gitDiffReport = newGitDiffReport(); - addGitDiffEntry("Test.java", 2, 7, gitDiffReport); - - var coverageReport = newCoverageReport(); - var coverageFileReport = newCoverageFileReport("Test.java", coverageReport); - newCoverageReportEntry(1, 3, testCase, coverageFileReport); - newCoverageReportEntry(5, 2, testCase, coverageFileReport); - - var solutionEntries = behavioralTestCaseService.generateBehavioralSolutionEntries(exercise); - - var expected1 = new ProgrammingExerciseSolutionEntry(); - expected1.setId(0L); - expected1.setFilePath("Test.java"); - expected1.setTestCase(testCase); - expected1.setLine(2); - expected1.setCode("B\nC"); - var expected2 = new ProgrammingExerciseSolutionEntry(); - expected2.setId(0L); - expected2.setFilePath("Test.java"); - expected2.setTestCase(testCase); - expected2.setLine(5); - expected2.setCode("E\nF"); - assertThat(solutionEntries).isNotNull(); - solutionEntries.forEach(solutionEntry -> solutionEntry.setId(0L)); - assertThat(solutionEntries).containsExactlyInAnyOrder(expected1, expected2); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CombineChangeBlocksTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CombineChangeBlocksTest.java deleted file mode 100644 index a9c8e9689bd7..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CombineChangeBlocksTest.java +++ /dev/null @@ -1,84 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.ArrayList; -import java.util.List; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralSolutionEntryGenerationException; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile.ChangeBlock; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.CombineChangeBlocks; - -class CombineChangeBlocksTest { - - private CombineChangeBlocks combineChangeBlocks; - - private GroupedFile groupedFile; - - @BeforeEach - void initBlackboard() { - BehavioralBlackboard blackboard = new BehavioralBlackboard(null, null, null); - var groupedFiles = new ArrayList(); - blackboard.setGroupedFiles(groupedFiles); - groupedFile = new GroupedFile("test.java", null, null, null); - groupedFiles.add(groupedFile); - - combineChangeBlocks = new CombineChangeBlocks(blackboard); - } - - @Test - void testNoCombining() throws BehavioralSolutionEntryGenerationException { - var changeBlocks = new ChangeBlock[] { new ChangeBlock(List.of(1)), new ChangeBlock(List.of(3)), new ChangeBlock(List.of(5)) }; - groupedFile.setCommonChanges(List.of(changeBlocks)); - - assertThat(combineChangeBlocks.executeCondition()).isTrue(); - assertThat(combineChangeBlocks.executeAction()).isFalse(); - assertThat(groupedFile.getCommonChanges()).containsExactly(changeBlocks); - } - - @Test - void testOneCombination1() throws BehavioralSolutionEntryGenerationException { - var changeBlocks = new ChangeBlock[] { new ChangeBlock(List.of(2)), new ChangeBlock(List.of(3)) }; - groupedFile.setCommonChanges(List.of(changeBlocks)); - - assertThat(combineChangeBlocks.executeCondition()).isTrue(); - assertThat(combineChangeBlocks.executeAction()).isTrue(); - assertThat(groupedFile.getCommonChanges()).containsExactly(new ChangeBlock(List.of(2, 3))); - } - - @Test - void testOneCombination2() throws BehavioralSolutionEntryGenerationException { - var changeBlocks = new ChangeBlock[] { new ChangeBlock(List.of(1, 2, 3, 4)), new ChangeBlock(List.of(3, 4, 5, 6, 7, 8)) }; - groupedFile.setCommonChanges(List.of(changeBlocks)); - - assertThat(combineChangeBlocks.executeCondition()).isTrue(); - assertThat(combineChangeBlocks.executeAction()).isTrue(); - assertThat(groupedFile.getCommonChanges()).containsExactly(new ChangeBlock(List.of(1, 2, 3, 4, 5, 6, 7, 8))); - } - - @Test - void testCombineManyIntoOne1() throws BehavioralSolutionEntryGenerationException { - var changeBlocks = new ChangeBlock[] { new ChangeBlock(List.of(2)), new ChangeBlock(List.of(3)), new ChangeBlock(List.of(4)), new ChangeBlock(List.of(5)) }; - groupedFile.setCommonChanges(List.of(changeBlocks)); - - assertThat(combineChangeBlocks.executeCondition()).isTrue(); - assertThat(combineChangeBlocks.executeAction()).isTrue(); - assertThat(groupedFile.getCommonChanges()).containsExactly(new ChangeBlock(List.of(2, 3, 4, 5))); - } - - @Test - void testCombineManyIntoOne2() throws BehavioralSolutionEntryGenerationException { - var changeBlocks = new ChangeBlock[] { new ChangeBlock(List.of(1, 2, 3)), new ChangeBlock(List.of(3, 4, 5)), new ChangeBlock(List.of(4, 5, 6)), - new ChangeBlock(List.of(7, 8, 9)) }; - groupedFile.setCommonChanges(List.of(changeBlocks)); - - assertThat(combineChangeBlocks.executeCondition()).isTrue(); - assertThat(combineChangeBlocks.executeAction()).isTrue(); - assertThat(groupedFile.getCommonChanges()).containsExactly(new ChangeBlock(List.of(1, 2, 3, 4, 5, 6, 7, 8, 9))); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CreateCommonChangeBlocksTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CreateCommonChangeBlocksTest.java deleted file mode 100644 index 40feb7c133be..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CreateCommonChangeBlocksTest.java +++ /dev/null @@ -1,58 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.ArrayList; -import java.util.Collections; -import java.util.List; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.CreateCommonChangeBlocks; - -class CreateCommonChangeBlocksTest { - - private CreateCommonChangeBlocks createCommonChangeBlocks; - - private GroupedFile groupedFile; - - @BeforeEach - void initBlackboard() { - BehavioralBlackboard blackboard = new BehavioralBlackboard(null, null, null); - var groupedFiles = new ArrayList(); - blackboard.setGroupedFiles(groupedFiles); - groupedFile = new GroupedFile("test.java", null, null, null); - groupedFiles.add(groupedFile); - - createCommonChangeBlocks = new CreateCommonChangeBlocks(blackboard); - } - - @Test - void testNoAction() { - groupedFile.setCommonLines(Collections.emptyList()); - groupedFile.setCommonChanges(Collections.emptyList()); - assertThat(createCommonChangeBlocks.executeCondition()).isFalse(); - } - - @Test - void testCreateOneChangeBlock() { - groupedFile.setCommonLines(Set.of(1, 2, 3)); - - assertThat(createCommonChangeBlocks.executeCondition()).isTrue(); - assertThat(createCommonChangeBlocks.executeAction()).isTrue(); - assertThat(groupedFile.getCommonChanges()).containsExactly(new GroupedFile.ChangeBlock(List.of(1, 2, 3))); - } - - @Test - void testCreateTwoChangeBlocks() { - groupedFile.setCommonLines(Set.of(1, 2, 4, 5)); - - assertThat(createCommonChangeBlocks.executeCondition()).isTrue(); - assertThat(createCommonChangeBlocks.executeAction()).isTrue(); - assertThat(groupedFile.getCommonChanges()).containsExactly(new GroupedFile.ChangeBlock(List.of(1, 2)), new GroupedFile.ChangeBlock(List.of(4, 5))); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CreateSolutionEntriesTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CreateSolutionEntriesTest.java deleted file mode 100644 index 20da41c8c3aa..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/CreateSolutionEntriesTest.java +++ /dev/null @@ -1,106 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.ArrayList; -import java.util.List; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.CreateSolutionEntries; - -class CreateSolutionEntriesTest { - - private BehavioralBlackboard blackboard; - - private CreateSolutionEntries createSolutionEntries; - - private GroupedFile groupedFile; - - @BeforeEach - void initBlackboard() { - blackboard = new BehavioralBlackboard(null, null, null); - var groupedFiles = new ArrayList(); - blackboard.setGroupedFiles(groupedFiles); - groupedFile = new GroupedFile("test.java", new ProgrammingExerciseTestCase(), null, null); - groupedFiles.add(groupedFile); - - createSolutionEntries = new CreateSolutionEntries(blackboard); - } - - @Test - void testNoAction() { - assertThat(createSolutionEntries.executeCondition()).isFalse(); - } - - @Test - void testNoChangesOnSecondCall() { - groupedFile.setFileContent("A\nB\nC\nD"); - groupedFile.setCommonChanges(List.of(new GroupedFile.ChangeBlock(List.of(2, 3), false))); - - assertThat(createSolutionEntries.executeCondition()).isTrue(); - assertThat(createSolutionEntries.executeAction()).isTrue(); - var solutionEntries = blackboard.getSolutionEntries(); - assertThat(createSolutionEntries.executeAction()).isFalse(); - assertThat(blackboard.getSolutionEntries()).isSameAs(solutionEntries); - } - - @Test - void testCreateOneSolutionEntry() { - groupedFile.setFileContent("A\nB\nC\nD"); - groupedFile.setCommonChanges(List.of(new GroupedFile.ChangeBlock(List.of(2, 3), false))); - - assertThat(createSolutionEntries.executeCondition()).isTrue(); - assertThat(createSolutionEntries.executeAction()).isTrue(); - var expected = new ProgrammingExerciseSolutionEntry(); - expected.setId(0L); - expected.setFilePath("test.java"); - expected.setTestCase(groupedFile.getTestCase()); - expected.setLine(2); - expected.setCode("B\nC"); - assertThat(blackboard.getSolutionEntries()).isNotNull().containsExactly(expected); - } - - @Test - void testCreateOneSolutionEntryIgnoringPotential() { - groupedFile.setFileContent("A\nB\nC\nD\nE"); - groupedFile.setCommonChanges(List.of(new GroupedFile.ChangeBlock(List.of(2, 3), false), new GroupedFile.ChangeBlock(List.of(4, 5), true))); - - assertThat(createSolutionEntries.executeCondition()).isTrue(); - assertThat(createSolutionEntries.executeAction()).isTrue(); - var expected = new ProgrammingExerciseSolutionEntry(); - expected.setId(0L); - expected.setFilePath("test.java"); - expected.setTestCase(groupedFile.getTestCase()); - expected.setLine(2); - expected.setCode("B\nC"); - assertThat(blackboard.getSolutionEntries()).isNotNull().containsExactly(expected); - } - - @Test - void testCreateTwoSolutionEntries() { - groupedFile.setFileContent("A\nB\nC\nD\nE"); - groupedFile.setCommonChanges(List.of(new GroupedFile.ChangeBlock(List.of(2, 3), false), new GroupedFile.ChangeBlock(List.of(4, 5), false))); - - assertThat(createSolutionEntries.executeCondition()).isTrue(); - assertThat(createSolutionEntries.executeAction()).isTrue(); - var expected1 = new ProgrammingExerciseSolutionEntry(); - expected1.setId(0L); - expected1.setFilePath("test.java"); - expected1.setTestCase(groupedFile.getTestCase()); - expected1.setLine(2); - expected1.setCode("B\nC"); - var expected2 = new ProgrammingExerciseSolutionEntry(); - expected2.setId(0L); - expected2.setFilePath("test.java"); - expected2.setTestCase(groupedFile.getTestCase()); - expected2.setLine(4); - expected2.setCode("D\nE"); - assertThat(blackboard.getSolutionEntries()).isNotNull().containsExactlyInAnyOrder(expected1, expected2); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/DropRemovedGitDiffEntriesTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/DropRemovedGitDiffEntriesTest.java deleted file mode 100644 index 450203b02931..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/DropRemovedGitDiffEntriesTest.java +++ /dev/null @@ -1,54 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.HashSet; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralSolutionEntryGenerationException; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.DropRemovedGitDiffEntries; - -class DropRemovedGitDiffEntriesTest { - - private BehavioralBlackboard blackboard; - - private ProgrammingExerciseGitDiffReport gitDiffReport; - - private DropRemovedGitDiffEntries dropRemovedGitDiffEntries; - - @BeforeEach - void setup() { - gitDiffReport = new ProgrammingExerciseGitDiffReport(); - gitDiffReport.setEntries(new HashSet<>()); - - blackboard = new BehavioralBlackboard(gitDiffReport, null, null); - dropRemovedGitDiffEntries = new DropRemovedGitDiffEntries(blackboard); - - } - - @Test - void testNoAction() { - assertThat(dropRemovedGitDiffEntries.executeCondition()).isFalse(); - } - - @Test - void testExecuteCondition() throws BehavioralSolutionEntryGenerationException { - var removedEntry = new ProgrammingExerciseGitDiffEntry(); - var addedEntry = new ProgrammingExerciseGitDiffEntry(); - addedEntry.setStartLine(1); - addedEntry.setLineCount(2); - gitDiffReport.setEntries(Set.of(removedEntry, addedEntry)); - - assertThat(dropRemovedGitDiffEntries.executeCondition()).isTrue(); - assertThat(dropRemovedGitDiffEntries.executeAction()).isTrue(); - var remainingEntries = blackboard.getGitDiffReport().getEntries(); - assertThat(dropRemovedGitDiffEntries.executeCondition()).isFalse(); - assertThat(remainingEntries).containsExactly(addedEntry); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/ExtractChangedLinesTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/ExtractChangedLinesTest.java deleted file mode 100644 index 016c4f5b6f21..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/ExtractChangedLinesTest.java +++ /dev/null @@ -1,71 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.ArrayList; -import java.util.Collections; -import java.util.HashSet; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.ExtractChangedLines; - -class ExtractChangedLinesTest { - - private ExtractChangedLines extractChangedLines; - - private GroupedFile groupedFile; - - private Set gitDiffEntries; - - @BeforeEach - void initBlackboard() { - BehavioralBlackboard blackboard = new BehavioralBlackboard(null, null, null); - var groupedFiles = new ArrayList(); - blackboard.setGroupedFiles(groupedFiles); - gitDiffEntries = new HashSet<>(); - groupedFile = new GroupedFile("test.java", null, gitDiffEntries, null); - groupedFiles.add(groupedFile); - - extractChangedLines = new ExtractChangedLines(blackboard); - } - - @Test - void testNoAction() { - groupedFile.setChangedLines(Collections.emptySet()); - assertThat(extractChangedLines.executeCondition()).isFalse(); - } - - @Test - void testExtractChangedLines1() { - var gitDiffEntry = new ProgrammingExerciseGitDiffEntry(); - gitDiffEntry.setStartLine(3); - gitDiffEntry.setLineCount(2); - gitDiffEntries.add(gitDiffEntry); - - assertThat(extractChangedLines.executeCondition()).isTrue(); - assertThat(extractChangedLines.executeAction()).isTrue(); - assertThat(groupedFile.getChangedLines()).containsExactly(3, 4); - } - - @Test - void testExtractChangedLines2() { - var gitDiffEntry1 = new ProgrammingExerciseGitDiffEntry(); - gitDiffEntry1.setStartLine(3); - gitDiffEntry1.setLineCount(2); - var gitDiffEntry2 = new ProgrammingExerciseGitDiffEntry(); - gitDiffEntry2.setStartLine(6); - gitDiffEntry2.setLineCount(3); - gitDiffEntries.add(gitDiffEntry1); - gitDiffEntries.add(gitDiffEntry2); - - assertThat(extractChangedLines.executeCondition()).isTrue(); - assertThat(extractChangedLines.executeAction()).isTrue(); - assertThat(groupedFile.getChangedLines()).containsExactly(3, 4, 6, 7, 8); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/ExtractCoveredLinesTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/ExtractCoveredLinesTest.java deleted file mode 100644 index 2be7206a11ae..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/ExtractCoveredLinesTest.java +++ /dev/null @@ -1,71 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.ArrayList; -import java.util.Collections; -import java.util.HashSet; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.ExtractCoveredLines; - -class ExtractCoveredLinesTest { - - private ExtractCoveredLines extractCoveredLines; - - private GroupedFile groupedFile; - - private Set coverageReportEntries; - - @BeforeEach - void initBlackboard() { - BehavioralBlackboard blackboard = new BehavioralBlackboard(null, null, null); - var groupedFiles = new ArrayList(); - blackboard.setGroupedFiles(groupedFiles); - coverageReportEntries = new HashSet<>(); - groupedFile = new GroupedFile("test.java", null, null, coverageReportEntries); - groupedFiles.add(groupedFile); - - extractCoveredLines = new ExtractCoveredLines(blackboard); - } - - @Test - void testNoAction() { - groupedFile.setCoveredLines(Collections.emptySet()); - assertThat(extractCoveredLines.executeCondition()).isFalse(); - } - - @Test - void testExtractChangedLines1() { - var reportEntry = new TestwiseCoverageReportEntry(); - reportEntry.setStartLine(3); - reportEntry.setLineCount(2); - coverageReportEntries.add(reportEntry); - - assertThat(extractCoveredLines.executeCondition()).isTrue(); - assertThat(extractCoveredLines.executeAction()).isTrue(); - assertThat(groupedFile.getCoveredLines()).containsExactly(3, 4); - } - - @Test - void testExtractChangedLines2() { - var reportEntry1 = new TestwiseCoverageReportEntry(); - reportEntry1.setStartLine(3); - reportEntry1.setLineCount(2); - var reportEntry2 = new TestwiseCoverageReportEntry(); - reportEntry2.setStartLine(6); - reportEntry2.setLineCount(3); - coverageReportEntries.add(reportEntry1); - coverageReportEntries.add(reportEntry2); - - assertThat(extractCoveredLines.executeCondition()).isTrue(); - assertThat(extractCoveredLines.executeAction()).isTrue(); - assertThat(groupedFile.getCoveredLines()).containsExactly(3, 4, 6, 7, 8); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/FindCommonLinesTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/FindCommonLinesTest.java deleted file mode 100644 index e36d44be008a..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/FindCommonLinesTest.java +++ /dev/null @@ -1,56 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.ArrayList; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.FindCommonLines; - -class FindCommonLinesTest { - - private FindCommonLines findCommonLines; - - private GroupedFile groupedFile; - - @BeforeEach - void initBlackboard() { - BehavioralBlackboard blackboard = new BehavioralBlackboard(null, null, null); - var groupedFiles = new ArrayList(); - blackboard.setGroupedFiles(groupedFiles); - groupedFile = new GroupedFile("test.java", null, null, null); - groupedFiles.add(groupedFile); - - findCommonLines = new FindCommonLines(blackboard); - } - - @Test - void testNoAction() { - assertThat(findCommonLines.executeCondition()).isFalse(); - } - - @Test - void testFindCommonLines() { - groupedFile.setCoveredLines(Set.of(1, 2, 3)); - groupedFile.setChangedLines(Set.of(2, 3, 4)); - - assertThat(findCommonLines.executeCondition()).isTrue(); - assertThat(findCommonLines.executeAction()).isTrue(); - assertThat(groupedFile.getCommonLines()).containsExactly(2, 3); - } - - @Test - void testFindNoCommonLines() { - groupedFile.setCoveredLines(Set.of(1, 2, 3)); - groupedFile.setChangedLines(Set.of(4, 5, 6)); - - assertThat(findCommonLines.executeCondition()).isTrue(); - assertThat(findCommonLines.executeAction()).isFalse(); - assertThat(groupedFile.getCommonLines()).isEmpty(); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/GroupGitDiffAndCoverageEntriesByFilePathAndTestCaseTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/GroupGitDiffAndCoverageEntriesByFilePathAndTestCaseTest.java deleted file mode 100644 index 1e624e5b130b..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/GroupGitDiffAndCoverageEntriesByFilePathAndTestCaseTest.java +++ /dev/null @@ -1,117 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; - -import java.util.Collections; -import java.util.HashSet; -import java.util.Set; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageFileReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseGitDiffReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.GroupGitDiffAndCoverageEntriesByFilePathAndTestCase; - -class GroupGitDiffAndCoverageEntriesByFilePathAndTestCaseTest { - - private BehavioralBlackboard blackboard; - - private GroupGitDiffAndCoverageEntriesByFilePathAndTestCase grouper; - - @BeforeEach - void initBlackboard() { - blackboard = new BehavioralBlackboard(new ProgrammingExerciseGitDiffReport(), new CoverageReport(), null); - blackboard.getGitDiffReport().setEntries(new HashSet<>()); - blackboard.getCoverageReport().setFileReports(new HashSet<>()); - - grouper = new GroupGitDiffAndCoverageEntriesByFilePathAndTestCase(blackboard); - } - - @Test - void testNoAction() { - blackboard.setGroupedFiles(Collections.emptyList()); - assertThat(grouper.executeCondition()).isFalse(); - } - - @Test - void testNoFiles() { - assertThat(grouper.executeCondition()).isTrue(); - assertThat(grouper.executeAction()).isFalse(); - } - - @Test - void testGroupSingleFileAndTest() { - var testCase = new ProgrammingExerciseTestCase(); - var gitDiffEntry1 = addGitDiffEntry("test.java", 1, 10); - var gitDiffEntry2 = addGitDiffEntry("test.java", 12, 6); - var coverageEntry1 = addCoverageEntry("test.java", 5, 5, testCase); - var coverageEntry2 = addCoverageEntry("test.java", 15, 2, testCase); - - assertThat(grouper.executeCondition()).isTrue(); - assertThat(grouper.executeAction()).isTrue(); - assertThat(blackboard.getGroupedFiles()).isNotNull().hasSize(1) - .containsExactly(new GroupedFile("test.java", testCase, Set.of(gitDiffEntry1, gitDiffEntry2), Set.of(coverageEntry1, coverageEntry2))); - } - - @Test - void testGroupTwoFilesSingleTest() { - var testCase = new ProgrammingExerciseTestCase(); - var gitDiffEntry1 = addGitDiffEntry("test.java", 1, 20); - var gitDiffEntry2 = addGitDiffEntry("test2.java", 1, 20); - var coverageEntry1 = addCoverageEntry("test.java", 5, 5, testCase); - var coverageEntry2 = addCoverageEntry("test2.java", 15, 2, testCase); - - assertThat(grouper.executeCondition()).isTrue(); - assertThat(grouper.executeAction()).isTrue(); - assertThat(blackboard.getGroupedFiles()).isNotNull().hasSize(2).containsExactlyInAnyOrder( - new GroupedFile("test.java", testCase, Set.of(gitDiffEntry1), Set.of(coverageEntry1)), - new GroupedFile("test2.java", testCase, Set.of(gitDiffEntry2), Set.of(coverageEntry2))); - } - - @Test - void testGroupSingleFileTwoTests() { - var testCase1 = new ProgrammingExerciseTestCase(); - var testCase2 = new ProgrammingExerciseTestCase(); - var gitDiffEntry = addGitDiffEntry("test.java", 1, 20); - var coverageEntry1 = addCoverageEntry("test.java", 5, 5, testCase1); - var coverageEntry2 = addCoverageEntry("test.java", 15, 2, testCase2); - - assertThat(grouper.executeCondition()).isTrue(); - assertThat(grouper.executeAction()).isTrue(); - assertThat(blackboard.getGroupedFiles()).isNotNull().hasSize(2).containsExactlyInAnyOrder( - new GroupedFile("test.java", testCase1, Set.of(gitDiffEntry), Set.of(coverageEntry1)), - new GroupedFile("test.java", testCase2, Set.of(gitDiffEntry), Set.of(coverageEntry2))); - } - - private ProgrammingExerciseGitDiffEntry addGitDiffEntry(String filePath, int startLine, int lineCount) { - var gitDiffEntry = new ProgrammingExerciseGitDiffEntry(); - gitDiffEntry.setFilePath(filePath); - gitDiffEntry.setStartLine(startLine); - gitDiffEntry.setStartLine(lineCount); - blackboard.getGitDiffReport().getEntries().add(gitDiffEntry); - return gitDiffEntry; - } - - private TestwiseCoverageReportEntry addCoverageEntry(String filePath, int startLine, int lineCount, ProgrammingExerciseTestCase testCase) { - var coverageFileReport = blackboard.getCoverageReport().getFileReports().stream().filter(fileReport -> filePath.equals(fileReport.getFilePath())).findFirst() - .orElse(new CoverageFileReport()); - coverageFileReport.setFilePath(filePath); - if (coverageFileReport.getTestwiseCoverageEntries() == null) { - coverageFileReport.setTestwiseCoverageEntries(new HashSet<>()); - } - var coverageReportEntry = new TestwiseCoverageReportEntry(); - coverageReportEntry.setStartLine(startLine); - coverageReportEntry.setLineCount(lineCount); - coverageReportEntry.setTestCase(testCase); - coverageFileReport.getTestwiseCoverageEntries().add(coverageReportEntry); - blackboard.getCoverageReport().getFileReports().add(coverageFileReport); - return coverageReportEntry; - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/InsertFileContentsTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/InsertFileContentsTest.java deleted file mode 100644 index e0ed10e61fa2..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/behavioral/InsertFileContentsTest.java +++ /dev/null @@ -1,60 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.behavioral; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.assertj.core.api.Assertions.assertThatExceptionOfType; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; - -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralBlackboard; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.BehavioralSolutionEntryGenerationException; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.GroupedFile; -import de.tum.cit.aet.artemis.programming.service.hestia.behavioral.knowledgesource.InsertFileContents; - -class InsertFileContentsTest { - - private InsertFileContents insertFileContents; - - private GroupedFile groupedFile; - - private Map solutionRepoFiles; - - @BeforeEach - void initBlackboard() { - solutionRepoFiles = new HashMap<>(); - BehavioralBlackboard blackboard = new BehavioralBlackboard(null, null, solutionRepoFiles); - var groupedFiles = new ArrayList(); - blackboard.setGroupedFiles(groupedFiles); - groupedFile = new GroupedFile("test.java", null, null, null); - groupedFiles.add(groupedFile); - - insertFileContents = new InsertFileContents(blackboard); - } - - @Test - void testNoAction() { - solutionRepoFiles.put("test.java", "Something else"); - groupedFile.setFileContent("Something"); - - assertThat(insertFileContents.executeCondition()).isFalse(); - } - - @Test - void testAddContent() throws BehavioralSolutionEntryGenerationException { - solutionRepoFiles.put("test.java", "Something"); - - assertThat(insertFileContents.executeCondition()).isTrue(); - assertThat(insertFileContents.executeAction()).isTrue(); - assertThat(groupedFile.getFileContent()).isEqualTo("Something"); - } - - @Test - void testInvalidContent() { - assertThat(insertFileContents.executeCondition()).isTrue(); - assertThatExceptionOfType(BehavioralSolutionEntryGenerationException.class).isThrownBy(() -> insertFileContents.executeAction()); - } -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/util/TestwiseCoverageTestUtil.java b/src/test/java/de/tum/cit/aet/artemis/programming/hestia/util/TestwiseCoverageTestUtil.java deleted file mode 100644 index 4dae98fee0b5..000000000000 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/util/TestwiseCoverageTestUtil.java +++ /dev/null @@ -1,49 +0,0 @@ -package de.tum.cit.aet.artemis.programming.hestia.util; - -import java.util.Map; -import java.util.Set; -import java.util.stream.Collectors; - -import de.tum.cit.aet.artemis.programming.domain.hestia.CoverageFileReport; -import de.tum.cit.aet.artemis.programming.domain.hestia.TestwiseCoverageReportEntry; - -public class TestwiseCoverageTestUtil { - - public static Map> generateCoverageFileReportByTestName() { - var fileReport1_1 = new CoverageFileReport(); - fileReport1_1.setFilePath("src/de/tum/in/ase/BubbleSort.java"); - var lineCountByStartLine1_1 = Map.ofEntries(Map.entry(15, 3), Map.entry(23, 1)); - var entries1_1 = lineCountByStartLine1_1.entrySet().stream().map((mapEntry) -> { - var entry = new TestwiseCoverageReportEntry(); - entry.setStartLine(mapEntry.getKey()); - entry.setLineCount(mapEntry.getValue()); - return entry; - }).collect(Collectors.toSet()); - fileReport1_1.setTestwiseCoverageEntries(entries1_1); - - var fileReport2_1 = new CoverageFileReport(); - fileReport2_1.setFilePath("src/de/tum/in/ase/BubbleSort.java"); - var lineCountByStartLine2_1 = Map.ofEntries(Map.entry(2, 1), Map.entry(16, 3)); - var entries2_1 = lineCountByStartLine2_1.entrySet().stream().map((mapEntry) -> { - var entry = new TestwiseCoverageReportEntry(); - entry.setStartLine(mapEntry.getKey()); - entry.setLineCount(mapEntry.getValue()); - return entry; - }).collect(Collectors.toSet()); - fileReport2_1.setTestwiseCoverageEntries(entries2_1); - - var fileReport2_2 = new CoverageFileReport(); - fileReport2_2.setFilePath("src/de/tum/in/ase/Context.java"); - var lineCountByStartLine2_2 = Map.ofEntries(Map.entry(1, 10)); - var entries2_2 = lineCountByStartLine2_2.entrySet().stream().map((mapEntry) -> { - var entry = new TestwiseCoverageReportEntry(); - entry.setStartLine(mapEntry.getKey()); - entry.setLineCount(mapEntry.getValue()); - return entry; - }).collect(Collectors.toSet()); - fileReport2_2.setTestwiseCoverageEntries(entries2_2); - - return Map.ofEntries(Map.entry("test1()", Set.of(fileReport1_1)), Map.entry("test2()", Set.of(fileReport2_1, fileReport2_2))); - } - -} diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIIntegrationTest.java index e3979dc830fb..52ccfd718e22 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIIntegrationTest.java @@ -173,7 +173,7 @@ void testBuildJobPersistence() { assertThat(buildJob.getCourseId()).isEqualTo(course.getId()); assertThat(buildJob.getExerciseId()).isEqualTo(programmingExercise.getId()); assertThat(buildJob.getParticipationId()).isEqualTo(studentParticipation.getId()); - assertThat(buildJob.getDockerImage()).isEqualTo(programmingExercise.getBuildConfig().getWindfile().getMetadata().docker().getFullImageName()); + assertThat(buildJob.getDockerImage()).isEqualTo(programmingExercise.getBuildConfig().getWindfile().metadata().docker().getFullImageName()); assertThat(buildJob.getRepositoryName()).isEqualTo(assignmentRepositorySlug); assertThat(buildJob.getBuildAgentAddress()).isNotEmpty(); assertThat(buildJob.getPriority()).isEqualTo(2); @@ -567,7 +567,7 @@ void testGetSubmissionReturnsWhenSubmissionProcessing() throws Exception { submission = programmingExerciseUtilService.addProgrammingSubmission(programmingExercise, submission, TEST_PREFIX + "student1"); JobTimingInfo jobTimingInfo = new JobTimingInfo(ZonedDateTime.now().minusSeconds(30), ZonedDateTime.now(), null, ZonedDateTime.now().plusSeconds(30), 60); - BuildConfig buildConfig = new BuildConfig(null, null, commitHash, commitHash, null, null, null, null, false, false, false, null, 0, null, null, null, null); + BuildConfig buildConfig = new BuildConfig(null, null, commitHash, commitHash, null, null, null, null, false, false, null, 0, null, null, null, null); BuildJobQueueItem buildJobQueueItem = new BuildJobQueueItem("1", "1", null, submission.getParticipation().getId(), 1L, programmingExercise.getId(), 0, 1, null, null, jobTimingInfo, buildConfig, null); diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIResourceIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIResourceIntegrationTest.java index 8ab4e99eb38e..4a5219c6ba7b 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIResourceIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIResourceIntegrationTest.java @@ -88,7 +88,7 @@ void createJobs() { JobTimingInfo jobTimingInfo2 = new JobTimingInfo(ZonedDateTime.now(), ZonedDateTime.now().plusMinutes(1), ZonedDateTime.now().plusMinutes(2), null, 20); JobTimingInfo jobTimingInfo3 = new JobTimingInfo(ZonedDateTime.now().minusMinutes(10), ZonedDateTime.now().minusMinutes(9), ZonedDateTime.now().plusSeconds(150), null, 20); - BuildConfig buildConfig = new BuildConfig("echo 'test'", "test", "test", "test", "test", "test", null, null, false, false, false, null, 0, null, null, null, null); + BuildConfig buildConfig = new BuildConfig("echo 'test'", "test", "test", "test", "test", "test", null, null, false, false, null, 0, null, null, null, null); RepositoryInfo repositoryInfo = new RepositoryInfo("test", null, RepositoryType.USER, "test", "test", "test", null, null); String memberAddress = hazelcastInstance.getCluster().getLocalMember().getAddress().toString(); @@ -282,7 +282,7 @@ void testGetFinishedBuildJobs_returnsFilteredJobs() throws Exception { // Create a failed job to filter for JobTimingInfo jobTimingInfo = new JobTimingInfo(ZonedDateTime.now().plusDays(1), ZonedDateTime.now().plusDays(1).plusMinutes(2), ZonedDateTime.now().plusDays(1).plusMinutes(10), null, 0); - BuildConfig buildConfig = new BuildConfig("echo 'test'", "test", "test", "test", "test", "test", null, null, false, false, false, null, 0, null, null, null, null); + BuildConfig buildConfig = new BuildConfig("echo 'test'", "test", "test", "test", "test", "test", null, null, false, false, null, 0, null, null, null, null); RepositoryInfo repositoryInfo = new RepositoryInfo("test", null, RepositoryType.USER, "test", "test", "test", null, null); var failedJob1 = new BuildJobQueueItem("5", "job5", buildAgent, 1, course.getId(), 1, 1, 1, BuildStatus.FAILED, repositoryInfo, jobTimingInfo, buildConfig, null); var jobResult = new Result().successful(false).rated(true).score(0D).assessmentType(AssessmentType.AUTOMATIC).completionDate(ZonedDateTime.now()); @@ -402,7 +402,7 @@ void testBuildJob() throws Exception { JobTimingInfo jobTimingInfo4 = new JobTimingInfo(now.plusSeconds(2), null, null, null, 24); JobTimingInfo jobTimingInfo5 = new JobTimingInfo(now.plusSeconds(3), null, null, null, 24); - BuildConfig buildConfig = new BuildConfig("echo 'test'", "test", "test", "test", "test", "test", null, null, false, false, false, null, 0, null, null, null, null); + BuildConfig buildConfig = new BuildConfig("echo 'test'", "test", "test", "test", "test", "test", null, null, false, false, null, 0, null, null, null, null); RepositoryInfo repositoryInfo = new RepositoryInfo("test", null, RepositoryType.USER, "test", "test", "test", null, null); var job1 = new BuildJobQueueItem("1", "job1", buildAgent, 1, course.getId(), 1, 1, 1, BuildStatus.SUCCESSFUL, repositoryInfo, jobTimingInfo1, buildConfig, null); diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIServiceTest.java index ef4e012901da..7ea4ceafddbf 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIServiceTest.java @@ -33,7 +33,7 @@ import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.RepositoryType; import de.tum.cit.aet.artemis.programming.dto.CheckoutDirectoriesDTO; -import de.tum.cit.aet.artemis.programming.service.aeolus.Windfile; +import de.tum.cit.aet.artemis.programming.dto.aeolus.Windfile; import de.tum.cit.aet.artemis.programming.service.ci.ContinuousIntegrationService.BuildStatus; class LocalCIServiceTest extends AbstractProgrammingIntegrationLocalCILocalVCTest { @@ -71,7 +71,7 @@ void testReturnCorrectBuildStatus() { ProgrammingExerciseStudentParticipation participation = participationUtilService.addStudentParticipationForProgrammingExercise(exercise, TEST_PREFIX + "student1"); JobTimingInfo jobTimingInfo = new JobTimingInfo(ZonedDateTime.now(), ZonedDateTime.now().plusMinutes(1), ZonedDateTime.now().plusMinutes(2), null, 0); - BuildConfig buildConfig = new BuildConfig("echo 'test'", "test", "test", "test", "test", "test", null, null, false, false, false, null, 0, null, null, null, null); + BuildConfig buildConfig = new BuildConfig("echo 'test'", "test", "test", "test", "test", "test", null, null, false, false, null, 0, null, null, null, null); RepositoryInfo repositoryInfo = new RepositoryInfo("test", null, RepositoryType.USER, "test", "test", "test", null, null); String memberAddress = hazelcastInstance.getCluster().getLocalMember().getAddress().toString(); @@ -114,7 +114,7 @@ void testRecreateBuildPlanForExercise() throws IOException { exercise.getBuildConfig().setBuildPlanConfiguration(null); continuousIntegrationService.recreateBuildPlansForExercise(exercise); script = buildScriptProviderService.getScriptFor(exercise.getProgrammingLanguage(), Optional.ofNullable(exercise.getProjectType()), exercise.isStaticCodeAnalysisEnabled(), - exercise.getBuildConfig().hasSequentialTestRuns(), exercise.getBuildConfig().isTestwiseCoverageEnabled()); + exercise.getBuildConfig().hasSequentialTestRuns()); Windfile windfile = aeolusTemplateService.getDefaultWindfileFor(exercise); String actualBuildConfig = exercise.getBuildConfig().getBuildPlanConfiguration(); String expectedBuildConfig = new ObjectMapper().writeValueAsString(windfile); @@ -133,7 +133,6 @@ void testGetScriptForWithoutCache() { programmingExercise.setProjectType(null); programmingExercise.setStaticCodeAnalysisEnabled(false); programmingExercise.getBuildConfig().setSequentialTestRuns(false); - programmingExercise.getBuildConfig().setTestwiseCoverageEnabled(false); String script = buildScriptProviderService.getScriptFor(programmingExercise); assertThat(script).isNotNull(); } diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalVCLocalCITestService.java b/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalVCLocalCITestService.java index 7ad94838921f..cb5d100e2d9f 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalVCLocalCITestService.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalVCLocalCITestService.java @@ -181,7 +181,7 @@ public void mockInspectImage(DockerClient dockerClient) { * @param dockerClient the DockerClient to mock. * @param resourceRegexPattern the regex pattern that the resource path must match. The resource path is the path of the file or directory inside the container. * @param dataToReturn the data to return inside the InputStream in form of a map. Each entry of the map will be one TarArchiveEntry with the key denoting the - * tarArchiveEntry.getName() and the value being the content of the TarArchiveEntry. There can be up to two dataToReturn entries, in which case + * tarArchiveEntry.name() and the value being the content of the TarArchiveEntry. There can be up to two dataToReturn entries, in which case * the first call to "copyArchiveFromContainerCmd().exec()" will return the first entry, and the second call will return the second entry. * @throws IOException if the InputStream cannot be created. */ @@ -213,7 +213,7 @@ public final void mockInputStreamReturnedFromContainer(DockerClient dockerClient } /** - * Create a BufferedInputStream from a map. Each entry of the map will be one TarArchiveEntry with the key denoting the tarArchiveEntry.getName() and the value being the + * Create a BufferedInputStream from a map. Each entry of the map will be one TarArchiveEntry with the key denoting the tarArchiveEntry.name() and the value being the * content. * The returned InputStream can be used to mock the InputStream returned by dockerClient.copyArchiveFromContainerCmd(String containerId, String resource).exec(). * diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/service/ConsistencyCheckTestService.java b/src/test/java/de/tum/cit/aet/artemis/programming/service/ConsistencyCheckTestService.java index 376fdaaf761e..7484326756bd 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/service/ConsistencyCheckTestService.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/service/ConsistencyCheckTestService.java @@ -95,7 +95,7 @@ public void testCheckConsistencyOfProgrammingExercise_missingVCSProject() throws var consistencyErrors = request.getList("/api/programming-exercises/" + exercise.getId() + "/consistency-check", HttpStatus.OK, ConsistencyErrorDTO.class); assertThat(consistencyErrors).hasSize(1); - assertThat(consistencyErrors.getFirst().getType()).isEqualTo(ConsistencyErrorDTO.ErrorType.VCS_PROJECT_MISSING); + assertThat(consistencyErrors.getFirst().type()).isEqualTo(ConsistencyErrorDTO.ErrorType.VCS_PROJECT_MISSING); } /** diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseFeedbackCreationServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseFeedbackCreationServiceTest.java index 2140b3586c7b..27591fdb993a 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseFeedbackCreationServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/service/ProgrammingExerciseFeedbackCreationServiceTest.java @@ -19,11 +19,11 @@ import de.tum.cit.aet.artemis.programming.AbstractProgrammingIntegrationIndependentTest; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCaseType; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.ProjectType; import de.tum.cit.aet.artemis.programming.domain.StaticCodeAnalysisTool; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTestCaseType; -import de.tum.cit.aet.artemis.programming.dto.AbstractBuildResultNotificationDTO; +import de.tum.cit.aet.artemis.programming.dto.BuildResultNotification; import de.tum.cit.aet.artemis.programming.dto.StaticCodeAnalysisIssue; import de.tum.cit.aet.artemis.programming.dto.StaticCodeAnalysisReportDTO; import de.tum.cit.aet.artemis.programming.util.ProgrammingExerciseFactory; @@ -219,7 +219,7 @@ void createFeedbackFromTestCaseSuccessfulNoMessage() { assertThat(createFeedbackFromTestCase("test1", List.of(), true)).isNull(); } - private AbstractBuildResultNotificationDTO generateResult(List successfulTests, List failedTests) { + private BuildResultNotification generateResult(List successfulTests, List failedTests) { return ProgrammingExerciseFactory.generateTestResultDTO(null, "SOLUTION", null, programmingExercise.getProgrammingLanguage(), false, successfulTests, failedTests, null, null, null); } diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/service/connectors/jenkins/build_plan/JenkinsPipelineScriptCreatorTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/service/connectors/jenkins/build_plan/JenkinsPipelineScriptCreatorTest.java index 34afe9865fd2..a6bfdccc28de 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/service/connectors/jenkins/build_plan/JenkinsPipelineScriptCreatorTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/service/connectors/jenkins/build_plan/JenkinsPipelineScriptCreatorTest.java @@ -28,7 +28,6 @@ void init() { programmingExercise.setProjectType(ProjectType.MAVEN_MAVEN); programmingExercise.setStaticCodeAnalysisEnabled(true); programmingExercise.getBuildConfig().setSequentialTestRuns(false); - programmingExercise.getBuildConfig().setTestwiseCoverageEnabled(false); programmingExercise.setReleaseDate(null); course.addExercises(programmingExercise); @@ -48,9 +47,7 @@ void testBuildPlanCreation() { void testReplacements() { jenkinsPipelineScriptCreator.createBuildPlanForExercise(programmingExercise); BuildPlan buildPlan = buildPlanRepository.findByProgrammingExercises_IdWithProgrammingExercises(programmingExercise.getId()).orElseThrow(); - assertThat(buildPlan.getBuildPlan()).doesNotContain("#isStaticCodeAnalysisEnabled", "#testWiseCoverage", "#dockerImage", "#dockerArgs") - // testwise coverage is disabled in the dummy exercise - .contains("isTestwiseCoverageEnabled = false && isSolutionBuild"); + assertThat(buildPlan.getBuildPlan()).doesNotContain("#isStaticCodeAnalysisEnabled", "#dockerImage", "#dockerArgs").contains("isSolutionBuild"); } @Test diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTaskTestRepository.java b/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTaskTestRepository.java index 7d79da2bdcb6..8da13dec7aa4 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTaskTestRepository.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTaskTestRepository.java @@ -1,18 +1,12 @@ package de.tum.cit.aet.artemis.programming.test_repository; -import java.util.Optional; import java.util.Set; -import jakarta.validation.constraints.NotNull; - import org.springframework.context.annotation.Primary; -import org.springframework.data.jpa.repository.Query; -import org.springframework.data.repository.query.Param; import org.springframework.stereotype.Repository; -import de.tum.cit.aet.artemis.core.exception.EntityNotFoundException; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseTaskRepository; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTask; +import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseTaskRepository; /** * Spring Data repository for the ProgrammingExerciseTask entity. @@ -22,31 +16,4 @@ public interface ProgrammingExerciseTaskTestRepository extends ProgrammingExerciseTaskRepository { Set findByExerciseId(Long exerciseId); - - /** - * Gets a task with its programming exercise, test cases and solution entries of the test cases - * - * @param entryId The id of the task - * @return The task with the given ID if found - * @throws EntityNotFoundException If no task with the given ID was found - */ - @NotNull - default ProgrammingExerciseTask findByIdWithTestCaseAndSolutionEntriesElseThrow(long entryId) throws EntityNotFoundException { - return getValueElseThrow(findByIdWithTestCaseAndSolutionEntries(entryId), entryId); - } - - /** - * Gets a task with its programming exercise, test cases and solution entries of the test cases - * - * @param entryId The id of the task - * @return The task with the given ID - */ - @Query(""" - SELECT t - FROM ProgrammingExerciseTask t - LEFT JOIN FETCH t.testCases tc - LEFT JOIN FETCH tc.solutionEntries - WHERE t.id = :entryId - """) - Optional findByIdWithTestCaseAndSolutionEntries(@Param("entryId") long entryId); } diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTestCaseTestRepository.java b/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTestCaseTestRepository.java index da42f6d383d3..b56c34de117c 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTestCaseTestRepository.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTestCaseTestRepository.java @@ -1,11 +1,8 @@ package de.tum.cit.aet.artemis.programming.test_repository; import java.util.Optional; -import java.util.Set; import org.springframework.context.annotation.Primary; -import org.springframework.data.jpa.repository.Query; -import org.springframework.data.repository.query.Param; import org.springframework.stereotype.Repository; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; @@ -16,18 +13,4 @@ public interface ProgrammingExerciseTestCaseTestRepository extends ProgrammingExerciseTestCaseRepository { Optional findByExerciseIdAndTestName(long exerciseId, String testName); - - /** - * Returns all test cases with the associated solution entries for a programming exercise - * - * @param exerciseId of the exercise - * @return all test cases with the associated solution entries - */ - @Query(""" - SELECT DISTINCT tc - FROM ProgrammingExerciseTestCase tc - LEFT JOIN FETCH tc.solutionEntries se - WHERE tc.exercise.id = :exerciseId - """) - Set findByExerciseIdWithSolutionEntries(@Param("exerciseId") long exerciseId); } diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTestRepository.java b/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTestRepository.java index 19c04aac1ffa..065ba6ade995 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTestRepository.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/test_repository/ProgrammingExerciseTestRepository.java @@ -30,17 +30,13 @@ public interface ProgrammingExerciseTestRepository extends ProgrammingExerciseRe LEFT JOIN FETCH p.templateParticipation LEFT JOIN FETCH p.solutionParticipation LEFT JOIN FETCH p.exampleSubmissions - LEFT JOIN FETCH p.exerciseHints eh - LEFT JOIN FETCH eh.solutionEntries LEFT JOIN FETCH p.tutorParticipations LEFT JOIN FETCH p.posts - LEFT JOIN FETCH p.testCases tc - LEFT JOIN FETCH tc.solutionEntries + LEFT JOIN FETCH p.testCases LEFT JOIN FETCH p.staticCodeAnalysisCategories LEFT JOIN FETCH p.auxiliaryRepositories LEFT JOIN FETCH p.tasks t LEFT JOIN FETCH t.testCases - LEFT JOIN FETCH t.exerciseHints LEFT JOIN FETCH p.plagiarismDetectionConfig LEFT JOIN FETCH p.buildConfig WHERE p.id = :exerciseId diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseFactory.java b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseFactory.java index 2a35129aaa9c..c946b4c68e7a 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseFactory.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseFactory.java @@ -126,7 +126,6 @@ private static void populateUnreleasedProgrammingExercise(ProgrammingExercise pr programmingExercise.setBuildConfig(new ProgrammingExerciseBuildConfig()); } programmingExercise.setStaticCodeAnalysisEnabled(false); - programmingExercise.getBuildConfig().setTestwiseCoverageEnabled(false); programmingExercise.setAssessmentType(AssessmentType.SEMI_AUTOMATIC); programmingExercise.setProgrammingLanguage(programmingLanguage); programmingExercise.getBuildConfig().setBuildScript("Some script"); @@ -173,7 +172,6 @@ public static ProgrammingExercise generateToBeImportedProgrammingExercise(String toBeImported.setTotalNumberOfAssessments(template.getTotalNumberOfAssessments()); toBeImported.setNumberOfComplaints(template.getNumberOfComplaints()); toBeImported.setNumberOfMoreFeedbackRequests(template.getNumberOfMoreFeedbackRequests()); - toBeImported.setExerciseHints(null); toBeImported.setSolutionParticipation(null); toBeImported.setTemplateParticipation(null); buildConfig.setSequentialTestRuns(template.getBuildConfig().hasSequentialTestRuns()); @@ -189,7 +187,6 @@ public static ProgrammingExercise generateToBeImportedProgrammingExercise(String toBeImported.setAllowOnlineEditor(template.isAllowOnlineEditor()); toBeImported.setAllowOfflineIde(template.isAllowOfflineIde()); toBeImported.setStaticCodeAnalysisEnabled(template.isStaticCodeAnalysisEnabled()); - buildConfig.setTestwiseCoverageEnabled(template.getBuildConfig().isTestwiseCoverageEnabled()); toBeImported.setTutorParticipations(null); toBeImported.setPosts(null); toBeImported.setStudentParticipations(null); @@ -242,7 +239,7 @@ public static TestResultsDTO generateTestResultDTO(String fullName, String repoN final var staticCodeAnalysisReports = enableStaticAnalysisReports ? generateStaticCodeAnalysisReports(programmingLanguage) : new ArrayList(); return new TestResultsDTO(successfulTestNames.size(), 0, 0, failedTestNames.size(), fullName, commits != null && !commits.isEmpty() ? commits : List.of(commitDTO), - List.of(testSuiteDto != null ? testSuiteDto : testSuite), staticCodeAnalysisReports, List.of(), buildRunDate != null ? buildRunDate : now(), false, logs); + List.of(testSuiteDto != null ? testSuiteDto : testSuite), staticCodeAnalysisReports, buildRunDate != null ? buildRunDate : now(), false, logs); } /** @@ -394,21 +391,20 @@ public static StaticCodeAnalysisCategory generateStaticCodeAnalysisCategory(Prog * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. */ public static void populateUnreleasedProgrammingExercise(ProgrammingExercise programmingExercise, String shortName, String title, boolean enableStaticCodeAnalysis) { - populateUnreleasedProgrammingExercise(programmingExercise, shortName, title, enableStaticCodeAnalysis, false, ProgrammingLanguage.JAVA); + populateUnreleasedProgrammingExercise(programmingExercise, shortName, title, enableStaticCodeAnalysis, ProgrammingLanguage.JAVA); } /** * Populates the provided programming exercise with the given short name, title, and other values. The release date of the exercise is set in the future. * - * @param programmingExercise The exercise to be populated. - * @param shortName The short name of the exercise. - * @param title The title of the exercise. - * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. - * @param enableTestwiseCoverageAnalysis True, if test wise coverage analysis should be enabled for the exercise. - * @param programmingLanguage The programming language used in the exercise. + * @param programmingExercise The exercise to be populated. + * @param shortName The short name of the exercise. + * @param title The title of the exercise. + * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. + * @param programmingLanguage The programming language used in the exercise. */ public static void populateUnreleasedProgrammingExercise(ProgrammingExercise programmingExercise, String shortName, String title, boolean enableStaticCodeAnalysis, - boolean enableTestwiseCoverageAnalysis, ProgrammingLanguage programmingLanguage) { + ProgrammingLanguage programmingLanguage) { programmingExercise.setProgrammingLanguage(programmingLanguage); programmingExercise.setShortName(shortName); programmingExercise.generateAndSetProjectKey(); @@ -444,7 +440,6 @@ else if (programmingLanguage == ProgrammingLanguage.C) { if (enableStaticCodeAnalysis) { programmingExercise.setMaxStaticCodeAnalysisPenalty(40); } - programmingExercise.getBuildConfig().setTestwiseCoverageEnabled(enableTestwiseCoverageAnalysis); // Note: no separators are allowed for Swift package names if (programmingLanguage == ProgrammingLanguage.SWIFT) { programmingExercise.setPackageName("swiftTest"); diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseResultTestService.java b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseResultTestService.java index 1ad25c969e6c..538e2890fd0a 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseResultTestService.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseResultTestService.java @@ -11,19 +11,15 @@ import static org.mockito.ArgumentMatchers.eq; import static org.mockito.Mockito.after; import static org.mockito.Mockito.argThat; -import static org.mockito.Mockito.doNothing; -import static org.mockito.Mockito.doReturn; import static org.mockito.Mockito.timeout; import static org.mockito.Mockito.times; import static org.mockito.Mockito.verify; import java.time.ZonedDateTime; -import java.util.Collections; import java.util.HashSet; import java.util.List; import java.util.Set; -import org.eclipse.jgit.api.errors.GitAPIException; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; import org.springframework.http.HttpHeaders; @@ -52,13 +48,12 @@ import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseStudentParticipation; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCaseType; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.SolutionProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.StaticCodeAnalysisTool; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTestCaseType; -import de.tum.cit.aet.artemis.programming.dto.AbstractBuildResultNotificationDTO; +import de.tum.cit.aet.artemis.programming.dto.BuildResultNotification; import de.tum.cit.aet.artemis.programming.dto.ResultDTO; -import de.tum.cit.aet.artemis.programming.hestia.util.TestwiseCoverageTestUtil; import de.tum.cit.aet.artemis.programming.repository.ParticipationVCSAccessTokenRepository; import de.tum.cit.aet.artemis.programming.repository.ProgrammingExerciseBuildConfigRepository; import de.tum.cit.aet.artemis.programming.repository.SolutionProgrammingExerciseParticipationRepository; @@ -152,10 +147,10 @@ public void setup(String userPrefix) { } public void setupForProgrammingLanguage(ProgrammingLanguage programmingLanguage) { - course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, false, programmingLanguage); + course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, programmingLanguage); programmingExercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); programmingExerciseUtilService.addTestCasesToProgrammingExercise(programmingExercise); - programmingExerciseWithStaticCodeAnalysis = programmingExerciseUtilService.addProgrammingExerciseToCourse(course, true, false, programmingLanguage); + programmingExerciseWithStaticCodeAnalysis = programmingExerciseUtilService.addProgrammingExerciseToCourse(course, true, programmingLanguage); staticCodeAnalysisService.createDefaultCategories(programmingExerciseWithStaticCodeAnalysis); // This is done to avoid an unproxy issue in the processNewResult method of the ResultService. solutionParticipation = solutionProgrammingExerciseRepository.findWithEagerResultsAndSubmissionsByProgrammingExerciseId(programmingExercise.getId()).orElseThrow(); @@ -180,7 +175,7 @@ public void tearDown() { } // Test - public void shouldUpdateFeedbackInSemiAutomaticResult(AbstractBuildResultNotificationDTO buildResultNotification, String loginName) throws Exception { + public void shouldUpdateFeedbackInSemiAutomaticResult(BuildResultNotification buildResultNotification, String loginName) throws Exception { // Make sure we only have one participation var participations = participationRepository.findByExerciseId(programmingExercise.getId()); for (ProgrammingExerciseStudentParticipation participation : participations) { @@ -229,7 +224,7 @@ public void shouldUpdateFeedbackInSemiAutomaticResult(AbstractBuildResultNotific assertThat(semiAutoResult.getFeedbacks().get(1).getType()).isEqualTo(FeedbackType.AUTOMATIC); } - private void postResult(AbstractBuildResultNotificationDTO requestBodyMap) throws Exception { + private void postResult(BuildResultNotification requestBodyMap) throws Exception { final var alteredObj = convertBuildResultToJsonObject(requestBodyMap); HttpHeaders httpHeaders = new HttpHeaders(); @@ -237,7 +232,7 @@ private void postResult(AbstractBuildResultNotificationDTO requestBodyMap) throw request.postWithoutLocation("/api/public/programming-exercises/new-result", alteredObj, HttpStatus.OK, httpHeaders); } - public static Object convertBuildResultToJsonObject(AbstractBuildResultNotificationDTO requestBodyMap) { + public static Object convertBuildResultToJsonObject(BuildResultNotification requestBodyMap) { ObjectMapper mapper = new ObjectMapper(); mapper.registerModule(new JavaTimeModule()); return mapper.convertValue(requestBodyMap, Object.class); @@ -255,7 +250,7 @@ private ProgrammingExerciseTestCase createTest(String testName, long testId, Pro } // Test - public void shouldUpdateTestCasesAndResultScoreFromSolutionParticipationResult(AbstractBuildResultNotificationDTO resultNotification, boolean withFailedTest) { + public void shouldUpdateTestCasesAndResultScoreFromSolutionParticipationResult(BuildResultNotification resultNotification, boolean withFailedTest) { // reset saved test weights to be all 1 var test2 = programmingExerciseTestCaseRepository.findByExerciseIdAndTestName(programmingExercise.getId(), "test2").orElseThrow(); var test3 = programmingExerciseTestCaseRepository.findByExerciseIdAndTestName(programmingExercise.getId(), "test3").orElseThrow(); @@ -285,8 +280,7 @@ public void shouldUpdateTestCasesAndResultScoreFromSolutionParticipationResult(A // test1 - test3 already exist, test4 should be newly created now. // All tests must have active = true since they are now used in the new solution result - assertThat(testCases).usingRecursiveFieldByFieldElementComparatorIgnoringFields("exercise", "id", "tasks", "solutionEntries", "coverageEntries") - .containsExactlyInAnyOrderElementsOf(expectedTestCases); + assertThat(testCases).usingRecursiveFieldByFieldElementComparatorIgnoringFields("exercise", "id", "tasks").containsExactlyInAnyOrderElementsOf(expectedTestCases); assertThat(result).isNotNull(); if (withFailedTest) { @@ -306,7 +300,7 @@ public void shouldUpdateTestCasesAndResultScoreFromSolutionParticipationResult(A } // Test - public void shouldStoreFeedbackForResultWithStaticCodeAnalysisReport(AbstractBuildResultNotificationDTO resultNotification, ProgrammingLanguage programmingLanguage) { + public void shouldStoreFeedbackForResultWithStaticCodeAnalysisReport(BuildResultNotification resultNotification, ProgrammingLanguage programmingLanguage) { final long participationId = programmingExerciseStudentParticipationStaticCodeAnalysis.getId(); final var resultRequestBody = convertBuildResultToJsonObject(resultNotification); final var result = gradingService.processNewProgrammingExerciseResult(programmingExerciseStudentParticipationStaticCodeAnalysis, resultRequestBody); @@ -331,7 +325,7 @@ public void shouldStoreFeedbackForResultWithStaticCodeAnalysisReport(AbstractBui } // Test - public void shouldGenerateNewManualResultIfManualAssessmentExists(AbstractBuildResultNotificationDTO resultNotification) { + public void shouldGenerateNewManualResultIfManualAssessmentExists(BuildResultNotification resultNotification) { activateFourTests(); var programmingSubmission = programmingExerciseUtilService.createProgrammingSubmission(programmingExerciseStudentParticipation, false); @@ -360,7 +354,7 @@ public void shouldGenerateNewManualResultIfManualAssessmentExists(AbstractBuildR } // Test - public void shouldRejectNotificationWithoutCommitHash(AbstractBuildResultNotificationDTO resultNotification) { + public void shouldRejectNotificationWithoutCommitHash(BuildResultNotification resultNotification) { final Object resultRequestBody = convertBuildResultToJsonObject(resultNotification); assertThatThrownBy(() -> gradingService.processNewProgrammingExerciseResult(programmingExerciseStudentParticipation, resultRequestBody)) .isInstanceOf(IllegalArgumentException.class).hasMessageContaining("does not specify the assignment commit hash."); @@ -375,36 +369,7 @@ private void activateFourTests() { } // Test - public void shouldGenerateTestwiseCoverageFileReports(AbstractBuildResultNotificationDTO resultNotification) throws GitAPIException { - // set testwise coverage analysis for programming exercise - programmingExercise.getBuildConfig().setTestwiseCoverageEnabled(true); - programmingExerciseBuildConfigRepository.save(programmingExercise.getBuildConfig()); - programmingExerciseRepository.save(programmingExercise); - solutionParticipation.setProgrammingExercise(programmingExercise); - solutionProgrammingExerciseRepository.save(solutionParticipation); - programmingExerciseUtilService.createProgrammingSubmission(solutionParticipation, false); - - // setup mocks - doReturn(null).when(gitService).getOrCheckoutRepository(any(), eq(true)); - doNothing().when(gitService).resetToOriginHead(any()); - doNothing().when(gitService).pullIgnoreConflicts(any()); - doReturn(Collections.emptyMap()).when(gitService).listFilesAndFolders(any()); - - var expectedReportsByTestName = TestwiseCoverageTestUtil.generateCoverageFileReportByTestName(); - - final var resultRequestBody = convertBuildResultToJsonObject(resultNotification); - final var result = gradingService.processNewProgrammingExerciseResult(solutionParticipation, resultRequestBody); - assertThat(result).isNotNull(); - var actualReportsByTestName = result.getCoverageFileReportsByTestCaseName(); - assertThat(actualReportsByTestName).usingRecursiveComparison().isEqualTo(expectedReportsByTestName); - - // the coverage result attribute is transient in the result and should not be saved to the database - var resultFromDatabase = resultRepository.findByIdElseThrow(result.getId()); - assertThat(resultFromDatabase.getCoverageFileReportsByTestCaseName()).isNull(); - } - - // Test - public void shouldIgnoreResultIfNotOnDefaultBranch(AbstractBuildResultNotificationDTO resultNotification) { + public void shouldIgnoreResultIfNotOnDefaultBranch(BuildResultNotification resultNotification) { solutionParticipation.setProgrammingExercise(programmingExercise); final var resultRequestBody = convertBuildResultToJsonObject(resultNotification); @@ -413,7 +378,7 @@ public void shouldIgnoreResultIfNotOnDefaultBranch(AbstractBuildResultNotificati } // Test - public void shouldCreateResultOnParticipationDefaultBranch(AbstractBuildResultNotificationDTO resultNotification) { + public void shouldCreateResultOnParticipationDefaultBranch(BuildResultNotification resultNotification) { programmingExerciseStudentParticipation.setProgrammingExercise(programmingExercise); programmingExerciseStudentParticipation.setBranch("branch"); @@ -424,7 +389,7 @@ public void shouldCreateResultOnParticipationDefaultBranch(AbstractBuildResultNo } // Test - public void shouldIgnoreResultIfNotOnParticipationBranch(AbstractBuildResultNotificationDTO resultNotification) { + public void shouldIgnoreResultIfNotOnParticipationBranch(BuildResultNotification resultNotification) { programmingExerciseStudentParticipation.setBranch("default"); programmingExerciseStudentParticipation.setProgrammingExercise(programmingExercise); @@ -434,7 +399,7 @@ public void shouldIgnoreResultIfNotOnParticipationBranch(AbstractBuildResultNoti } // Test - public void shouldCreateResultOnCustomDefaultBranch(String defaultBranch, AbstractBuildResultNotificationDTO resultNotification) { + public void shouldCreateResultOnCustomDefaultBranch(String defaultBranch, BuildResultNotification resultNotification) { programmingExercise.getBuildConfig().setBranch(defaultBranch); programmingExerciseBuildConfigRepository.save(programmingExercise.getBuildConfig()); programmingExercise = programmingExerciseRepository.save(programmingExercise); @@ -447,8 +412,7 @@ public void shouldCreateResultOnCustomDefaultBranch(String defaultBranch, Abstra } // Test - public void shouldCorrectlyNotifyStudentsAboutNewResults(AbstractBuildResultNotificationDTO resultNotification, WebsocketMessagingService websocketMessagingService) - throws Exception { + public void shouldCorrectlyNotifyStudentsAboutNewResults(BuildResultNotification resultNotification, WebsocketMessagingService websocketMessagingService) throws Exception { programmingExerciseUtilService.addTestCasesToProgrammingExercise(programmingExercise); var programmingSubmission = programmingExerciseUtilService.createProgrammingSubmission(programmingExerciseStudentParticipation, false); @@ -471,8 +435,7 @@ public void shouldCorrectlyNotifyStudentsAboutNewResults(AbstractBuildResultNoti } // Test - public void shouldNotNotifyStudentsAboutNewResults(AbstractBuildResultNotificationDTO resultNotification, WebsocketMessagingService websocketMessagingService) - throws Exception { + public void shouldNotNotifyStudentsAboutNewResults(BuildResultNotification resultNotification, WebsocketMessagingService websocketMessagingService) throws Exception { programmingExerciseUtilService.addTestCasesToProgrammingExercise(programmingExercise); var programmingSubmission = programmingExerciseUtilService.createProgrammingSubmission(programmingExerciseStudentParticipation, false); @@ -486,7 +449,7 @@ public void shouldNotNotifyStudentsAboutNewResults(AbstractBuildResultNotificati } // Test - public void shouldRemoveTestCaseNamesFromWebsocketNotification(AbstractBuildResultNotificationDTO resultNotification, WebsocketMessagingService websocketMessagingService) + public void shouldRemoveTestCaseNamesFromWebsocketNotification(BuildResultNotification resultNotification, WebsocketMessagingService websocketMessagingService) throws Exception { var programmingSubmission = programmingExerciseUtilService.createProgrammingSubmission(programmingExerciseStudentParticipation, false); programmingExerciseStudentParticipation.addSubmission(programmingSubmission); @@ -500,7 +463,7 @@ public void shouldRemoveTestCaseNamesFromWebsocketNotification(AbstractBuildResu } // Test - public void shouldUpdateParticipantScoresOnlyOnce(AbstractBuildResultNotificationDTO resultNotification, InstanceMessageSendService instanceMessageSendService) { + public void shouldUpdateParticipantScoresOnlyOnce(BuildResultNotification resultNotification, InstanceMessageSendService instanceMessageSendService) { final var resultRequestBody = convertBuildResultToJsonObject(resultNotification); gradingService.processNewProgrammingExerciseResult(programmingExerciseStudentParticipation, resultRequestBody); diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseTestService.java b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseTestService.java index e097a79f772f..eb0268471ccf 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseTestService.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseTestService.java @@ -120,6 +120,7 @@ import de.tum.cit.aet.artemis.programming.domain.AuxiliaryRepository; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseStudentParticipation; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; @@ -128,8 +129,6 @@ import de.tum.cit.aet.artemis.programming.domain.StaticCodeAnalysisCategory; import de.tum.cit.aet.artemis.programming.domain.VcsRepositoryUri; import de.tum.cit.aet.artemis.programming.domain.build.BuildLogStatisticsEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.domain.submissionpolicy.LockRepositoryPolicy; import de.tum.cit.aet.artemis.programming.dto.BuildLogStatisticsDTO; import de.tum.cit.aet.artemis.programming.repository.AuxiliaryRepositoryRepository; @@ -784,7 +783,6 @@ public void createAndImportJavaProgrammingExercise(boolean staticCodeAnalysisEna sourceExercise.setTasks(Collections.singletonList(task)); programmingExerciseTaskRepository.save(task); programmingExerciseRepository.save(sourceExercise); - programmingExerciseUtilService.addHintsToExercise(sourceExercise); // Reset because we will add mocks for new requests mockDelegate.resetMockProvider(); @@ -819,7 +817,8 @@ public void createAndImportJavaProgrammingExercise(boolean staticCodeAnalysisEna importedExercise = programmingExerciseUtilService.loadProgrammingExerciseWithEagerReferences(importedExercise); // Check that the tasks were imported correctly (see #5474) - assertThat(programmingExerciseTaskRepository.findByExerciseId(importedExercise.getId())).hasSameSizeAs(sourceExercise.getTasks()); + var importedExerciseTasks = programmingExerciseTaskRepository.findByExerciseId(importedExercise.getId()); + assertThat(importedExerciseTasks).hasSameSizeAs(sourceExercise.getTasks()); } // TEST @@ -828,10 +827,9 @@ public void importExercise_created(ProgrammingLanguage programmingLanguage, bool // Setup exercises for import ProgrammingExercise sourceExercise = programmingExerciseUtilService.addCourseWithOneProgrammingExerciseAndStaticCodeAnalysisCategories(programmingLanguage); sourceExercise.setPlagiarismDetectionConfig(PlagiarismDetectionConfig.createDefault()); - sourceExercise = programmingExerciseRepository.save(sourceExercise); sourceExercise.setStaticCodeAnalysisEnabled(staticCodeAnalysisEnabled); + sourceExercise = programmingExerciseRepository.save(sourceExercise); programmingExerciseUtilService.addTestCasesToProgrammingExercise(sourceExercise); - programmingExerciseUtilService.addHintsToExercise(sourceExercise); sourceExercise = programmingExerciseUtilService.loadProgrammingExerciseWithEagerReferences(sourceExercise); ProgrammingExercise exerciseToBeImported = ProgrammingExerciseFactory.generateToBeImportedProgrammingExercise("ImportTitle", "imported", sourceExercise, courseUtilService.addEmptyCourse()); @@ -868,15 +866,7 @@ public void importExercise_created(ProgrammingLanguage programmingLanguage, bool var sourceTestCaseIds = sourceExercise.getTestCases().stream().map(ProgrammingExerciseTestCase::getId).collect(Collectors.toSet()); assertThat(importedTestCaseIds).doesNotContainAnyElementsOf(sourceTestCaseIds); assertThat(importedExercise.getTestCases()).usingRecursiveFieldByFieldElementComparator() - .usingRecursiveFieldByFieldElementComparatorIgnoringFields("id", "exercise", "tasks", "solutionEntries", "coverageEntries") - .containsExactlyInAnyOrderElementsOf(sourceExercise.getTestCases()); - - // Assert correct creation of hints - var importedHintIds = importedExercise.getExerciseHints().stream().map(ExerciseHint::getId).collect(Collectors.toSet()); - var sourceHintIds = sourceExercise.getExerciseHints().stream().map(ExerciseHint::getId).collect(Collectors.toSet()); - assertThat(importedHintIds).doesNotContainAnyElementsOf(sourceHintIds); - assertThat(importedExercise.getExerciseHints()).usingRecursiveFieldByFieldElementComparatorIgnoringFields("id", "exercise", "exerciseHintActivations") - .containsExactlyInAnyOrderElementsOf(sourceExercise.getExerciseHints()); + .usingRecursiveFieldByFieldElementComparatorIgnoringFields("id", "exercise", "tasks").containsExactlyInAnyOrderElementsOf(sourceExercise.getTestCases()); // Assert creation of new build plan ids assertThat(importedExercise.getSolutionParticipation().getBuildPlanId()).isNotBlank().isNotEqualTo(sourceExercise.getSolutionParticipation().getBuildPlanId()); @@ -895,7 +885,6 @@ public void updateBuildPlanURL() throws Exception { sourceExercise.setStaticCodeAnalysisEnabled(staticCodeAnalysisEnabled); sourceExercise.getBuildConfig().generateAndSetBuildPlanAccessSecret(); programmingExerciseUtilService.addTestCasesToProgrammingExercise(sourceExercise); - programmingExerciseUtilService.addHintsToExercise(sourceExercise); programmingExerciseBuildConfigRepository.save(sourceExercise.getBuildConfig()); sourceExercise = programmingExerciseUtilService.loadProgrammingExerciseWithEagerReferences(sourceExercise); ProgrammingExercise exerciseToBeImported = ProgrammingExerciseFactory.generateToBeImportedProgrammingExercise("ImportTitle", "imported", sourceExercise, @@ -979,7 +968,6 @@ public void testImportProgrammingExercise_team_modeChange() throws Exception { ProgrammingExercise sourceExercise = programmingExerciseUtilService.addCourseWithOneProgrammingExerciseAndStaticCodeAnalysisCategories(); sourceExercise.setMode(ExerciseMode.INDIVIDUAL); programmingExerciseUtilService.addTestCasesToProgrammingExercise(sourceExercise); - programmingExerciseUtilService.addHintsToExercise(sourceExercise); sourceExercise = programmingExerciseUtilService.loadProgrammingExerciseWithEagerReferences(sourceExercise); sourceExercise.setCourse(sourceExercise.getCourseViaExerciseGroupOrCourseMember()); programmingExerciseRepository.save(sourceExercise); @@ -1019,7 +1007,7 @@ public void testImportProgrammingExercise_individual_modeChange() throws Excepti ProgrammingExercise sourceExercise = programmingExerciseUtilService.addCourseWithOneProgrammingExerciseAndStaticCodeAnalysisCategories(); sourceExercise.setMode(TEAM); programmingExerciseUtilService.addTestCasesToProgrammingExercise(sourceExercise); - programmingExerciseUtilService.addHintsToExercise(sourceExercise); + programmingExerciseRepository.save(sourceExercise); sourceExercise = programmingExerciseUtilService.loadProgrammingExerciseWithEagerReferences(sourceExercise); var teamAssignmentConfig = new TeamAssignmentConfig(); teamAssignmentConfig.setExercise(sourceExercise); @@ -1186,7 +1174,6 @@ public void importProgrammingExerciseAsPartOfExamImport() throws Exception { ProgrammingExercise sourceExercise = programmingExerciseUtilService.addProgrammingExerciseToExam(sourceExam, 0); sourceExercise.setStaticCodeAnalysisEnabled(false); programmingExerciseUtilService.addTestCasesToProgrammingExercise(sourceExercise); - programmingExerciseUtilService.addHintsToExercise(sourceExercise); sourceExercise = programmingExerciseUtilService.loadProgrammingExerciseWithEagerReferences(sourceExercise); // Setup to be imported exam and exercise @@ -1231,15 +1218,7 @@ public void importProgrammingExerciseAsPartOfExamImport() throws Exception { var sourceTestCaseIds = sourceExercise.getTestCases().stream().map(ProgrammingExerciseTestCase::getId).toList(); assertThat(importedTestCaseIds).doesNotContainAnyElementsOf(sourceTestCaseIds); assertThat(importedExercise.getTestCases()).usingRecursiveFieldByFieldElementComparator() - .usingRecursiveFieldByFieldElementComparatorIgnoringFields("id", "exercise", "tasks", "solutionEntries", "coverageEntries") - .containsExactlyInAnyOrderElementsOf(sourceExercise.getTestCases()); - - // Assert correct creation of hints - var importedHintIds = importedExercise.getExerciseHints().stream().map(ExerciseHint::getId).toList(); - var sourceHintIds = sourceExercise.getExerciseHints().stream().map(ExerciseHint::getId).toList(); - assertThat(importedHintIds).doesNotContainAnyElementsOf(sourceHintIds); - assertThat(importedExercise.getExerciseHints()).usingRecursiveFieldByFieldElementComparatorIgnoringFields("id", "exercise", "exerciseHintActivations") - .containsExactlyInAnyOrderElementsOf(sourceExercise.getExerciseHints()); + .usingRecursiveFieldByFieldElementComparatorIgnoringFields("id", "exercise", "tasks").containsExactlyInAnyOrderElementsOf(sourceExercise.getTestCases()); } // TEST diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseUtilService.java b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseUtilService.java index 7178f4376a08..2c4321d22b79 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseUtilService.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingExerciseUtilService.java @@ -18,7 +18,6 @@ import java.util.Collections; import java.util.HashSet; import java.util.List; -import java.util.Set; import java.util.stream.Collectors; import org.apache.commons.io.FileUtils; @@ -52,6 +51,7 @@ import de.tum.cit.aet.artemis.exercise.util.ExerciseUtilService; import de.tum.cit.aet.artemis.programming.domain.AuxiliaryRepository; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; +import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseTestCase; import de.tum.cit.aet.artemis.programming.domain.ProgrammingLanguage; import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; @@ -60,10 +60,6 @@ import de.tum.cit.aet.artemis.programming.domain.SolutionProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.TemplateProgrammingExerciseParticipation; import de.tum.cit.aet.artemis.programming.domain.build.BuildPlanType; -import de.tum.cit.aet.artemis.programming.domain.hestia.CodeHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ExerciseHint; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseSolutionEntry; -import de.tum.cit.aet.artemis.programming.domain.hestia.ProgrammingExerciseTask; import de.tum.cit.aet.artemis.programming.domain.submissionpolicy.SubmissionPolicy; import de.tum.cit.aet.artemis.programming.repository.AuxiliaryRepositoryRepository; import de.tum.cit.aet.artemis.programming.repository.BuildPlanRepository; @@ -72,9 +68,6 @@ import de.tum.cit.aet.artemis.programming.repository.StaticCodeAnalysisCategoryRepository; import de.tum.cit.aet.artemis.programming.repository.SubmissionPolicyRepository; import de.tum.cit.aet.artemis.programming.repository.TemplateProgrammingExerciseParticipationRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.CodeHintRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ExerciseHintRepository; -import de.tum.cit.aet.artemis.programming.repository.hestia.ProgrammingExerciseSolutionEntryRepository; import de.tum.cit.aet.artemis.programming.service.GitService; import de.tum.cit.aet.artemis.programming.test_repository.ProgrammingExerciseTaskTestRepository; import de.tum.cit.aet.artemis.programming.test_repository.ProgrammingExerciseTestCaseTestRepository; @@ -142,18 +135,9 @@ public class ProgrammingExerciseUtilService { @Autowired private StudentParticipationTestRepository studentParticipationRepo; - @Autowired - private ExerciseHintRepository exerciseHintRepository; - @Autowired private ProgrammingExerciseTaskTestRepository programmingExerciseTaskRepository; - @Autowired - private ProgrammingExerciseSolutionEntryRepository solutionEntryRepository; - - @Autowired - private CodeHintRepository codeHintRepository; - @Autowired private ProgrammingExerciseTestRepository programmingExerciseTestRepository; @@ -376,7 +360,7 @@ public Course addCourseWithOneProgrammingExercise() { * @return The created course with a programming exercise. */ public Course addCourseWithOneProgrammingExercise(boolean enableStaticCodeAnalysis) { - return addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, false, ProgrammingLanguage.JAVA); + return addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, ProgrammingLanguage.JAVA); } /** @@ -388,36 +372,33 @@ public Course addCourseWithOneProgrammingExercise(boolean enableStaticCodeAnalys * @return The created course with a programming exercise. */ public Course addCourseWithOneProgrammingExercise(boolean enableStaticCodeAnalysis, String title, String shortName) { - return addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, false, ProgrammingLanguage.JAVA, title, shortName); + return addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, ProgrammingLanguage.JAVA, title, shortName); } /** * Creates and saves a course with a programming exercise. Uses Programming as the title and TSTEXC as the short name of the exercise. * - * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. - * @param enableTestwiseCoverageAnalysis True, if test wise coverage analysis should be enabled for the exercise. - * @param programmingLanguage The programming language fo the exercise. + * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. + * @param programmingLanguage The programming language fo the exercise. * @return The created course with a programming exercise. */ - public Course addCourseWithOneProgrammingExercise(boolean enableStaticCodeAnalysis, boolean enableTestwiseCoverageAnalysis, ProgrammingLanguage programmingLanguage) { - return addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, enableTestwiseCoverageAnalysis, programmingLanguage, "Programming", "TSTEXC"); + public Course addCourseWithOneProgrammingExercise(boolean enableStaticCodeAnalysis, ProgrammingLanguage programmingLanguage) { + return addCourseWithOneProgrammingExercise(enableStaticCodeAnalysis, programmingLanguage, "Programming", "TSTEXC"); } /** * Creates and saves a course with a programming exercise. * - * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. - * @param enableTestwiseCoverageAnalysis True, if test wise coverage analysis should be enabled for the exercise. - * @param programmingLanguage The programming language fo the exercise. - * @param title The title of the exercise. - * @param shortName The short name of the exercise. + * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. + * @param programmingLanguage The programming language fo the exercise. + * @param title The title of the exercise. + * @param shortName The short name of the exercise. * @return The created course with a programming exercise. */ - public Course addCourseWithOneProgrammingExercise(boolean enableStaticCodeAnalysis, boolean enableTestwiseCoverageAnalysis, ProgrammingLanguage programmingLanguage, - String title, String shortName) { + public Course addCourseWithOneProgrammingExercise(boolean enableStaticCodeAnalysis, ProgrammingLanguage programmingLanguage, String title, String shortName) { var course = CourseFactory.generateCourse(null, PAST_TIMESTAMP, FUTURE_FUTURE_TIMESTAMP, new HashSet<>(), "tumuser", "tutor", "editor", "instructor"); course = courseRepo.save(course); - addProgrammingExerciseToCourse(course, enableStaticCodeAnalysis, enableTestwiseCoverageAnalysis, programmingLanguage, title, shortName, null); + addProgrammingExerciseToCourse(course, enableStaticCodeAnalysis, programmingLanguage, title, shortName, null); course = courseRepo.findByIdWithExercisesAndExerciseDetailsAndLecturesElseThrow(course.getId()); for (var exercise : course.getExercises()) { if (exercise instanceof ProgrammingExercise) { @@ -446,7 +427,7 @@ public ProgrammingExercise addProgrammingExerciseToCourse(Course course) { * @return The programming exercise which was added to the course. */ public ProgrammingExercise addProgrammingExerciseToCourse(Course course, boolean enableStaticCodeAnalysis) { - return addProgrammingExerciseToCourse(course, enableStaticCodeAnalysis, false, ProgrammingLanguage.JAVA); + return addProgrammingExerciseToCourse(course, enableStaticCodeAnalysis, ProgrammingLanguage.JAVA); } /** @@ -458,56 +439,51 @@ public ProgrammingExercise addProgrammingExerciseToCourse(Course course, boolean * @return The programming exercise which was added to the course. */ public ProgrammingExercise addProgrammingExerciseToCourse(Course course, boolean enableStaticCodeAnalysis, ZonedDateTime assessmentDueDate) { - return addProgrammingExerciseToCourse(course, enableStaticCodeAnalysis, false, ProgrammingLanguage.JAVA, assessmentDueDate); + return addProgrammingExerciseToCourse(course, enableStaticCodeAnalysis, ProgrammingLanguage.JAVA, assessmentDueDate); } /** * Adds a programming exercise to the given course. Uses Programming as the title and TSTEXC as the short name of the exercise. * - * @param course The course to which the exercise should be added. - * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. - * @param enableTestwiseCoverageAnalysis True, if test wise coverage analysis should be enabled for the exercise. - * @param programmingLanguage The programming language used in the exercise. - * @param assessmentDueDate The assessment due date of the exercise. + * @param course The course to which the exercise should be added. + * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. + * @param programmingLanguage The programming language used in the exercise. + * @param assessmentDueDate The assessment due date of the exercise. * @return The programming exercise which was added to the course. */ - public ProgrammingExercise addProgrammingExerciseToCourse(Course course, boolean enableStaticCodeAnalysis, boolean enableTestwiseCoverageAnalysis, - ProgrammingLanguage programmingLanguage, ZonedDateTime assessmentDueDate) { - return addProgrammingExerciseToCourse(course, enableStaticCodeAnalysis, enableTestwiseCoverageAnalysis, programmingLanguage, "Programming", "TSTEXC", assessmentDueDate); + public ProgrammingExercise addProgrammingExerciseToCourse(Course course, boolean enableStaticCodeAnalysis, ProgrammingLanguage programmingLanguage, + ZonedDateTime assessmentDueDate) { + return addProgrammingExerciseToCourse(course, enableStaticCodeAnalysis, programmingLanguage, "Programming", "TSTEXC", assessmentDueDate); } /** * Adds a programming exercise without an assessment due date to the given course. Uses Programming as the title and TSTEXC as the short name of the * exercise. * - * @param course The course to which the exercise should be added. - * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. - * @param enableTestwiseCoverageAnalysis True, if test wise coverage analysis should be enabled for the exercise. - * @param programmingLanguage The programming language used in the exercise. + * @param course The course to which the exercise should be added. + * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. + * @param programmingLanguage The programming language used in the exercise. * @return The programming exercise which was added to the course. */ - public ProgrammingExercise addProgrammingExerciseToCourse(Course course, boolean enableStaticCodeAnalysis, boolean enableTestwiseCoverageAnalysis, - ProgrammingLanguage programmingLanguage) { - return addProgrammingExerciseToCourse(course, enableStaticCodeAnalysis, enableTestwiseCoverageAnalysis, programmingLanguage, "Programming", "TSTEXC", null); + public ProgrammingExercise addProgrammingExerciseToCourse(Course course, boolean enableStaticCodeAnalysis, ProgrammingLanguage programmingLanguage) { + return addProgrammingExerciseToCourse(course, enableStaticCodeAnalysis, programmingLanguage, "Programming", "TSTEXC", null); } /** * Adds a programming exercise to the given course. * - * @param course The course to which the exercise should be added. - * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. - * @param enableTestwiseCoverageAnalysis True, if test wise coverage analysis should be enabled for the exercise. - * @param programmingLanguage The programming language used in the exercise. - * @param title The title of the exercise. - * @param shortName The short name of the exercise. - * @param assessmentDueDate The assessment due date of the exercise. + * @param course The course to which the exercise should be added. + * @param enableStaticCodeAnalysis True, if the static code analysis should be enabled for the exercise. + * @param programmingLanguage The programming language used in the exercise. + * @param title The title of the exercise. + * @param shortName The short name of the exercise. + * @param assessmentDueDate The assessment due date of the exercise. * @return The programming exercise which was added to the course. */ - public ProgrammingExercise addProgrammingExerciseToCourse(Course course, boolean enableStaticCodeAnalysis, boolean enableTestwiseCoverageAnalysis, - ProgrammingLanguage programmingLanguage, String title, String shortName, ZonedDateTime assessmentDueDate) { + public ProgrammingExercise addProgrammingExerciseToCourse(Course course, boolean enableStaticCodeAnalysis, ProgrammingLanguage programmingLanguage, String title, + String shortName, ZonedDateTime assessmentDueDate) { var programmingExercise = (ProgrammingExercise) new ProgrammingExercise().course(course); - ProgrammingExerciseFactory.populateUnreleasedProgrammingExercise(programmingExercise, shortName, title, enableStaticCodeAnalysis, enableTestwiseCoverageAnalysis, - programmingLanguage); + ProgrammingExerciseFactory.populateUnreleasedProgrammingExercise(programmingExercise, shortName, title, enableStaticCodeAnalysis, programmingLanguage); programmingExercise.setAssessmentDueDate(assessmentDueDate); programmingExercise.setPresentationScoreEnabled(course.getPresentationScore() != 0); @@ -544,27 +520,6 @@ public Course addCourseWithNamedProgrammingExercise(String programmingExerciseTi return courseRepo.findByIdWithExercisesAndExerciseDetailsAndLecturesElseThrow(course.getId()); } - /** - * Creates and saves a course with a programming exercise and 3 active, always visible test cases with different weights. - * - * @return The newly created course with a programming exercise. - */ - public Course addCourseWithOneProgrammingExerciseAndSpecificTestCases() { - Course course = addCourseWithOneProgrammingExercise(); - ProgrammingExercise programmingExercise = exerciseUtilService.findProgrammingExerciseWithTitle(course.getExercises(), "Programming"); - - List testCases = new ArrayList<>(); - testCases.add(new ProgrammingExerciseTestCase().testName("testClass[BubbleSort]").weight(1.0).active(true).exercise(programmingExercise).bonusMultiplier(1D).bonusPoints(0D) - .visibility(Visibility.ALWAYS)); - testCases.add(new ProgrammingExerciseTestCase().testName("testMethods[Context]").weight(2.0).active(true).exercise(programmingExercise).bonusMultiplier(1D).bonusPoints(0D) - .visibility(Visibility.ALWAYS)); - testCases.add(new ProgrammingExerciseTestCase().testName("testMethods[Policy]").weight(3.0).active(true).exercise(programmingExercise).bonusMultiplier(1D).bonusPoints(0D) - .visibility(Visibility.ALWAYS)); - testCaseRepository.saveAll(testCases); - - return courseRepo.findByIdWithEagerExercisesElseThrow(course.getId()); - } - /** * Creates and saves a course with a java programming exercise with static code analysis enabled. * @@ -581,7 +536,7 @@ public ProgrammingExercise addCourseWithOneProgrammingExerciseAndStaticCodeAnaly * @return The newly created programming exercise. */ public ProgrammingExercise addCourseWithOneProgrammingExerciseAndStaticCodeAnalysisCategories(ProgrammingLanguage programmingLanguage) { - Course course = addCourseWithOneProgrammingExercise(true, false, programmingLanguage); + Course course = addCourseWithOneProgrammingExercise(true, programmingLanguage); ProgrammingExercise programmingExercise = exerciseUtilService.findProgrammingExerciseWithTitle(course.getExercises(), "Programming"); programmingExercise = programmingExerciseRepository.save(programmingExercise); programmingExercise = programmingExerciseRepository.findWithBuildConfigById(programmingExercise.getId()).orElseThrow(); @@ -863,27 +818,6 @@ public ProgrammingSubmission addProgrammingSubmissionToResultAndParticipation(Re return submissionRepository.save(submission); } - /** - * Adds 3 hints to the given programming exercise. Each hint has a unique content and title. All have a display threshold of 3. - * - * @param exercise The exercise to which hints should be added. - */ - public void addHintsToExercise(ProgrammingExercise exercise) { - ExerciseHint exerciseHint1 = new ExerciseHint().content("content 1").exercise(exercise).title("title 1"); - ExerciseHint exerciseHint2 = new ExerciseHint().content("content 2").exercise(exercise).title("title 2"); - ExerciseHint exerciseHint3 = new ExerciseHint().content("content 3").exercise(exercise).title("title 3"); - exerciseHint1.setDisplayThreshold((short) 3); - exerciseHint2.setDisplayThreshold((short) 3); - exerciseHint3.setDisplayThreshold((short) 3); - Set hints = new HashSet<>(); - hints.add(exerciseHint1); - hints.add(exerciseHint2); - hints.add(exerciseHint3); - exercise.setExerciseHints(hints); - exerciseHintRepository.saveAll(hints); - programmingExerciseRepository.save(exercise); - } - /** * Adds a task for each test case and adds it to the problem statement of the programming exercise. * @@ -909,48 +843,6 @@ public void addTasksToProgrammingExercise(ProgrammingExercise programmingExercis programmingExerciseRepository.save(programmingExercise); } - /** - * Adds a solution entry to each test case of the given programming exercise. - * - * @param programmingExercise The exercise to which solution entries should be added. - */ - public void addSolutionEntriesToProgrammingExercise(ProgrammingExercise programmingExercise) { - for (ProgrammingExerciseTestCase testCase : programmingExercise.getTestCases()) { - var solutionEntry = new ProgrammingExerciseSolutionEntry(); - solutionEntry.setFilePath("test.txt"); - solutionEntry.setLine(1); - solutionEntry.setCode("Line for " + testCase.getTestName()); - solutionEntry.setTestCase(testCase); - - testCase.setSolutionEntries(Collections.singleton(solutionEntry)); - solutionEntryRepository.save(solutionEntry); - } - } - - /** - * Adds a code hint to each task of the given programming exercise. - * - * @param programmingExercise The programming exercise to which code hints should be added. - */ - public void addCodeHintsToProgrammingExercise(ProgrammingExercise programmingExercise) { - for (ProgrammingExerciseTask task : programmingExercise.getTasks()) { - var solutionEntries = task.getTestCases().stream().flatMap(testCase -> testCase.getSolutionEntries().stream()).collect(Collectors.toSet()); - var codeHint = new CodeHint(); - codeHint.setTitle("Code Hint for " + task.getTaskName()); - codeHint.setContent("Content for " + task.getTaskName()); - codeHint.setExercise(programmingExercise); - codeHint.setSolutionEntries(solutionEntries); - codeHint.setProgrammingExerciseTask(task); - - programmingExercise.getExerciseHints().add(codeHint); - codeHint = codeHintRepository.save(codeHint); - for (ProgrammingExerciseSolutionEntry solutionEntry : solutionEntries) { - solutionEntry.setCodeHint(codeHint); - solutionEntryRepository.save(solutionEntry); - } - } - } - /** * Loads a programming exercise with eager references from the repository. * diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingSubmissionAndResultIntegrationTestService.java b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingSubmissionAndResultIntegrationTestService.java index 3a53d1b77ada..992f9ac6e8b5 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingSubmissionAndResultIntegrationTestService.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingSubmissionAndResultIntegrationTestService.java @@ -55,7 +55,7 @@ public class ProgrammingSubmissionAndResultIntegrationTestService { public ProgrammingExerciseParticipation participation; public void setUp_shouldSetSubmissionDateForBuildCorrectlyIfOnlyOnePushIsReceived(String userPrefix) { - Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, false, JAVA); + Course course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(false, JAVA); programmingExercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); programmingExercise = programmingExerciseRepository.findWithEagerStudentParticipationsStudentAndLegalSubmissionsById(programmingExercise.getId()).orElseThrow(); participation = participationUtilService.addStudentParticipationForProgrammingExercise(programmingExercise, userPrefix + "student1"); diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/util/HestiaUtilTestService.java b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingUtilTestService.java similarity index 96% rename from src/test/java/de/tum/cit/aet/artemis/programming/hestia/util/HestiaUtilTestService.java rename to src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingUtilTestService.java index 92e857ca6eea..52a58e3f8040 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/hestia/util/HestiaUtilTestService.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/util/ProgrammingUtilTestService.java @@ -1,5 +1,6 @@ -package de.tum.cit.aet.artemis.programming.hestia.util; +package de.tum.cit.aet.artemis.programming.util; +import static de.tum.cit.aet.artemis.programming.service.AbstractGitService.linkRepositoryForExistingGit; import static org.mockito.ArgumentMatchers.any; import static org.mockito.ArgumentMatchers.anyBoolean; import static org.mockito.ArgumentMatchers.eq; @@ -31,12 +32,9 @@ import de.tum.cit.aet.artemis.programming.test_repository.ProgrammingExerciseStudentParticipationTestRepository; import de.tum.cit.aet.artemis.programming.test_repository.ProgrammingExerciseTestRepository; import de.tum.cit.aet.artemis.programming.test_repository.ProgrammingSubmissionTestRepository; -import de.tum.cit.aet.artemis.programming.util.GitUtilService; -import de.tum.cit.aet.artemis.programming.util.LocalRepository; -import de.tum.cit.aet.artemis.programming.util.ProgrammingExerciseUtilService; /** - * Utility service specifically used for testing Hestia related functionality. + * Utility service specifically used for testing programming exercises. * This currently includes: * - Setting up a template repository * - Setting up a solution repository @@ -45,7 +43,7 @@ * In the future this service will be extended to make testing of the code hint generation easier. */ @Service -public class HestiaUtilTestService { +public class ProgrammingUtilTestService { @Autowired private GitService gitService; @@ -237,7 +235,7 @@ public ProgrammingSubmission setupSubmission(Map files, Programm anyBoolean()); var participation = participationUtilService.addStudentParticipationForProgrammingExerciseForLocalRepo(exercise, login, participationRepo.localRepoFile.toURI()); - doReturn(gitService.linkRepositoryForExistingGit(participationRepo.originRepoFile.toPath(), null, "main", true)).when(gitService).getBareRepository(any()); + doReturn(linkRepositoryForExistingGit(participationRepo.originRepoFile.toPath(), null, "main", true)).when(gitService).getBareRepository(any()); var submission = ParticipationFactory.generateProgrammingSubmission(true, commitsList.getFirst().getId().getName(), SubmissionType.MANUAL); participation = programmingExerciseStudentParticipationRepository .findWithSubmissionsByExerciseIdAndParticipationIds(exercise.getId(), Collections.singletonList(participation.getId())).getFirst(); diff --git a/src/test/java/de/tum/cit/aet/artemis/shared/architecture/module/AbstractModuleServiceArchitectureTest.java b/src/test/java/de/tum/cit/aet/artemis/shared/architecture/module/AbstractModuleServiceArchitectureTest.java index cd43dd981e51..c242dfdb768b 100644 --- a/src/test/java/de/tum/cit/aet/artemis/shared/architecture/module/AbstractModuleServiceArchitectureTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/shared/architecture/module/AbstractModuleServiceArchitectureTest.java @@ -3,7 +3,6 @@ import static com.tngtech.archunit.core.domain.JavaModifier.ABSTRACT; import static com.tngtech.archunit.core.domain.JavaModifier.FINAL; import static com.tngtech.archunit.lang.SimpleConditionEvent.violated; -import static com.tngtech.archunit.lang.syntax.ArchRuleDefinition.classes; import org.junit.jupiter.api.Test; import org.springframework.scheduling.annotation.Async; @@ -22,8 +21,8 @@ import de.tum.cit.aet.artemis.core.security.DomainUserDetailsService; import de.tum.cit.aet.artemis.core.security.jwt.JWTCookieService; import de.tum.cit.aet.artemis.lti.service.OAuth2JWKSService; +import de.tum.cit.aet.artemis.programming.service.GitDiffReportParserService; import de.tum.cit.aet.artemis.programming.service.localci.LocalCIWebsocketMessagingService; -import de.tum.cit.aet.artemis.programming.web.GitDiffReportParserService; import de.tum.cit.aet.artemis.shared.architecture.AbstractArchitectureTest; public abstract class AbstractModuleServiceArchitectureTest extends AbstractArchitectureTest implements ModuleArchitectureTest { diff --git a/src/test/javascript/spec/component/code-editor/code-editor-student-container.component.spec.ts b/src/test/javascript/spec/component/code-editor/code-editor-student-container.component.spec.ts index abe6ded99d34..2fef51c01555 100644 --- a/src/test/javascript/spec/component/code-editor/code-editor-student-container.component.spec.ts +++ b/src/test/javascript/spec/component/code-editor/code-editor-student-container.component.spec.ts @@ -10,7 +10,6 @@ import { MockProgrammingExerciseParticipationService } from '../../helpers/mocks import { GuidedTourService } from 'app/guided-tour/guided-tour.service'; import { SubmissionPolicyService } from 'app/exercises/programming/manage/services/submission-policy.service'; import { AlertService } from 'app/core/util/alert.service'; -import { ExerciseHintService } from 'app/exercises/shared/exercise-hint/shared/exercise-hint.service'; import { of } from 'rxjs'; import { ProgrammingExerciseStudentParticipation } from 'app/entities/participation/programming-exercise-student-participation.model'; import { ActivatedRoute } from '@angular/router'; @@ -43,7 +42,6 @@ describe('CodeEditorStudentContainerComponent', () => { MockProvider(GuidedTourService), MockProvider(SubmissionPolicyService), MockProvider(AlertService), - MockProvider(ExerciseHintService), ], }) .compileComponents() diff --git a/src/test/javascript/spec/component/exam/manage/programming-exam-diff.component.spec.ts b/src/test/javascript/spec/component/exam/manage/programming-exam-diff.component.spec.ts index 3a0d9d6e640c..ed85aa75ceb7 100644 --- a/src/test/javascript/spec/component/exam/manage/programming-exam-diff.component.spec.ts +++ b/src/test/javascript/spec/component/exam/manage/programming-exam-diff.component.spec.ts @@ -6,12 +6,12 @@ import { MockComponent, MockPipe } from 'ng-mocks'; import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; import { of } from 'rxjs'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from '../../../../../../main/webapp/app/entities/programming-exercise-git-diff-report.model'; import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; import { MockNgbModalService } from '../../../helpers/mocks/service/mock-ngb-modal.service'; -import { GitDiffReportModalComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report-modal.component'; -import { ProgrammingExerciseGitDiffEntry } from 'app/entities/hestia/programming-exercise-git-diff-entry.model'; +import { GitDiffReportModalComponent } from '../../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-report-modal.component'; +import { ProgrammingExerciseGitDiffEntry } from '../../../../../../main/webapp/app/entities/programming-exercise-git-diff-entry.model'; import { IncludedInScoreBadgeComponent } from 'app/exercises/shared/exercise-headers/included-in-score-badge.component'; import { CachedRepositoryFilesService } from 'app/exercises/programming/manage/services/cached-repository-files.service'; diff --git a/src/test/javascript/spec/component/exercise-hint/manage/exercise-hint-detail.component.spec.ts b/src/test/javascript/spec/component/exercise-hint/manage/exercise-hint-detail.component.spec.ts deleted file mode 100644 index bbc956f3715d..000000000000 --- a/src/test/javascript/spec/component/exercise-hint/manage/exercise-hint-detail.component.spec.ts +++ /dev/null @@ -1,37 +0,0 @@ -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { ActivatedRoute } from '@angular/router'; -import { of } from 'rxjs'; - -import { ExerciseHintDetailComponent } from 'app/exercises/shared/exercise-hint/manage/exercise-hint-detail.component'; -import { ArtemisTestModule } from '../../../test.module'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; - -describe('ExerciseHint Management Detail Component', () => { - let comp: ExerciseHintDetailComponent; - let fixture: ComponentFixture; - const exerciseHint = new ExerciseHint(); - exerciseHint.id = 123; - const route = { data: of({ exerciseHint }), params: of({ exerciseId: 1 }) } as any as ActivatedRoute; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - declarations: [ExerciseHintDetailComponent], - providers: [{ provide: ActivatedRoute, useValue: route }], - }) - .overrideTemplate(ExerciseHintDetailComponent, '') - .compileComponents(); - fixture = TestBed.createComponent(ExerciseHintDetailComponent); - comp = fixture.componentInstance; - }); - - describe('onInit', () => { - it('should call load all on init', () => { - // WHEN - comp.ngOnInit(); - - // THEN - expect(comp.exerciseHint).toEqual(expect.objectContaining({ id: 123 })); - }); - }); -}); diff --git a/src/test/javascript/spec/component/exercise-hint/manage/exercise-hint-update.component.spec.ts b/src/test/javascript/spec/component/exercise-hint/manage/exercise-hint-update.component.spec.ts deleted file mode 100644 index fe8282591d2b..000000000000 --- a/src/test/javascript/spec/component/exercise-hint/manage/exercise-hint-update.component.spec.ts +++ /dev/null @@ -1,239 +0,0 @@ -import { ComponentFixture, TestBed, fakeAsync, flush, tick } from '@angular/core/testing'; -import { FormBuilder, FormsModule } from '@angular/forms'; -import { of } from 'rxjs'; -import { HttpResponse } from '@angular/common/http'; - -import { ExerciseHintUpdateComponent } from 'app/exercises/shared/exercise-hint/manage/exercise-hint-update.component'; -import { ArtemisTestModule } from '../../../test.module'; -import { TranslateService } from '@ngx-translate/core'; -import { MockComponent, MockProvider } from 'ng-mocks'; -import { HelpIconComponent } from 'app/shared/components/help-icon.component'; -import { ExerciseHintService } from 'app/exercises/shared/exercise-hint/shared/exercise-hint.service'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; -import { ActivatedRoute } from '@angular/router'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { ProgrammingExerciseServerSideTask } from 'app/entities/hestia/programming-exercise-task.model'; -import { ProgrammingExerciseTestCase } from 'app/entities/programming/programming-exercise-test-case.model'; -import { ProgrammingExercise, ProgrammingLanguage } from 'app/entities/programming/programming-exercise.model'; -import { CodeHint } from 'app/entities/hestia/code-hint-model'; -import { CodeHintService } from 'app/exercises/shared/exercise-hint/services/code-hint.service'; -import { ProfileService } from 'app/shared/layouts/profiles/profile.service'; -import { MockProfileService } from '../../../helpers/mocks/service/mock-profile.service'; -import { IrisSettingsService } from 'app/iris/settings/shared/iris-settings.service'; -import { IrisSettings } from 'app/entities/iris/settings/iris-settings.model'; -import { ProfileInfo } from 'app/shared/layouts/profiles/profile-info.model'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { PROFILE_IRIS } from 'app/app.constants'; -import { MarkdownEditorMonacoComponent } from 'app/shared/markdown-editor/monaco/markdown-editor-monaco.component'; - -describe('ExerciseHint Management Update Component', () => { - let comp: ExerciseHintUpdateComponent; - let fixture: ComponentFixture; - let service: ExerciseHintService; - let codeHintService: CodeHintService; - let programmingExerciseService: ProgrammingExerciseService; - - const task1 = new ProgrammingExerciseServerSideTask(); - task1.id = 1; - task1.taskName = 'Task 1'; - task1.testCases = [new ProgrammingExerciseTestCase(), new ProgrammingExerciseTestCase()]; - - const task2 = new ProgrammingExerciseServerSideTask(); - task2.id = 2; - task2.taskName = 'Task 2'; - task2.testCases = [new ProgrammingExerciseTestCase(), new ProgrammingExerciseTestCase()]; - - const programmingExercise = new ProgrammingExercise(undefined, undefined); - programmingExercise.id = 15; - - const exerciseHint = new ExerciseHint(); - const route = { data: of({ exerciseHint, exercise: programmingExercise }), params: of({ courseId: 12 }) } as any as ActivatedRoute; - - beforeEach(fakeAsync(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule, FormsModule], - declarations: [ExerciseHintUpdateComponent, MockComponent(MarkdownEditorMonacoComponent), MockComponent(HelpIconComponent)], - providers: [ - FormBuilder, - MockProvider(ProgrammingExerciseService), - MockProvider(ExerciseHintService), - MockProvider(TranslateService), - MockProvider(IrisSettingsService), - MockProfileService, - { provide: ActivatedRoute, useValue: route }, - ], - }) - .compileComponents() - .then(() => { - fixture = TestBed.createComponent(ExerciseHintUpdateComponent); - comp = fixture.componentInstance; - - service = TestBed.inject(ExerciseHintService); - codeHintService = TestBed.inject(CodeHintService); - programmingExerciseService = TestBed.inject(ProgrammingExerciseService); - flush(); - }); - })); - - afterEach(() => { - exerciseHint.programmingExerciseTask = undefined; - jest.restoreAllMocks(); - }); - - it('should load params and data onInit', () => { - const exercise = new ProgrammingExercise(undefined, undefined); - exercise.programmingLanguage = ProgrammingLanguage.JAVA; - exercise.id = 15; - - comp.ngOnInit(); - - expect(comp.exercise?.id).toBe(15); - expect(comp.exerciseHint).toEqual(exerciseHint); - expect(comp.courseId).toBe(12); - }); - - it('should load and set tasks for exercise hint', fakeAsync(() => { - exerciseHint.id = 4; - comp.exerciseHint = exerciseHint; - const getTasksSpy = jest.spyOn(programmingExerciseService, 'getTasksAndTestsExtractedFromProblemStatement').mockReturnValue(of([task1, task2])); - comp.exerciseHint.programmingExerciseTask = task2; - - comp.ngOnInit(); - - expect(getTasksSpy).toHaveBeenCalledOnce(); - expect(getTasksSpy).toHaveBeenCalledWith(15); - tick(); - expect(comp.tasks).toEqual([task1, task2]); - expect(comp.exerciseHint.programmingExerciseTask).toEqual(task2); - })); - - it('should load and set first tasks to exercise hint', fakeAsync(() => { - comp.exerciseHint = exerciseHint; - const getTasksSpy = jest.spyOn(programmingExerciseService, 'getTasksAndTestsExtractedFromProblemStatement').mockReturnValue(of([task1, task2])); - - comp.ngOnInit(); - - expect(getTasksSpy).toHaveBeenCalledOnce(); - expect(getTasksSpy).toHaveBeenCalledWith(15); - tick(); - expect(comp.tasks).toEqual([task1, task2]); - expect(comp.exerciseHint.programmingExerciseTask).toEqual(task1); - })); - - it('should update description and content using iris', fakeAsync(() => { - // GIVEN - const codeHint1 = new CodeHint(); - codeHint1.id = 123; - codeHint1.programmingExerciseTask = task2; - codeHint1.solutionEntries = [new ProgrammingExerciseSolutionEntry()]; - const codeHint2 = new CodeHint(); - codeHint2.id = 123; - codeHint2.programmingExerciseTask = task2; - codeHint2.content = 'Hello Content'; - codeHint2.description = 'Hello Description'; - - jest.spyOn(codeHintService, 'generateDescriptionForCodeHint').mockReturnValue(of(new HttpResponse({ body: codeHint2 }))); - comp.exerciseHint = codeHint1; - comp.courseId = 1; - comp.exercise = programmingExercise; - - // WHEN - comp.generateDescriptionForCodeHint(); - tick(); - - // THEN - expect(codeHintService.generateDescriptionForCodeHint).toHaveBeenCalledWith(15, 123); - expect(comp.isGeneratingDescription).toBeFalse(); - expect(comp.exerciseHint.content).toBe('Hello Content'); - expect(comp.exerciseHint.description).toBe('Hello Description'); - })); - - it.each<[string[]]>([[[]], [[PROFILE_IRIS]]])( - 'should load iris settings only if profile iris is active', - fakeAsync((activeProfiles: string[]) => { - // Mock getProfileInfo to return activeProfiles - const profileService = TestBed.inject(ProfileService); - jest.spyOn(profileService, 'getProfileInfo').mockReturnValue(of({ activeProfiles } as any as ProfileInfo)); - - // Mock getTasksAndTestsExtractedFromProblemStatement - jest.spyOn(programmingExerciseService, 'getTasksAndTestsExtractedFromProblemStatement').mockReturnValue(of([])); - - const fakeSettings = {} as any as IrisSettings; - - // Mock getCombinedProgrammingExerciseSettings - const irisSettingsService = TestBed.inject(IrisSettingsService); - const getCombinedProgrammingExerciseSettingsSpy = jest.spyOn(irisSettingsService, 'getCombinedExerciseSettings').mockReturnValue(of(fakeSettings)); - - // Run ngOnInit - comp.ngOnInit(); - tick(); - - if (activeProfiles.includes(PROFILE_IRIS)) { - // Should have called getCombinedProgrammingExerciseSettings if 'iris' is active - expect(getCombinedProgrammingExerciseSettingsSpy).toHaveBeenCalledOnce(); - expect(comp.irisSettings).toBe(fakeSettings); - } else { - // Should not have called getCombinedProgrammingExerciseSettings if 'iris' is not active - expect(getCombinedProgrammingExerciseSettingsSpy).not.toHaveBeenCalled(); - expect(comp.irisSettings).toBeUndefined(); - } - }), - ); - - it('should generate descriptions', () => { - const codeHint = new CodeHint(); - codeHint.id = 123; - codeHint.programmingExerciseTask = task2; - codeHint.solutionEntries = [new ProgrammingExerciseSolutionEntry()]; - - comp.exerciseHint = codeHint; - comp.courseId = 1; - comp.exercise = programmingExercise; - - const generateDescSpy = jest.spyOn(codeHintService, 'generateDescriptionForCodeHint'); - - comp.generateDescriptionForCodeHint(); - - expect(generateDescSpy).toHaveBeenCalledOnce(); - expect(generateDescSpy).toHaveBeenCalledWith(15, 123); - }); - - describe('save', () => { - it('should call update service on save for existing entity', fakeAsync(() => { - // GIVEN - const entity = new ExerciseHint(); - entity.id = 123; - entity.programmingExerciseTask = task2; - jest.spyOn(service, 'update').mockReturnValue(of(new HttpResponse({ body: entity }))); - comp.exerciseHint = entity; - comp.courseId = 1; - comp.exercise = programmingExercise; - - // WHEN - comp.save(); - tick(); // simulate async - - // THEN - expect(service.update).toHaveBeenCalledWith(15, entity); - expect(comp.isSaving).toBeFalse(); - })); - - it('should call create service on save for new entity', fakeAsync(() => { - // GIVEN - const entity = new ExerciseHint(); - entity.programmingExerciseTask = task2; - jest.spyOn(service, 'create').mockReturnValue(of(new HttpResponse({ body: entity }))); - comp.exerciseHint = entity; - comp.courseId = 1; - comp.exercise = programmingExercise; - - // WHEN - comp.save(); - tick(); // simulate async - - // THEN - expect(service.create).toHaveBeenCalledWith(15, entity); - expect(comp.isSaving).toBeFalse(); - })); - }); -}); diff --git a/src/test/javascript/spec/component/exercise-hint/manage/exercise-hint.component.spec.ts b/src/test/javascript/spec/component/exercise-hint/manage/exercise-hint.component.spec.ts deleted file mode 100644 index b7b531efa5a8..000000000000 --- a/src/test/javascript/spec/component/exercise-hint/manage/exercise-hint.component.spec.ts +++ /dev/null @@ -1,90 +0,0 @@ -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { of } from 'rxjs'; -import { HttpHeaders, HttpResponse } from '@angular/common/http'; -import { ActivatedRoute } from '@angular/router'; - -import { ExerciseHintComponent } from 'app/exercises/shared/exercise-hint/manage/exercise-hint.component'; -import { ArtemisTestModule } from '../../../test.module'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; -import { ExerciseHintService } from 'app/exercises/shared/exercise-hint/shared/exercise-hint.service'; -import { MockActivatedRoute } from '../../../helpers/mocks/activated-route/mock-activated-route'; -import { EventManager } from 'app/core/util/event-manager.service'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; - -describe('ExerciseHint Management Component', () => { - let comp: ExerciseHintComponent; - let fixture: ComponentFixture; - let service: ExerciseHintService; - let eventManager: EventManager; - - const programmingExercise = new ProgrammingExercise(undefined, undefined); - programmingExercise.id = 15; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - declarations: [ExerciseHintComponent], - providers: [{ provide: ActivatedRoute, useValue: new MockActivatedRoute({ exerciseId: 15, exercise: programmingExercise }) }], - }) - .compileComponents() - .then(() => { - fixture = TestBed.createComponent(ExerciseHintComponent); - comp = fixture.componentInstance; - service = TestBed.inject(ExerciseHintService); - eventManager = TestBed.inject(EventManager); - }); - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should call load all on init with exerciseId from route', () => { - // GIVEN - const headers = new HttpHeaders().append('link', 'link;link'); - const hint = new ExerciseHint(); - hint.id = 123; - - jest.spyOn(service, 'findByExerciseId').mockReturnValue( - of( - new HttpResponse({ - body: [hint], - headers, - }), - ), - ); - - // WHEN - comp.ngOnInit(); - - // THEN - expect(service.findByExerciseId).toHaveBeenCalledOnce(); - expect(service.findByExerciseId).toHaveBeenCalledWith(15); - expect(comp.exerciseHints[0]).toEqual(expect.objectContaining({ id: 123 })); - }); - - it('should invoke hint deletion', () => { - const deleteHintMock = jest.spyOn(service, 'delete').mockReturnValue(of({} as HttpResponse)); - const broadcastSpy = jest.spyOn(eventManager, 'broadcast'); - const exerciseHint = new ExerciseHint(); - exerciseHint.id = 123; - comp.exerciseHints = [exerciseHint]; - comp.exercise = programmingExercise; - - comp.deleteExerciseHint(123); - - expect(deleteHintMock).toHaveBeenCalledOnce(); - expect(deleteHintMock).toHaveBeenCalledWith(15, 123); - expect(broadcastSpy).toHaveBeenCalledOnce(); - expect(broadcastSpy).toHaveBeenCalledWith({ - name: 'exerciseHintListModification', - content: 'Deleted an exerciseHint', - }); - }); - - it('should track item id', () => { - const exerciseHint = new ExerciseHint(); - exerciseHint.id = 1; - expect(comp.trackId(0, exerciseHint)).toBe(1); - }); -}); diff --git a/src/test/javascript/spec/component/exercise-hint/participate/exercise-hint-button-overlay.component.spec.ts b/src/test/javascript/spec/component/exercise-hint/participate/exercise-hint-button-overlay.component.spec.ts deleted file mode 100644 index 005b1c67b1a6..000000000000 --- a/src/test/javascript/spec/component/exercise-hint/participate/exercise-hint-button-overlay.component.spec.ts +++ /dev/null @@ -1,57 +0,0 @@ -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap'; - -import { ArtemisTestModule } from '../../../test.module'; -import { ExerciseHintButtonOverlayComponent } from 'app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; -import { MockNgbModalService } from '../../../helpers/mocks/service/mock-ngb-modal.service'; -import { ExerciseHintStudentDialogComponent } from 'app/exercises/shared/exercise-hint/participate/exercise-hint-student-dialog.component'; - -describe('Exercise Hint Button Overlay Component', () => { - let comp: ExerciseHintButtonOverlayComponent; - let fixture: ComponentFixture; - - let modalService: NgbModal; - - const availableExerciseHint = new ExerciseHint(); - availableExerciseHint.id = 1; - const activatedExerciseHint = new ExerciseHint(); - activatedExerciseHint.id = 2; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - providers: [{ provide: NgbModal, useClass: MockNgbModalService }], - }) - .compileComponents() - .then(() => { - fixture = TestBed.createComponent(ExerciseHintButtonOverlayComponent); - comp = fixture.componentInstance; - - modalService = TestBed.inject(NgbModal); - - comp.availableExerciseHints = [availableExerciseHint]; - comp.activatedExerciseHints = [activatedExerciseHint]; - }); - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should open modal with exercise hints', () => { - const componentInstance = { availableExerciseHints: [], activatedExerciseHints: [] }; - const result = new Promise((resolve) => resolve(true)); - const openModalSpy = jest.spyOn(modalService, 'open').mockReturnValue({ - componentInstance, - result, - }); - - comp.openModal(); - - expect(openModalSpy).toHaveBeenCalledOnce(); - expect(openModalSpy).toHaveBeenCalledWith(ExerciseHintStudentDialogComponent, { size: 'lg', backdrop: 'static' }); - expect(componentInstance.availableExerciseHints).toEqual([availableExerciseHint]); - expect(componentInstance.activatedExerciseHints).toEqual([activatedExerciseHint]); - }); -}); diff --git a/src/test/javascript/spec/component/exercise-hint/participate/exercise-hint-expandable.component.spec.ts b/src/test/javascript/spec/component/exercise-hint/participate/exercise-hint-expandable.component.spec.ts deleted file mode 100644 index 0dbdd2699087..000000000000 --- a/src/test/javascript/spec/component/exercise-hint/participate/exercise-hint-expandable.component.spec.ts +++ /dev/null @@ -1,82 +0,0 @@ -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { of } from 'rxjs'; -import { TranslateService } from '@ngx-translate/core'; -import { MockModule, MockPipe } from 'ng-mocks'; -import { HttpResponse } from '@angular/common/http'; -import { MatExpansionModule } from '@angular/material/expansion'; - -import { ArtemisTestModule } from '../../../test.module'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; -import { ExerciseHintResponse, ExerciseHintService } from 'app/exercises/shared/exercise-hint/shared/exercise-hint.service'; -import { ExerciseHintExpandableComponent } from 'app/exercises/shared/exercise-hint/participate/exercise-hint-expandable.component'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { StarRatingComponent } from 'app/exercises/shared/rating/star-rating/star-rating.component'; -import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; -import { MockTranslateService } from '../../../helpers/mocks/service/mock-translate.service'; -import { CastToCodeHintPipe } from 'app/exercises/shared/exercise-hint/services/code-hint-cast.pipe'; - -describe('Exercise Hint Expandable Component', () => { - let comp: ExerciseHintExpandableComponent; - let fixture: ComponentFixture; - - let service: ExerciseHintService; - - const exercise = new ProgrammingExercise(undefined, undefined); - exercise.id = 1; - const availableExerciseHint = new ExerciseHint(); - availableExerciseHint.id = 2; - availableExerciseHint.exercise = exercise; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule, MockModule(MatExpansionModule)], - declarations: [ExerciseHintExpandableComponent, MockPipe(ArtemisTranslatePipe), StarRatingComponent, CastToCodeHintPipe], - providers: [{ provide: TranslateService, useClass: MockTranslateService }], - }) - .compileComponents() - .then(() => { - fixture = TestBed.createComponent(ExerciseHintExpandableComponent); - comp = fixture.componentInstance; - - service = TestBed.inject(ExerciseHintService); - comp.exerciseHint = availableExerciseHint; - }); - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should load available exercise hint content', () => { - const activateHintSpy = jest.spyOn(service, 'activateExerciseHint').mockReturnValue(of({ body: availableExerciseHint } as ExerciseHintResponse)); - comp.displayHintContent(); - expect(activateHintSpy).toHaveBeenCalledOnce(); - expect(activateHintSpy).toHaveBeenCalledWith(1, 2); - expect(comp.expanded).toBeTrue(); - expect(comp.isLoading).toBeFalse(); - expect(comp.hasUsed).toBeTrue(); - expect(comp.exerciseHint).toEqual(availableExerciseHint); - }); - - it('should not load content when hint has already been used', () => { - comp.hasUsed = true; - const activateHintSpy = jest.spyOn(service, 'activateExerciseHint').mockReturnValue(of({ body: availableExerciseHint } as ExerciseHintResponse)); - comp.displayHintContent(); - expect(activateHintSpy).not.toHaveBeenCalled(); - }); - - it('should collapse exercise hint content', () => { - comp.collapseContent(); - expect(comp.expanded).toBeFalse(); - }); - - it('should rate exercise hint', () => { - const rateSpy = jest.spyOn(service, 'rateExerciseHint').mockReturnValue(of({} as HttpResponse)); - comp.exerciseHint = availableExerciseHint; - comp.onRate({ oldValue: 0, newValue: 4, starRating: new StarRatingComponent() }); - - expect(rateSpy).toHaveBeenCalledOnce(); - expect(rateSpy).toHaveBeenCalledWith(1, 2, 4); - expect(comp.exerciseHint.currentUserRating).toBe(4); - }); -}); diff --git a/src/test/javascript/spec/component/exercise-hint/shared/code-hint-container.component.spec.ts b/src/test/javascript/spec/component/exercise-hint/shared/code-hint-container.component.spec.ts deleted file mode 100644 index d7852f272f21..000000000000 --- a/src/test/javascript/spec/component/exercise-hint/shared/code-hint-container.component.spec.ts +++ /dev/null @@ -1,91 +0,0 @@ -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { ArtemisTestModule } from '../../../test.module'; -import { CodeHintContainerComponent } from 'app/exercises/shared/exercise-hint/shared/code-hint-container.component'; -import { CodeHint } from 'app/entities/hestia/code-hint-model'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { CodeHintService } from 'app/exercises/shared/exercise-hint/services/code-hint.service'; - -describe('ExerciseHint Management Component', () => { - let comp: CodeHintContainerComponent; - let fixture: ComponentFixture; - - let service: CodeHintService; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - declarations: [CodeHintContainerComponent], - }) - .compileComponents() - .then(() => { - fixture = TestBed.createComponent(CodeHintContainerComponent); - comp = fixture.componentInstance; - - service = TestBed.inject(CodeHintService); - }); - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should sort solution entries by file name', () => { - const solutionEntry1 = new ProgrammingExerciseSolutionEntry(); - solutionEntry1.line = 1; - solutionEntry1.filePath = 'b'; - const solutionEntry2 = new ProgrammingExerciseSolutionEntry(); - solutionEntry2.line = 1; - solutionEntry2.filePath = 'a'; - const codeHint = new CodeHint(); - codeHint.solutionEntries = [solutionEntry1, solutionEntry2]; - - comp.codeHint = codeHint; - - comp.ngOnInit(); - expect(comp.sortedSolutionEntries).toEqual([solutionEntry2, solutionEntry1]); - }); - - it('should sort solution entries for same file name but different line', () => { - const solutionEntry1 = new ProgrammingExerciseSolutionEntry(); - solutionEntry1.line = 3; - solutionEntry1.filePath = 'a'; - const solutionEntry2 = new ProgrammingExerciseSolutionEntry(); - solutionEntry2.line = 2; - solutionEntry2.filePath = 'a'; - const codeHint = new CodeHint(); - codeHint.solutionEntries = [solutionEntry1, solutionEntry2]; - - comp.codeHint = codeHint; - - comp.ngOnInit(); - expect(comp.sortedSolutionEntries).toEqual([solutionEntry2, solutionEntry1]); - }); - - it('should remove solution entry', () => { - const solutionEntry1 = new ProgrammingExerciseSolutionEntry(); - solutionEntry1.line = 3; - solutionEntry1.filePath = 'a'; - solutionEntry1.id = 1; - const solutionEntry2 = new ProgrammingExerciseSolutionEntry(); - solutionEntry2.line = 2; - solutionEntry2.filePath = 'a'; - - const exercise = new ProgrammingExercise(undefined, undefined); - exercise.id = 2; - - const codeHint = new CodeHint(); - codeHint.solutionEntries = [solutionEntry1, solutionEntry2]; - codeHint.exercise = exercise; - codeHint.id = 3; - - comp.codeHint = codeHint; - - const removeEntrySpy = jest.spyOn(service, 'removeSolutionEntryFromCodeHint'); - - comp.removeEntryFromCodeHint(solutionEntry1.id); - - expect(removeEntrySpy).toHaveBeenCalledOnce(); - expect(removeEntrySpy).toHaveBeenCalledWith(2, 3, 1); - }); -}); diff --git a/src/test/javascript/spec/component/exercise-hint/shared/solution-entry.component.spec.ts b/src/test/javascript/spec/component/exercise-hint/shared/solution-entry.component.spec.ts deleted file mode 100644 index 4b6bc884a62e..000000000000 --- a/src/test/javascript/spec/component/exercise-hint/shared/solution-entry.component.spec.ts +++ /dev/null @@ -1,55 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; -import { MockComponent, MockPipe } from 'ng-mocks'; -import { SolutionEntryComponent } from 'app/exercises/shared/exercise-hint/shared/solution-entry.component'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { MonacoEditorComponent } from 'app/shared/monaco-editor/monaco-editor.component'; - -describe('Solution Entry Component', () => { - let comp: SolutionEntryComponent; - let fixture: ComponentFixture; - let solutionEntry: ProgrammingExerciseSolutionEntry; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - declarations: [SolutionEntryComponent, MockComponent(MonacoEditorComponent), MockPipe(ArtemisTranslatePipe)], - }).compileComponents(); - fixture = TestBed.createComponent(SolutionEntryComponent); - comp = fixture.componentInstance; - - comp.solutionEntry = new ProgrammingExerciseSolutionEntry(); - comp.solutionEntry.id = 123; - comp.solutionEntry.filePath = '/src/de/Test.java'; - comp.solutionEntry.line = 1; - comp.solutionEntry.code = 'ABC'; - - solutionEntry = comp.solutionEntry; - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should correctly setup editors', () => { - const setStartLineNumberStub = jest.spyOn(comp.editor, 'setStartLineNumber').mockImplementation(); - const changeModelStub = jest.spyOn(comp.editor, 'changeModel').mockImplementation(); - fixture.detectChanges(); - - expect(setStartLineNumberStub).toHaveBeenCalledExactlyOnceWith(solutionEntry.line); - expect(changeModelStub).toHaveBeenCalledExactlyOnceWith(solutionEntry.filePath, solutionEntry.code); - }); - - it('should update the editor height', () => { - const contentHeight = 123; - comp.onContentSizeChange(contentHeight); - expect(comp.editorHeight).toEqual(contentHeight); - }); - - it('should update the solution entry code', () => { - const newCode = 'DEF'; - comp.onEditorContentChange(newCode); - expect(comp.solutionEntry.code).toEqual(newCode); - }); -}); diff --git a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-file-panel-title.component.spec.ts b/src/test/javascript/spec/component/git-diff-report/git-diff-file-panel-title.component.spec.ts similarity index 89% rename from src/test/javascript/spec/component/hestia/git-diff-report/git-diff-file-panel-title.component.spec.ts rename to src/test/javascript/spec/component/git-diff-report/git-diff-file-panel-title.component.spec.ts index 86380ebed0a7..292f65f903cd 100644 --- a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-file-panel-title.component.spec.ts +++ b/src/test/javascript/spec/component/git-diff-report/git-diff-file-panel-title.component.spec.ts @@ -1,6 +1,6 @@ -import { ArtemisTestModule } from '../../../test.module'; +import { ArtemisTestModule } from '../../test.module'; import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { GitDiffFilePanelTitleComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-file-panel-title.component'; +import { GitDiffFilePanelTitleComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel-title.component'; describe('GitDiffFilePanelTitleComponent', () => { let comp: GitDiffFilePanelTitleComponent; diff --git a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-file-panel.component.spec.ts b/src/test/javascript/spec/component/git-diff-report/git-diff-file-panel.component.spec.ts similarity index 85% rename from src/test/javascript/spec/component/hestia/git-diff-report/git-diff-file-panel.component.spec.ts rename to src/test/javascript/spec/component/git-diff-report/git-diff-file-panel.component.spec.ts index 26627dba3e8f..e62d3af1f108 100644 --- a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-file-panel.component.spec.ts +++ b/src/test/javascript/spec/component/git-diff-report/git-diff-file-panel.component.spec.ts @@ -1,13 +1,13 @@ -import { ArtemisTestModule } from '../../../test.module'; +import { ArtemisTestModule } from '../../test.module'; import { ComponentFixture, TestBed } from '@angular/core/testing'; import { MockComponent, MockDirective } from 'ng-mocks'; -import { GitDiffLineStatComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component'; -import { ProgrammingExerciseGitDiffEntry } from 'app/entities/hestia/programming-exercise-git-diff-entry.model'; -import { GitDiffFilePanelComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-file-panel.component'; +import { GitDiffLineStatComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-line-stat.component'; +import { ProgrammingExerciseGitDiffEntry } from '../../../../../main/webapp/app/entities/programming-exercise-git-diff-entry.model'; +import { GitDiffFilePanelComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel.component'; import { NgbAccordionBody, NgbAccordionButton, NgbAccordionCollapse, NgbAccordionDirective, NgbAccordionHeader, NgbAccordionItem } from '@ng-bootstrap/ng-bootstrap'; -import { GitDiffFileComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-file.component'; -import { GitDiffFilePanelTitleComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-file-panel-title.component'; -import { MonacoDiffEditorComponent } from '../../../../../../main/webapp/app/shared/monaco-editor/monaco-diff-editor.component'; +import { GitDiffFileComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-file.component'; +import { GitDiffFilePanelTitleComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel-title.component'; +import { MonacoDiffEditorComponent } from '../../../../../main/webapp/app/shared/monaco-editor/monaco-diff-editor.component'; describe('GitDiffFilePanelComponent', () => { let comp: GitDiffFilePanelComponent; diff --git a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-file.component.spec.ts b/src/test/javascript/spec/component/git-diff-report/git-diff-file.component.spec.ts similarity index 83% rename from src/test/javascript/spec/component/hestia/git-diff-report/git-diff-file.component.spec.ts rename to src/test/javascript/spec/component/git-diff-report/git-diff-file.component.spec.ts index 17007596e8a8..2eef960038b3 100644 --- a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-file.component.spec.ts +++ b/src/test/javascript/spec/component/git-diff-report/git-diff-file.component.spec.ts @@ -1,8 +1,8 @@ -import { ArtemisTestModule } from '../../../test.module'; +import { ArtemisTestModule } from '../../test.module'; import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { GitDiffFileComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-file.component'; -import { MockResizeObserver } from '../../../helpers/mocks/service/mock-resize-observer'; -import { MonacoDiffEditorComponent } from '../../../../../../main/webapp/app/shared/monaco-editor/monaco-diff-editor.component'; +import { GitDiffFileComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-file.component'; +import { MockResizeObserver } from '../../helpers/mocks/service/mock-resize-observer'; +import { MonacoDiffEditorComponent } from '../../../../../main/webapp/app/shared/monaco-editor/monaco-diff-editor.component'; function getDiffEntryWithPaths(previousFilePath?: string, filePath?: string) { return { diff --git a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-line-stat.component.spec.ts b/src/test/javascript/spec/component/git-diff-report/git-diff-line-stat.component.spec.ts similarity index 81% rename from src/test/javascript/spec/component/hestia/git-diff-report/git-diff-line-stat.component.spec.ts rename to src/test/javascript/spec/component/git-diff-report/git-diff-line-stat.component.spec.ts index 2ea540371804..dafced43bada 100644 --- a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-line-stat.component.spec.ts +++ b/src/test/javascript/spec/component/git-diff-report/git-diff-line-stat.component.spec.ts @@ -1,7 +1,7 @@ -import { ArtemisTestModule } from '../../../test.module'; +import { ArtemisTestModule } from '../../test.module'; import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { GitDiffLineStatComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component'; -import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; +import { GitDiffLineStatComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-line-stat.component'; +import { ArtemisTranslatePipe } from '../../../../../main/webapp/app/shared/pipes/artemis-translate.pipe'; import { MockPipe } from 'ng-mocks'; describe('GitDiffLineStatComponent', () => { diff --git a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-modal.component.spec.ts b/src/test/javascript/spec/component/git-diff-report/git-diff-modal.component.spec.ts similarity index 91% rename from src/test/javascript/spec/component/hestia/git-diff-report/git-diff-modal.component.spec.ts rename to src/test/javascript/spec/component/git-diff-report/git-diff-modal.component.spec.ts index 7fc9a8693d0e..e2da096b0e41 100644 --- a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-modal.component.spec.ts +++ b/src/test/javascript/spec/component/git-diff-report/git-diff-modal.component.spec.ts @@ -1,13 +1,13 @@ -import { GitDiffReportComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report.component'; -import { ArtemisTestModule } from '../../../test.module'; -import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; +import { GitDiffReportComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-report.component'; +import { ArtemisTestModule } from '../../test.module'; +import { ArtemisTranslatePipe } from '../../../../../main/webapp/app/shared/pipes/artemis-translate.pipe'; import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { GitDiffReportModalComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report-modal.component'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; +import { GitDiffReportModalComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-report-modal.component'; +import { ProgrammingExerciseService } from '../../../../../main/webapp/app/exercises/programming/manage/services/programming-exercise.service'; import { of, throwError } from 'rxjs'; -import { ProgrammingExerciseParticipationService } from 'app/exercises/programming/manage/services/programming-exercise-participation.service'; +import { ProgrammingExerciseParticipationService } from '../../../../../main/webapp/app/exercises/programming/manage/services/programming-exercise-participation.service'; import { MockComponent, MockPipe } from 'ng-mocks'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from '../../../../../main/webapp/app/entities/programming-exercise-git-diff-report.model'; import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; describe('GitDiffReportModalComponent', () => { diff --git a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-report.component.spec.ts b/src/test/javascript/spec/component/git-diff-report/git-diff-report.component.spec.ts similarity index 90% rename from src/test/javascript/spec/component/hestia/git-diff-report/git-diff-report.component.spec.ts rename to src/test/javascript/spec/component/git-diff-report/git-diff-report.component.spec.ts index 143c150edc57..8e8a531d3a1b 100644 --- a/src/test/javascript/spec/component/hestia/git-diff-report/git-diff-report.component.spec.ts +++ b/src/test/javascript/spec/component/git-diff-report/git-diff-report.component.spec.ts @@ -1,13 +1,13 @@ -import { ArtemisTestModule } from '../../../test.module'; +import { ArtemisTestModule } from '../../test.module'; import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; +import { ArtemisTranslatePipe } from '../../../../../main/webapp/app/shared/pipes/artemis-translate.pipe'; import { MockComponent, MockPipe } from 'ng-mocks'; -import { GitDiffLineStatComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-line-stat.component'; -import { GitDiffReportComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report.component'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; -import { ProgrammingExerciseGitDiffEntry } from 'app/entities/hestia/programming-exercise-git-diff-entry.model'; -import { NgbTooltipMocksModule } from '../../../helpers/mocks/directive/ngbTooltipMocks.module'; -import { GitDiffFilePanelComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-file-panel.component'; +import { GitDiffLineStatComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-line-stat.component'; +import { GitDiffReportComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-report.component'; +import { ProgrammingExerciseGitDiffReport } from '../../../../../main/webapp/app/entities/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffEntry } from '../../../../../main/webapp/app/entities/programming-exercise-git-diff-entry.model'; +import { NgbTooltipMocksModule } from '../../helpers/mocks/directive/ngbTooltipMocks.module'; +import { GitDiffFilePanelComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-file-panel.component'; describe('ProgrammingExerciseGitDiffReport Component', () => { let comp: GitDiffReportComponent; diff --git a/src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-overview.component.spec.ts b/src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-overview.component.spec.ts deleted file mode 100644 index 360a08238306..000000000000 --- a/src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-overview.component.spec.ts +++ /dev/null @@ -1,176 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { CodeHint, CodeHintGenerationStep } from 'app/entities/hestia/code-hint-model'; -import { CodeHintGenerationOverviewComponent } from 'app/exercises/programming/hestia/generation-overview/code-hint-generation-overview/code-hint-generation-overview.component'; -import { MockActivatedRoute } from '../../../helpers/mocks/activated-route/mock-activated-route'; -import { ActivatedRoute } from '@angular/router'; -import { CoverageReport } from 'app/entities/hestia/coverage-report.model'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; - -describe('CodeHintGenerationOverview Component', () => { - let comp: CodeHintGenerationOverviewComponent; - let fixture: ComponentFixture; - - const exercise = new ProgrammingExercise(undefined, undefined); - exercise.id = 1; - exercise.buildConfig!.testwiseCoverageEnabled = true; - - let latestStepSpy: jest.SpyInstance; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - declarations: [CodeHintGenerationOverviewComponent], - providers: [{ provide: ActivatedRoute, useValue: new MockActivatedRoute({ exercise }) }], - }).compileComponents(); - fixture = TestBed.createComponent(CodeHintGenerationOverviewComponent); - comp = fixture.componentInstance; - - latestStepSpy = jest.spyOn(comp, 'setLatestPerformedStep'); - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should set all step status on init', () => { - comp.ngOnInit(); - - const expectedStatus = new Map(); - expectedStatus.set(CodeHintGenerationStep.GIT_DIFF, false); - expectedStatus.set(CodeHintGenerationStep.COVERAGE, false); - expectedStatus.set(CodeHintGenerationStep.SOLUTION_ENTRIES, false); - expectedStatus.set(CodeHintGenerationStep.CODE_HINTS, false); - - expect(comp.isPerformedByStep).toEqual(expectedStatus); - expect(comp.allowBehavioralEntryGeneration).toBeTrue(); - expect(comp.currentStep).toEqual(CodeHintGenerationStep.GIT_DIFF); - }); - - it('should set next step as current step', () => { - comp.currentStep = CodeHintGenerationStep.GIT_DIFF; - comp.onNextStep(); - expect(comp.currentStep).toBe(CodeHintGenerationStep.COVERAGE); - }); - - it('should set previous step as current step', () => { - comp.currentStep = CodeHintGenerationStep.COVERAGE; - comp.onPreviousStep(); - expect(comp.currentStep).toBe(CodeHintGenerationStep.GIT_DIFF); - }); - - it('should check if next step is available', () => { - const expectedStatus = new Map(); - comp.isPerformedByStep = expectedStatus; - comp.currentStep = CodeHintGenerationStep.GIT_DIFF; - expectedStatus.set(CodeHintGenerationStep.GIT_DIFF, false); - - let isAvailable = comp.isNextStepAvailable(); - expect(isAvailable).toBeFalse(); - - expectedStatus.set(CodeHintGenerationStep.GIT_DIFF, true); - - isAvailable = comp.isNextStepAvailable(); - expect(isAvailable).toBeTrue(); - }); - - it('should set specific step as current step', () => { - comp.currentStep = CodeHintGenerationStep.GIT_DIFF; - comp.onStepChange(CodeHintGenerationStep.COVERAGE); - expect(comp.currentStep).toBe(CodeHintGenerationStep.COVERAGE); - }); - - it('should update diff step status', () => { - latestStepSpy.mockReturnValue(undefined); - comp.isPerformedByStep = new Map(); - comp.onDiffReportLoaded(undefined); - - expect(latestStepSpy).toHaveBeenCalledOnce(); - expect(latestStepSpy).toHaveBeenCalledWith(CodeHintGenerationStep.GIT_DIFF); - expect(comp.isPerformedByStep.get(CodeHintGenerationStep.GIT_DIFF)).toBeFalse(); - - comp.onDiffReportLoaded(new ProgrammingExerciseGitDiffReport()); - expect(latestStepSpy).toHaveBeenCalledTimes(2); - expect(latestStepSpy).toHaveBeenNthCalledWith(2, CodeHintGenerationStep.GIT_DIFF); - expect(comp.isPerformedByStep.get(CodeHintGenerationStep.GIT_DIFF)).toBeTrue(); - }); - - it('should update coverage step status', () => { - latestStepSpy.mockReturnValue(undefined); - comp.isPerformedByStep = new Map(); - comp.onCoverageReportLoaded(undefined); - - expect(latestStepSpy).toHaveBeenCalledOnce(); - expect(latestStepSpy).toHaveBeenCalledWith(CodeHintGenerationStep.COVERAGE); - expect(comp.isPerformedByStep.get(CodeHintGenerationStep.COVERAGE)).toBeFalse(); - - comp.onCoverageReportLoaded(new CoverageReport()); - expect(latestStepSpy).toHaveBeenCalledTimes(2); - expect(latestStepSpy).toHaveBeenNthCalledWith(2, CodeHintGenerationStep.COVERAGE); - expect(comp.isPerformedByStep.get(CodeHintGenerationStep.COVERAGE)).toBeTrue(); - }); - - it('should update solution entry step status', () => { - latestStepSpy.mockReturnValue(undefined); - comp.isPerformedByStep = new Map(); - comp.onSolutionEntryChanges(undefined); - - expect(latestStepSpy).toHaveBeenCalledOnce(); - expect(latestStepSpy).toHaveBeenCalledWith(CodeHintGenerationStep.SOLUTION_ENTRIES); - expect(comp.isPerformedByStep.get(CodeHintGenerationStep.SOLUTION_ENTRIES)).toBeFalse(); - - comp.onSolutionEntryChanges([]); - expect(latestStepSpy).toHaveBeenCalledTimes(2); - expect(latestStepSpy).toHaveBeenNthCalledWith(2, CodeHintGenerationStep.SOLUTION_ENTRIES); - expect(comp.isPerformedByStep.get(CodeHintGenerationStep.SOLUTION_ENTRIES)).toBeFalse(); - - comp.onSolutionEntryChanges([new ProgrammingExerciseSolutionEntry()]); - expect(latestStepSpy).toHaveBeenCalledTimes(3); - expect(latestStepSpy).toHaveBeenNthCalledWith(3, CodeHintGenerationStep.SOLUTION_ENTRIES); - expect(comp.isPerformedByStep.get(CodeHintGenerationStep.SOLUTION_ENTRIES)).toBeTrue(); - }); - - it('should update code hint step status', () => { - latestStepSpy.mockReturnValue(undefined); - comp.isPerformedByStep = new Map(); - comp.onCodeHintsLoaded(undefined); - - expect(latestStepSpy).toHaveBeenCalledOnce(); - expect(latestStepSpy).toHaveBeenCalledWith(CodeHintGenerationStep.CODE_HINTS); - expect(comp.isPerformedByStep.get(CodeHintGenerationStep.CODE_HINTS)).toBeFalse(); - - comp.onCodeHintsLoaded([]); - expect(latestStepSpy).toHaveBeenCalledTimes(2); - expect(latestStepSpy).toHaveBeenNthCalledWith(2, CodeHintGenerationStep.CODE_HINTS); - expect(comp.isPerformedByStep.get(CodeHintGenerationStep.CODE_HINTS)).toBeFalse(); - - comp.onCodeHintsLoaded([new CodeHint()]); - expect(latestStepSpy).toHaveBeenCalledTimes(3); - expect(latestStepSpy).toHaveBeenNthCalledWith(3, CodeHintGenerationStep.CODE_HINTS); - expect(comp.isPerformedByStep.get(CodeHintGenerationStep.CODE_HINTS)).toBeTrue(); - }); - - it('should not update latest performed step for update on previous step', () => { - comp.currentStep = CodeHintGenerationStep.SOLUTION_ENTRIES; - comp.isPerformedByStep = new Map(); - comp.isPerformedByStep.set(CodeHintGenerationStep.SOLUTION_ENTRIES, true); - - comp.isPerformedByStep.set(CodeHintGenerationStep.GIT_DIFF, true); - comp.setLatestPerformedStep(CodeHintGenerationStep.GIT_DIFF); - - expect(comp.currentStep).toBe(CodeHintGenerationStep.SOLUTION_ENTRIES); - }); - - it('should select latest performed step for update on later step', () => { - comp.currentStep = CodeHintGenerationStep.GIT_DIFF; - comp.isPerformedByStep = new Map(); - comp.isPerformedByStep.set(CodeHintGenerationStep.GIT_DIFF, true); - - comp.isPerformedByStep.set(CodeHintGenerationStep.SOLUTION_ENTRIES, true); - comp.setLatestPerformedStep(CodeHintGenerationStep.SOLUTION_ENTRIES); - - expect(comp.currentStep).toBe(CodeHintGenerationStep.SOLUTION_ENTRIES); - }); -}); diff --git a/src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-status.component.spec.ts b/src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-status.component.spec.ts deleted file mode 100644 index 7e0443998f36..000000000000 --- a/src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-status.component.spec.ts +++ /dev/null @@ -1,28 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { CodeHintGenerationStep } from 'app/entities/hestia/code-hint-model'; -import { CodeHintGenerationStatusComponent } from 'app/exercises/programming/hestia/generation-overview/code-hint-generation-status/code-hint-generation-status.component'; - -describe('CodeHintGenerationStatus Component', () => { - let comp: CodeHintGenerationStatusComponent; - let fixture: ComponentFixture; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - }).compileComponents(); - fixture = TestBed.createComponent(CodeHintGenerationStatusComponent); - comp = fixture.componentInstance; - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should select step', () => { - const stepChangeSpy = jest.spyOn(comp.onStepChange, 'emit'); - comp.onSelectStep(CodeHintGenerationStep.GIT_DIFF); - expect(stepChangeSpy).toHaveBeenCalledOnce(); - expect(stepChangeSpy).toHaveBeenCalledWith(CodeHintGenerationStep.GIT_DIFF); - }); -}); diff --git a/src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-step.component.spec.ts b/src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-step.component.spec.ts deleted file mode 100644 index f0e9e9999233..000000000000 --- a/src/test/javascript/spec/component/hestia/generation-overview/code-hint-generation-step.component.spec.ts +++ /dev/null @@ -1,69 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { of } from 'rxjs'; -import { CodeHintGenerationStepComponent } from 'app/exercises/programming/hestia/generation-overview/steps/code-hint-generation-step/code-hint-generation-step.component'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { CodeHintService } from 'app/exercises/shared/exercise-hint/services/code-hint.service'; -import { CodeHint } from 'app/entities/hestia/code-hint-model'; - -describe('CodeHintGenerationStep Component', () => { - let comp: CodeHintGenerationStepComponent; - let fixture: ComponentFixture; - - let exerciseService: ProgrammingExerciseService; - let codeHintService: CodeHintService; - - let onHintsLoadedSpy: jest.SpyInstance; - - let exercise: ProgrammingExercise; - let codeHints: CodeHint[]; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - }).compileComponents(); - fixture = TestBed.createComponent(CodeHintGenerationStepComponent); - comp = fixture.componentInstance; - - exerciseService = TestBed.inject(ProgrammingExerciseService); - codeHintService = TestBed.inject(CodeHintService); - - onHintsLoadedSpy = jest.spyOn(comp.onCodeHintsLoaded, 'emit'); - - exercise = new ProgrammingExercise(undefined, undefined); - exercise.id = 1; - comp.exercise = exercise; - - codeHints = [new CodeHint(), new CodeHint()]; - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should load all code hints on init', () => { - const loadHintsSpy = jest.spyOn(exerciseService, 'getCodeHintsForExercise').mockReturnValue(of(codeHints)); - - comp.ngOnInit(); - - expect(loadHintsSpy).toHaveBeenCalledOnce(); - expect(loadHintsSpy).toHaveBeenCalledWith(1); - expect(comp.codeHints).toEqual(codeHints); - expect(comp.isLoading).toBeFalse(); - expect(onHintsLoadedSpy).toHaveBeenCalledOnce(); - }); - - it('should generate code hints', () => { - const generateHintsSpy = jest.spyOn(codeHintService, 'generateCodeHintsForExercise').mockReturnValue(of(codeHints)); - - comp.generateCodeHints(true, 'recreateHintsButton'); - - expect(generateHintsSpy).toHaveBeenCalledOnce(); - expect(generateHintsSpy).toHaveBeenCalledWith(1, true); - expect(comp.codeHints).toEqual(codeHints); - expect(comp.isLoading).toBeFalse(); - expect(onHintsLoadedSpy).toHaveBeenCalledOnce(); - expect(onHintsLoadedSpy).toHaveBeenCalledWith(codeHints); - }); -}); diff --git a/src/test/javascript/spec/component/hestia/generation-overview/coverage-generation-step.component.spec.ts b/src/test/javascript/spec/component/hestia/generation-overview/coverage-generation-step.component.spec.ts deleted file mode 100644 index cda7ed0c1f60..000000000000 --- a/src/test/javascript/spec/component/hestia/generation-overview/coverage-generation-step.component.spec.ts +++ /dev/null @@ -1,65 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { of } from 'rxjs'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { CoverageGenerationStepComponent } from 'app/exercises/programming/hestia/generation-overview/steps/coverage-generation-step/coverage-generation-step.component'; -import { CoverageReport } from 'app/entities/hestia/coverage-report.model'; - -describe('CoverageGenerationStep Component', () => { - let comp: CoverageGenerationStepComponent; - let fixture: ComponentFixture; - - let exerciseService: ProgrammingExerciseService; - - let onCoverageLoadedSpy: jest.SpyInstance; - - let exercise: ProgrammingExercise; - let fileContentByPath: Map; - let coverageReport: CoverageReport; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - }).compileComponents(); - fixture = TestBed.createComponent(CoverageGenerationStepComponent); - comp = fixture.componentInstance; - - exerciseService = TestBed.inject(ProgrammingExerciseService); - - onCoverageLoadedSpy = jest.spyOn(comp.onCoverageLoaded, 'emit'); - - exercise = new ProgrammingExercise(undefined, undefined); - exercise.id = 1; - comp.exercise = exercise; - - fileContentByPath = new Map(); - fileContentByPath.set('A.java', 'abc'); - fileContentByPath.set('B.java', 'def'); - - coverageReport = new CoverageReport(); - coverageReport.id = 2; - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should load all code hints on init', () => { - const loadFilesSpy = jest.spyOn(exerciseService, 'getSolutionRepositoryTestFilesWithContent').mockReturnValue(of(fileContentByPath)); - const loadReportSpy = jest.spyOn(exerciseService, 'getLatestFullTestwiseCoverageReport').mockReturnValue(of(coverageReport)); - - comp.ngOnInit(); - - expect(loadFilesSpy).toHaveBeenCalledOnce(); - expect(loadFilesSpy).toHaveBeenCalledWith(1); - expect(loadReportSpy).toHaveBeenCalledOnce(); - expect(loadReportSpy).toHaveBeenCalledWith(1); - - expect(comp.fileContentByPath).toEqual(fileContentByPath); - expect(comp.coverageReport).toEqual(coverageReport); - expect(comp.isLoading).toBeFalse(); - expect(onCoverageLoadedSpy).toHaveBeenCalledOnce(); - expect(onCoverageLoadedSpy).toHaveBeenCalledWith(coverageReport); - }); -}); diff --git a/src/test/javascript/spec/component/hestia/generation-overview/diff-generation-step.component.spec.ts b/src/test/javascript/spec/component/hestia/generation-overview/diff-generation-step.component.spec.ts deleted file mode 100644 index 6d9f4cca6346..000000000000 --- a/src/test/javascript/spec/component/hestia/generation-overview/diff-generation-step.component.spec.ts +++ /dev/null @@ -1,67 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { of } from 'rxjs'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { DiffGenerationStepComponent } from 'app/exercises/programming/hestia/generation-overview/steps/diff-generation-step/diff-generation-step.component'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; - -describe('DiffGenerationStep Component', () => { - let comp: DiffGenerationStepComponent; - let fixture: ComponentFixture; - - let exerciseService: ProgrammingExerciseService; - - let onGitDiffLoadedSpy: jest.SpyInstance; - - let exercise: ProgrammingExercise; - let fileContentByPath: Map; - let gitDiffReport: ProgrammingExerciseGitDiffReport; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - }).compileComponents(); - fixture = TestBed.createComponent(DiffGenerationStepComponent); - comp = fixture.componentInstance; - - exerciseService = TestBed.inject(ProgrammingExerciseService); - - onGitDiffLoadedSpy = jest.spyOn(comp.onGitDiffLoaded, 'emit'); - - exercise = new ProgrammingExercise(undefined, undefined); - exercise.id = 1; - comp.exercise = exercise; - - fileContentByPath = new Map(); - fileContentByPath.set('A.java', 'abc'); - fileContentByPath.set('B.java', 'def'); - - gitDiffReport = new ProgrammingExerciseGitDiffReport(); - gitDiffReport.programmingExercise = exercise; - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should load all code hints on init', () => { - const loadTemplateFilesSpy = jest.spyOn(exerciseService, 'getTemplateRepositoryTestFilesWithContent').mockReturnValue(of(fileContentByPath)); - const loadSolutionFilesSpy = jest.spyOn(exerciseService, 'getSolutionRepositoryTestFilesWithContent').mockReturnValue(of(fileContentByPath)); - const loadReportSpy = jest.spyOn(exerciseService, 'getDiffReport').mockReturnValue(of(gitDiffReport)); - - comp.ngOnInit(); - - expect(loadTemplateFilesSpy).toHaveBeenCalledOnce(); - expect(loadTemplateFilesSpy).toHaveBeenCalledWith(1); - expect(loadSolutionFilesSpy).toHaveBeenCalledOnce(); - expect(loadSolutionFilesSpy).toHaveBeenCalledWith(1); - expect(loadReportSpy).toHaveBeenCalledOnce(); - expect(loadReportSpy).toHaveBeenCalledWith(1); - - expect(comp.gitDiffReport).toEqual(gitDiffReport); - expect(comp.isLoading).toBeFalse(); - expect(onGitDiffLoadedSpy).toHaveBeenCalledOnce(); - expect(onGitDiffLoadedSpy).toHaveBeenCalledWith(gitDiffReport); - }); -}); diff --git a/src/test/javascript/spec/component/hestia/generation-overview/manual-solution-entry-creation-modal.component.spec.ts b/src/test/javascript/spec/component/hestia/generation-overview/manual-solution-entry-creation-modal.component.spec.ts deleted file mode 100644 index 9720294dde47..000000000000 --- a/src/test/javascript/spec/component/hestia/generation-overview/manual-solution-entry-creation-modal.component.spec.ts +++ /dev/null @@ -1,97 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { ManualSolutionEntryCreationModalComponent } from 'app/exercises/programming/hestia/generation-overview/manual-solution-entry-creation-modal/manual-solution-entry-creation-modal.component'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { ProgrammingExerciseSolutionEntryService } from 'app/exercises/shared/exercise-hint/services/programming-exercise-solution-entry.service'; -import { ProgrammingExerciseTestCase } from 'app/entities/programming/programming-exercise-test-case.model'; -import { of } from 'rxjs'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; -import { SolutionEntryComponent } from 'app/exercises/shared/exercise-hint/shared/solution-entry.component'; -import { MockComponent } from 'ng-mocks'; - -describe('ManualSolutionEntryCreationModal Component', () => { - let comp: ManualSolutionEntryCreationModalComponent; - let fixture: ComponentFixture; - - let exerciseService: ProgrammingExerciseService; - let solutionEntryService: ProgrammingExerciseSolutionEntryService; - let modalService: NgbActiveModal; - - let testCases: ProgrammingExerciseTestCase[]; - const solutionFiles = ['A.java', 'B.java']; - - let testCaseSpy: jest.SpyInstance; - let getFileNamesSpy: jest.SpyInstance; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - declarations: [ManualSolutionEntryCreationModalComponent, MockComponent(SolutionEntryComponent)], - }).compileComponents(); - fixture = TestBed.createComponent(ManualSolutionEntryCreationModalComponent); - comp = fixture.componentInstance; - - exerciseService = TestBed.inject(ProgrammingExerciseService); - solutionEntryService = TestBed.inject(ProgrammingExerciseSolutionEntryService); - modalService = TestBed.inject(NgbActiveModal); - - const testCase1 = new ProgrammingExerciseTestCase(); - testCase1.id = 2; - const testCase2 = new ProgrammingExerciseTestCase(); - testCase2.id = 3; - testCases = [testCase1, testCase2]; - testCaseSpy = jest.spyOn(exerciseService, 'getAllTestCases').mockReturnValue(of(testCases)); - - getFileNamesSpy = jest.spyOn(exerciseService, 'getSolutionFileNames').mockReturnValue(of(solutionFiles)); - - comp.exerciseId = 1; - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should load test cases and files on init', () => { - comp.ngOnInit(); - - expect(testCaseSpy).toHaveBeenCalledOnce(); - expect(testCaseSpy).toHaveBeenCalledWith(1); - expect(comp.testCases).toEqual(testCases); - expect(getFileNamesSpy).toHaveBeenCalledOnce(); - expect(getFileNamesSpy).toHaveBeenCalledWith(1); - expect(comp.solutionRepositoryFileNames).toEqual(solutionFiles); - }); - - it('should create manual entry', () => { - const entryToCreate = new ProgrammingExerciseSolutionEntry(); - entryToCreate.testCase = testCases[0]; - comp.solutionEntry = entryToCreate; - - const createdEntry = new ProgrammingExerciseSolutionEntry(); - createdEntry.id = 4; - const createSpy = jest.spyOn(solutionEntryService, 'createSolutionEntry').mockReturnValue(of(createdEntry)); - const createdEmitterSpy = jest.spyOn(comp.onEntryCreated, 'emit'); - const closedSpy = jest.spyOn(modalService, 'close'); - - comp.onCreateEntry(); - - expect(createSpy).toHaveBeenCalledOnce(); - expect(createSpy).toHaveBeenCalledWith(1, 2, entryToCreate); - expect(createdEmitterSpy).toHaveBeenCalledOnce(); - expect(createdEmitterSpy).toHaveBeenCalledWith(createdEntry); - expect(closedSpy).toHaveBeenCalledOnce(); - }); - - it('should close modal', () => { - const closedSpy = jest.spyOn(modalService, 'close'); - comp.clear(); - expect(closedSpy).toHaveBeenCalledOnce(); - }); - - it('should setup editor when file path changes', () => { - comp.solutionEntryComponent = { setupEditor: jest.fn() } as unknown as SolutionEntryComponent; - comp.onUpdateFilePath(); - expect(comp.solutionEntryComponent.setupEditor).toHaveBeenCalledOnce(); - }); -}); diff --git a/src/test/javascript/spec/component/hestia/generation-overview/solution-entry-details-modal.component.spec.ts b/src/test/javascript/spec/component/hestia/generation-overview/solution-entry-details-modal.component.spec.ts deleted file mode 100644 index 7852a6aa8f5f..000000000000 --- a/src/test/javascript/spec/component/hestia/generation-overview/solution-entry-details-modal.component.spec.ts +++ /dev/null @@ -1,62 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { ProgrammingExerciseSolutionEntryService } from 'app/exercises/shared/exercise-hint/services/programming-exercise-solution-entry.service'; -import { ProgrammingExerciseTestCase } from 'app/entities/programming/programming-exercise-test-case.model'; -import { of } from 'rxjs'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; -import { SolutionEntryDetailsModalComponent } from 'app/exercises/programming/hestia/generation-overview/solution-entry-details-modal/solution-entry-details-modal.component'; - -describe('SolutionEntryDetailsModal Component', () => { - let comp: SolutionEntryDetailsModalComponent; - let fixture: ComponentFixture; - - let solutionEntryService: ProgrammingExerciseSolutionEntryService; - let modalService: NgbActiveModal; - - let solutionEntry: ProgrammingExerciseSolutionEntry; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - }).compileComponents(); - fixture = TestBed.createComponent(SolutionEntryDetailsModalComponent); - comp = fixture.componentInstance; - - solutionEntryService = TestBed.inject(ProgrammingExerciseSolutionEntryService); - modalService = TestBed.inject(NgbActiveModal); - - comp.exerciseId = 1; - - const testCase = new ProgrammingExerciseTestCase(); - testCase.id = 2; - - solutionEntry = new ProgrammingExerciseSolutionEntry(); - solutionEntry.id = 3; - solutionEntry.testCase = testCase; - - comp.solutionEntry = solutionEntry; - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should create manual entry', () => { - const updateSpy = jest.spyOn(solutionEntryService, 'updateSolutionEntry').mockReturnValue(of(solutionEntry)); - const closedSpy = jest.spyOn(modalService, 'close'); - - comp.saveSolutionEntry(); - - expect(updateSpy).toHaveBeenCalledOnce(); - expect(updateSpy).toHaveBeenCalledWith(1, 2, 3, solutionEntry); - expect(closedSpy).toHaveBeenCalledOnce(); - expect(comp.solutionEntry).toEqual(solutionEntry); - }); - - it('should close modal', () => { - const closedSpy = jest.spyOn(modalService, 'close'); - comp.clear(); - expect(closedSpy).toHaveBeenCalledOnce(); - }); -}); diff --git a/src/test/javascript/spec/component/hestia/generation-overview/solution-entry-generation-step.component.spec.ts b/src/test/javascript/spec/component/hestia/generation-overview/solution-entry-generation-step.component.spec.ts deleted file mode 100644 index 9e4ff7118bc9..000000000000 --- a/src/test/javascript/spec/component/hestia/generation-overview/solution-entry-generation-step.component.spec.ts +++ /dev/null @@ -1,210 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { Observable, of } from 'rxjs'; -import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { SolutionEntryGenerationStepComponent } from 'app/exercises/programming/hestia/generation-overview/steps/solution-entry-generation-step/solution-entry-generation-step.component'; -import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap'; -import { AlertService } from 'app/core/util/alert.service'; -import { ProgrammingExerciseSolutionEntryService } from 'app/exercises/shared/exercise-hint/services/programming-exercise-solution-entry.service'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; -import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; -import { MockComponent, MockPipe, MockProvider } from 'ng-mocks'; -import { EventEmitter } from '@angular/core'; -import { ProgrammingExerciseTestCase, ProgrammingExerciseTestCaseType } from 'app/entities/programming/programming-exercise-test-case.model'; -import { SortingOrder } from 'app/shared/table/pageable-table'; -import { HelpIconComponent } from 'app/shared/components/help-icon.component'; - -describe('SolutionEntryGenerationStep Component', () => { - let comp: SolutionEntryGenerationStepComponent; - let fixture: ComponentFixture; - - let exerciseService: ProgrammingExerciseService; - let modalService: NgbModal; - let alertService: AlertService; - let solutionEntryService: ProgrammingExerciseSolutionEntryService; - - let onEntryUpdatedSpy: jest.SpyInstance; - - let exercise: ProgrammingExercise; - let entry1: ProgrammingExerciseSolutionEntry; - let entry2: ProgrammingExerciseSolutionEntry; - let solutionEntries: ProgrammingExerciseSolutionEntry[]; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - declarations: [SolutionEntryGenerationStepComponent, MockPipe(ArtemisTranslatePipe), MockComponent(HelpIconComponent)], - providers: [MockProvider(NgbModal), MockProvider(AlertService), MockProvider(ArtemisTranslatePipe)], - }).compileComponents(); - fixture = TestBed.createComponent(SolutionEntryGenerationStepComponent); - comp = fixture.componentInstance; - - exerciseService = TestBed.inject(ProgrammingExerciseService); - modalService = TestBed.inject(NgbModal); - alertService = TestBed.inject(AlertService); - solutionEntryService = TestBed.inject(ProgrammingExerciseSolutionEntryService); - - onEntryUpdatedSpy = jest.spyOn(comp.onEntryUpdate, 'emit'); - - exercise = new ProgrammingExercise(undefined, undefined); - exercise.id = 1; - comp.exercise = exercise; - comp.solutionEntries = []; - - entry1 = new ProgrammingExerciseSolutionEntry(); - entry1.id = 2; - const testCase1 = new ProgrammingExerciseTestCase(); - testCase1.id = 4; - testCase1.type = ProgrammingExerciseTestCaseType.STRUCTURAL; - testCase1.testName = 'test1'; - entry1.testCase = testCase1; - - entry2 = new ProgrammingExerciseSolutionEntry(); - entry2.id = 3; - const testCase2 = new ProgrammingExerciseTestCase(); - testCase2.id = 5; - testCase2.type = ProgrammingExerciseTestCaseType.BEHAVIORAL; - testCase2.testName = 'test2'; - entry2.testCase = testCase2; - - solutionEntries = [entry1, entry2]; - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should load all solution entries on init', () => { - const loadEntriesSpy = jest.spyOn(solutionEntryService, 'getSolutionEntriesForExercise').mockReturnValue(of(solutionEntries)); - - comp.ngOnInit(); - - expect(loadEntriesSpy).toHaveBeenCalledOnce(); - expect(loadEntriesSpy).toHaveBeenCalledWith(1); - expect(comp.solutionEntries).toEqual(solutionEntries); - expect(comp.isLoading).toBeFalse(); - expect(onEntryUpdatedSpy).toHaveBeenCalledOnce(); - expect(onEntryUpdatedSpy).toHaveBeenCalledWith(solutionEntries); - }); - - it('should open editable solution entry details modal', () => { - const mockModal = { - result: Promise.resolve(), - componentInstance: {}, - } as NgbModalRef; - const openSpy = jest.spyOn(modalService, 'open').mockReturnValue(mockModal); - - comp.openSolutionEntryModal(entry1, true); - - expect(openSpy).toHaveBeenCalledOnce(); - expect(mockModal.componentInstance.exerciseId).toBe(1); - expect(mockModal.componentInstance.solutionEntry).toEqual(entry1); - expect(mockModal.componentInstance.isEditable).toBeTrue(); - }); - - it('should open manual solution entry creation modal', () => { - const mockModal = { - result: Promise.resolve(), - componentInstance: {}, - } as NgbModalRef; - const emitter = new EventEmitter(); - mockModal.componentInstance.onEntryCreated = emitter; - - const openSpy = jest.spyOn(modalService, 'open').mockReturnValue(mockModal); - - comp.openManualEntryCreationModal(); - - expect(openSpy).toHaveBeenCalledOnce(); - expect(mockModal.componentInstance.exerciseId).toBe(1); - - emitter.emit(entry1); - expect(comp.solutionEntries).toHaveLength(1); - expect(comp.solutionEntries).toContain(entry1); - }); - - it('should generate structural solution entries', () => { - const generateSpy = jest.spyOn(exerciseService, 'createStructuralSolutionEntries').mockReturnValue(of([entry1])); - const successSpy = jest.spyOn(alertService, 'success'); - // only one behavioral entry - comp.solutionEntries = [entry2]; - - comp.onGenerateStructuralSolutionEntries(); - - expect(generateSpy).toHaveBeenCalledOnce(); - expect(generateSpy).toHaveBeenCalledWith(1); - expect(successSpy).toHaveBeenCalledOnce(); - expect(comp.solutionEntries).toContainAllValues([entry1, entry2]); - expect(comp.testCaseSortOrder).toBeUndefined(); - expect(onEntryUpdatedSpy).toHaveBeenCalledOnce(); - expect(onEntryUpdatedSpy).toHaveBeenCalledWith([entry2, entry1]); - }); - - it('should generate behavioral solution entries', () => { - const generateSpy = jest.spyOn(exerciseService, 'createBehavioralSolutionEntries').mockReturnValue(of([entry2])); - const successSpy = jest.spyOn(alertService, 'success'); - // only one structural entry - comp.solutionEntries = [entry1]; - - comp.onGenerateBehavioralSolutionEntries(); - - expect(generateSpy).toHaveBeenCalledOnce(); - expect(generateSpy).toHaveBeenCalledWith(1); - expect(successSpy).toHaveBeenCalledOnce(); - expect(comp.solutionEntries).toContainAllValues([entry1, entry2]); - expect(comp.testCaseSortOrder).toBeUndefined(); - expect(onEntryUpdatedSpy).toHaveBeenCalledOnce(); - expect(onEntryUpdatedSpy).toHaveBeenCalledWith(solutionEntries); - }); - - it('should delete all solution entries', () => { - const deleteAllSpy = jest.spyOn(solutionEntryService, 'deleteAllSolutionEntriesForExercise').mockReturnValue(new Observable((s) => s.next())); - comp.deleteAllSolutionEntries(); - - expect(deleteAllSpy).toHaveBeenCalledOnce(); - expect(deleteAllSpy).toHaveBeenCalledWith(1); - expect(comp.solutionEntries).toHaveLength(0); - expect(onEntryUpdatedSpy).toHaveBeenCalledOnce(); - expect(onEntryUpdatedSpy).toHaveBeenCalledWith([]); - }); - - it('should delete individual solution entry', () => { - const deleteIndividualSpy = jest.spyOn(solutionEntryService, 'deleteSolutionEntry').mockReturnValue(new Observable((s) => s.next())); - comp.solutionEntries = solutionEntries; - - comp.deleteSolutionEntry(entry1); - - expect(deleteIndividualSpy).toHaveBeenCalledOnce(); - expect(deleteIndividualSpy).toHaveBeenCalledWith(1, 4, 2); - expect(onEntryUpdatedSpy).toHaveBeenCalledOnce(); - expect(onEntryUpdatedSpy).toHaveBeenCalledWith([entry2]); - expect(comp.solutionEntries).toEqual([entry2]); - }); - - it('should sort tests ascending from no order', () => { - comp.solutionEntries = [entry2, entry1]; - comp.changeTestCaseSortOrder(); - expect(comp.solutionEntries).toEqual([entry1, entry2]); - expect(comp.testCaseSortOrder).toBe(SortingOrder.ASCENDING); - }); - - it('should sort tests descending from ascending', () => { - comp.testCaseSortOrder = SortingOrder.ASCENDING; - comp.solutionEntries = [entry1, entry2]; - - comp.changeTestCaseSortOrder(); - - expect(comp.solutionEntries).toEqual([entry2, entry1]); - expect(comp.testCaseSortOrder).toBe(SortingOrder.DESCENDING); - }); - - it('should sort tests ascending from descending', () => { - comp.testCaseSortOrder = SortingOrder.DESCENDING; - comp.solutionEntries = [entry2, entry1]; - - comp.changeTestCaseSortOrder(); - - expect(comp.solutionEntries).toEqual([entry1, entry2]); - expect(comp.testCaseSortOrder).toBe(SortingOrder.ASCENDING); - }); -}); diff --git a/src/test/javascript/spec/component/hestia/testwise-coverage-report/testwise-coverage-file.component.spec.ts b/src/test/javascript/spec/component/hestia/testwise-coverage-report/testwise-coverage-file.component.spec.ts deleted file mode 100644 index a35ac8f46b3b..000000000000 --- a/src/test/javascript/spec/component/hestia/testwise-coverage-report/testwise-coverage-file.component.spec.ts +++ /dev/null @@ -1,116 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { NoopAnimationsModule } from '@angular/platform-browser/animations'; -import { TestwiseCoverageFileComponent } from 'app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-file.component'; -import { ProgrammingExerciseTestCase } from 'app/entities/programming/programming-exercise-test-case.model'; -import { TestwiseCoverageReportEntry } from 'app/entities/hestia/testwise-coverage-report-entry.model'; -import { CoverageFileReport } from 'app/entities/hestia/coverage-file-report.model'; -import { MatExpansionModule } from '@angular/material/expansion'; -import { MockComponent } from 'ng-mocks'; -import { MonacoEditorComponent } from 'app/shared/monaco-editor/monaco-editor.component'; - -describe('TestwiseCoverageFile Component', () => { - let comp: TestwiseCoverageFileComponent; - let fixture: ComponentFixture; - let reportEntry1: TestwiseCoverageReportEntry; - let reportEntry2: TestwiseCoverageReportEntry; - let fileReport: CoverageFileReport; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule, MatExpansionModule, NoopAnimationsModule], - declarations: [TestwiseCoverageFileComponent, MockComponent(MonacoEditorComponent)], - }) - .compileComponents() - .then(() => { - fixture = TestBed.createComponent(TestwiseCoverageFileComponent); - comp = fixture.componentInstance; - - const testCase1 = { - id: 1, - testName: 'testBubbleSort()', - } as ProgrammingExerciseTestCase; - const testCase2 = { - id: 2, - testName: 'testBubbleSortForSmallLists()', - } as ProgrammingExerciseTestCase; - - reportEntry1 = { - id: 1, - startLine: 4, - lineCount: 2, - testCase: testCase1, - } as TestwiseCoverageReportEntry; - reportEntry2 = { - id: 2, - startLine: 7, - lineCount: 2, - testCase: testCase2, - } as TestwiseCoverageReportEntry; - - fileReport = { - id: 1, - lineCount: 10, - filePath: 'src/de/tum/in/ase/BubbleSort.java', - coveredLineCount: 5, - testwiseCoverageEntries: [reportEntry1, reportEntry2], - } as CoverageFileReport; - comp.fileContent = '\n\n\n\n\n\n\n\n\nlast'; - comp.fileName = 'src/de/tum/in/ase/BubbleSort.java'; - comp.fileReport = fileReport; - }); - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should initially create and add all ranges correctly', () => { - const highlightLinesSpy = jest.spyOn(comp.editor, 'highlightLines'); - - comp.ngOnInit(); - - expect(highlightLinesSpy).toHaveBeenCalledTimes(2); - expect(highlightLinesSpy).toHaveBeenCalledWith( - 4, - 5, - TestwiseCoverageFileComponent.COVERED_LINE_HIGHLIGHT_CLASS, - TestwiseCoverageFileComponent.COVERED_LINE_HIGHLIGHT_CLASS, - ); - expect(highlightLinesSpy).toHaveBeenCalledWith( - 7, - 8, - TestwiseCoverageFileComponent.COVERED_LINE_HIGHLIGHT_CLASS, - TestwiseCoverageFileComponent.COVERED_LINE_HIGHLIGHT_CLASS, - ); - }); - - it('should calculate covered line ratio correctly', () => { - comp.fileReport.testwiseCoverageEntries = [reportEntry1]; - comp.ngOnInit(); - expect(comp.proportionCoveredLines).toBe(0.2); - expect(comp.proportionString).toBe('20.0 %'); - }); - - it('should update on input coverage data changes', () => { - comp.fileReport.testwiseCoverageEntries = [reportEntry1, reportEntry2]; - comp.ngOnInit(); - expect(comp.proportionCoveredLines).toBe(0.4); - expect(comp.proportionString).toBe('40.0 %'); - - const highlightLinesSpy = jest.spyOn(comp.editor, 'highlightLines'); - - fileReport.testwiseCoverageEntries = [reportEntry1]; - comp.fileReport = fileReport; - fixture.detectChanges(); - - expect(comp.proportionCoveredLines).toBe(0.2); - expect(comp.proportionString).toBe('20.0 %'); - expect(highlightLinesSpy).toHaveBeenCalledExactlyOnceWith( - 4, - 5, - TestwiseCoverageFileComponent.COVERED_LINE_HIGHLIGHT_CLASS, - TestwiseCoverageFileComponent.COVERED_LINE_HIGHLIGHT_CLASS, - ); - }); -}); diff --git a/src/test/javascript/spec/component/hestia/testwise-coverage-report/testwise-coverage-report.component.spec.ts b/src/test/javascript/spec/component/hestia/testwise-coverage-report/testwise-coverage-report.component.spec.ts deleted file mode 100644 index 1c2e97853b5b..000000000000 --- a/src/test/javascript/spec/component/hestia/testwise-coverage-report/testwise-coverage-report.component.spec.ts +++ /dev/null @@ -1,133 +0,0 @@ -import { ArtemisTestModule } from '../../../test.module'; -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { TestwiseCoverageReportComponent } from 'app/exercises/programming/hestia/testwise-coverage-report/testwise-coverage-report.component'; -import { CoverageReport } from 'app/entities/hestia/coverage-report.model'; -import { CoverageFileReport } from 'app/entities/hestia/coverage-file-report.model'; -import { ProgrammingExerciseTestCase } from 'app/entities/programming/programming-exercise-test-case.model'; -import { TestwiseCoverageReportEntry } from 'app/entities/hestia/testwise-coverage-report-entry.model'; - -describe('TestwiseCoverageReport Component', () => { - let comp: TestwiseCoverageReportComponent; - let fixture: ComponentFixture; - - let report: CoverageReport; - let reportEntries: TestwiseCoverageReportEntry[]; - let fileReports: CoverageFileReport[]; - let fileContentByName: Map; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule], - declarations: [TestwiseCoverageReportComponent], - }) - .compileComponents() - .then(() => { - fixture = TestBed.createComponent(TestwiseCoverageReportComponent); - comp = fixture.componentInstance; - - const testCase1 = { - id: 1, - testName: 'testBubbleSort()', - } as ProgrammingExerciseTestCase; - const testCase2 = { - id: 2, - testName: 'testBubbleSortForSmallLists()', - } as ProgrammingExerciseTestCase; - const testCase3 = { - id: 3, - testName: 'testMergeSort()', - } as ProgrammingExerciseTestCase; - - const reportEntry1 = { - id: 1, - startLine: 4, - lineCount: 2, - testCase: testCase1, - } as TestwiseCoverageReportEntry; - const reportEntry2 = { - id: 2, - startLine: 7, - lineCount: 2, - testCase: testCase2, - } as TestwiseCoverageReportEntry; - const reportEntry3 = { - id: 3, - startLine: 1, - lineCount: 2, - testCase: testCase3, - } as TestwiseCoverageReportEntry; - reportEntries = [reportEntry1, reportEntry2, reportEntry3]; - - fileReports = [ - { - filePath: 'src/de/tum/in/ase/BubbleSort.java', - lineCount: 10, - coveredLineCount: 5, - testwiseCoverageEntries: [reportEntry1, reportEntry2], - } as CoverageFileReport, - { - filePath: 'src/de/tum/in/ase/MergeSort.java', - lineCount: 2, - coveredLineCount: 1, - testwiseCoverageEntries: [reportEntry3], - } as CoverageFileReport, - ]; - - report = { - id: 1, - fileReports, - coveredLineRatio: 0.5, - } as CoverageReport; - comp.report = report; - - fileContentByName = new Map(); - // file with 10 lines, 4 covered - fileContentByName.set( - 'src/de/tum/in/ase/BubbleSort.java', - 'package de.tum.in.ase;\n\ncovered\ncovered\nuncovered\ncovered\ncovered\nuncovered\nuncovered\nuncovered', - ); - // file with 2 lines, 2 covered - fileContentByName.set('src/de/tum/in/ase/MergeSort.java', 'covered\ncovered'); - comp.fileContentByPath = fileContentByName; - - comp.ngOnInit(); - }); - }); - - it('should initially display coverage for all test cases', () => { - const expectedMap = new Map(); - expectedMap.set('testBubbleSort()', true); - expectedMap.set('testBubbleSortForSmallLists()', true); - expectedMap.set('testMergeSort()', true); - - expect(comp.displayedTestCaseNames.size).toBe(3); - expect(comp.displayedTestCaseNames).toEqual(expectedMap); - }); - - it('should set file report by file name with for all test cases', () => { - const expectedMap = new Map(); - expectedMap.set('src/de/tum/in/ase/BubbleSort.java', fileReports[0]); - expectedMap.set('src/de/tum/in/ase/MergeSort.java', fileReports[1]); - - expect(comp.fileReportByFileName).toEqual(expectedMap); - }); - - it('should filter coverage file reports for test case', () => { - comp.changeReportsBySelectedTestCases('testBubbleSort()'); - expect(comp.displayedTestCaseNames.get('testBubbleSort()')).toBeFalse(); - expect(comp.fileReportByFileName.size).toBe(2); - expect(comp.fileReportByFileName.get('src/de/tum/in/ase/BubbleSort.java')?.testwiseCoverageEntries).toEqual([reportEntries[1]]); - }); - - it('should create empty file report if no report exists for file', () => { - comp.fileContentByPath.set('notexisting.java', '\n\n'); - comp.ngOnInit(); - - expect(comp.fileReportByFileName.get('notexisting.java')).toEqual({ - lineCount: 3, - coveredLineCount: 0, - filePath: 'notexisting.java', - testwiseCoverageEntries: [], - } as CoverageFileReport); - }); -}); diff --git a/src/test/javascript/spec/component/localci/build-agents/build-agent-details.component.spec.ts b/src/test/javascript/spec/component/localci/build-agents/build-agent-details.component.spec.ts index 606bc59a0ed8..7224aaf85123 100644 --- a/src/test/javascript/spec/component/localci/build-agents/build-agent-details.component.spec.ts +++ b/src/test/javascript/spec/component/localci/build-agents/build-agent-details.component.spec.ts @@ -59,7 +59,6 @@ describe('BuildAgentDetailsComponent', () => { projectType: 'Maven', scaEnabled: false, sequentialTestRunsEnabled: false, - testwiseCoverageEnabled: false, resultPaths: [], }; diff --git a/src/test/javascript/spec/component/localci/build-agents/build-agent-summary.component.spec.ts b/src/test/javascript/spec/component/localci/build-agents/build-agent-summary.component.spec.ts index 0d798185a915..623f92e6482f 100644 --- a/src/test/javascript/spec/component/localci/build-agents/build-agent-summary.component.spec.ts +++ b/src/test/javascript/spec/component/localci/build-agents/build-agent-summary.component.spec.ts @@ -56,7 +56,6 @@ describe('BuildAgentSummaryComponent', () => { projectType: 'Maven', scaEnabled: false, sequentialTestRunsEnabled: false, - testwiseCoverageEnabled: false, resultPaths: [], }; diff --git a/src/test/javascript/spec/component/localci/build-agents/build-agents.service.spec.ts b/src/test/javascript/spec/component/localci/build-agents/build-agents.service.spec.ts index 18b9f0135b83..69fb9bca2acd 100644 --- a/src/test/javascript/spec/component/localci/build-agents/build-agents.service.spec.ts +++ b/src/test/javascript/spec/component/localci/build-agents/build-agents.service.spec.ts @@ -42,7 +42,6 @@ describe('BuildAgentsService', () => { projectType: 'Maven', scaEnabled: false, sequentialTestRunsEnabled: false, - testwiseCoverageEnabled: false, resultPaths: [], }; diff --git a/src/test/javascript/spec/component/localci/build-queue/build-queue.component.spec.ts b/src/test/javascript/spec/component/localci/build-queue/build-queue.component.spec.ts index 4c53661e99d4..b22a67ee3f15 100644 --- a/src/test/javascript/spec/component/localci/build-queue/build-queue.component.spec.ts +++ b/src/test/javascript/spec/component/localci/build-queue/build-queue.component.spec.ts @@ -89,7 +89,6 @@ describe('BuildQueueComponent', () => { projectType: 'Maven', scaEnabled: false, sequentialTestRunsEnabled: false, - testwiseCoverageEnabled: false, resultPaths: [], }, }, @@ -125,7 +124,6 @@ describe('BuildQueueComponent', () => { projectType: 'Maven', scaEnabled: false, sequentialTestRunsEnabled: false, - testwiseCoverageEnabled: false, resultPaths: [], }, }, @@ -163,7 +161,6 @@ describe('BuildQueueComponent', () => { projectType: 'Maven', scaEnabled: false, sequentialTestRunsEnabled: false, - testwiseCoverageEnabled: false, resultPaths: [], }, }, @@ -199,7 +196,6 @@ describe('BuildQueueComponent', () => { projectType: 'Maven', scaEnabled: false, sequentialTestRunsEnabled: false, - testwiseCoverageEnabled: false, resultPaths: [], }, }, diff --git a/src/test/javascript/spec/component/localci/build-queue/build-queue.service.spec.ts b/src/test/javascript/spec/component/localci/build-queue/build-queue.service.spec.ts index 73b4ac460f63..16a2d81d37ed 100644 --- a/src/test/javascript/spec/component/localci/build-queue/build-queue.service.spec.ts +++ b/src/test/javascript/spec/component/localci/build-queue/build-queue.service.spec.ts @@ -80,7 +80,6 @@ describe('BuildQueueService', () => { buildConfig.projectType = 'type1'; buildConfig.scaEnabled = false; buildConfig.sequentialTestRunsEnabled = false; - buildConfig.testwiseCoverageEnabled = false; buildConfig.resultPaths = ['path1']; elem1.id = '1'; diff --git a/src/test/javascript/spec/component/localvc/commit-details-view.component.spec.ts b/src/test/javascript/spec/component/localvc/commit-details-view.component.spec.ts index fb964f7ea127..7625253cc587 100644 --- a/src/test/javascript/spec/component/localvc/commit-details-view.component.spec.ts +++ b/src/test/javascript/spec/component/localvc/commit-details-view.component.spec.ts @@ -12,10 +12,10 @@ import { MockComponent, MockPipe } from 'ng-mocks'; import { CommitDetailsViewComponent } from 'app/localvc/commit-details-view/commit-details-view.component'; import { ProgrammingExerciseService } from 'app/exercises/programming/manage/services/programming-exercise.service'; import { MockProgrammingExerciseService } from '../../helpers/mocks/service/mock-programming-exercise.service'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from '../../../../../main/webapp/app/entities/programming-exercise-git-diff-report.model'; import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; import { ArtemisDatePipe } from 'app/shared/pipes/artemis-date.pipe'; -import { GitDiffReportComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report.component'; +import { GitDiffReportComponent } from '../../../../../main/webapp/app/exercises/programming/git-diff-report/git-diff-report.component'; import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; import { HttpResponse } from '@angular/common/http'; diff --git a/src/test/javascript/spec/component/overview/exercise-details/course-exercise-details.component.spec.ts b/src/test/javascript/spec/component/overview/exercise-details/course-exercise-details.component.spec.ts index 4f2a2c9b8331..cfe00096d639 100644 --- a/src/test/javascript/spec/component/overview/exercise-details/course-exercise-details.component.spec.ts +++ b/src/test/javascript/spec/component/overview/exercise-details/course-exercise-details.component.spec.ts @@ -59,7 +59,6 @@ import { LockRepositoryPolicy } from 'app/entities/submission-policy.model'; import { PlagiarismCasesService } from 'app/course/plagiarism-cases/shared/plagiarism-cases.service'; import { PlagiarismVerdict } from 'app/exercises/shared/plagiarism/types/PlagiarismVerdict'; import { AlertService } from 'app/core/util/alert.service'; -import { ExerciseHintButtonOverlayComponent } from 'app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component'; import { ProgrammingExerciseExampleSolutionRepoDownloadComponent } from 'app/exercises/programming/shared/actions/programming-exercise-example-solution-repo-download.component'; import { ResetRepoButtonComponent } from 'app/shared/components/reset-repo-button/reset-repo-button.component'; import { ProblemStatementComponent } from 'app/overview/exercise-details/problem-statement/problem-statement.component'; @@ -71,7 +70,6 @@ import { ScienceService } from 'app/shared/science/science.service'; import { MockScienceService } from '../../../helpers/mocks/service/mock-science-service'; import { ScienceEventType } from 'app/shared/science/science.model'; import { PROFILE_IRIS } from 'app/app.constants'; -import { ExerciseHintService } from 'app/exercises/shared/exercise-hint/shared/exercise-hint.service'; import { CourseInformationSharingConfiguration } from 'app/entities/course.model'; import { provideHttpClient } from '@angular/common/http'; @@ -80,15 +78,12 @@ describe('CourseExerciseDetailsComponent', () => { let fixture: ComponentFixture; let profileService: ProfileService; let exerciseService: ExerciseService; - let exerciseHintService: ExerciseHintService; let teamService: TeamService; let participationService: ParticipationService; let participationWebsocketService: ParticipationWebsocketService; let complaintService: ComplaintService; let getProfileInfoMock: jest.SpyInstance; let getExerciseDetailsMock: jest.SpyInstance; - let getActivatedExerciseHintsMock: jest.SpyInstance; - let getAvailableExerciseHintsMock: jest.SpyInstance; let mergeStudentParticipationMock: jest.SpyInstance; let subscribeForParticipationChangesMock: jest.SpyInstance; let participationWebsockerBehaviourSubject: BehaviorSubject; @@ -152,7 +147,6 @@ describe('CourseExerciseDetailsComponent', () => { MockComponent(ResultHistoryComponent), MockComponent(ResultComponent), MockComponent(ComplaintsStudentViewComponent), - MockComponent(ExerciseHintButtonOverlayComponent), MockComponent(ProgrammingExerciseExampleSolutionRepoDownloadComponent), MockComponent(ProblemStatementComponent), MockComponent(ResetRepoButtonComponent), @@ -189,7 +183,6 @@ describe('CourseExerciseDetailsComponent', () => { MockProvider(PlagiarismCasesService), MockProvider(AlertService), MockProvider(IrisSettingsService), - MockProvider(ExerciseHintService), ], }) .compileComponents() @@ -231,10 +224,6 @@ describe('CourseExerciseDetailsComponent', () => { scienceService = TestBed.inject(ScienceService); logEventStub = jest.spyOn(scienceService, 'logEvent'); - exerciseHintService = TestBed.inject(ExerciseHintService); - getActivatedExerciseHintsMock = jest.spyOn(exerciseHintService, 'getActivatedExerciseHints'); - getAvailableExerciseHintsMock = jest.spyOn(exerciseHintService, 'getAvailableExerciseHints'); - participationService = TestBed.inject(ParticipationService); mergeStudentParticipationMock = jest.spyOn(participationService, 'mergeStudentParticipations'); }); @@ -335,16 +324,6 @@ describe('CourseExerciseDetailsComponent', () => { expect(comp.exampleSolutionCollapsed).toBeFalse(); }); - it('should activate hint', () => { - comp.availableExerciseHints = [{ id: 1 }, { id: 2 }]; - comp.activatedExerciseHints = []; - - const activatedHint = comp.availableExerciseHints[0]; - comp.onHintActivated(activatedHint); - expect(comp.availableExerciseHints).not.toContain(activatedHint); - expect(comp.activatedExerciseHints).toContain(activatedHint); - }); - it('should sort results by completion date in ascending order', () => { const result1 = { completionDate: dayjs().subtract(2, 'days') } as Result; const result2 = { completionDate: dayjs().subtract(1, 'day') } as Result; @@ -432,16 +411,9 @@ describe('CourseExerciseDetailsComponent', () => { const newParticipation = { ...participation, submissions: [submission, { id: submissionId + 1 }] }; - getActivatedExerciseHintsMock.mockReturnValue(of({ body: [] })); - getAvailableExerciseHintsMock.mockReturnValue(of({ body: [] })); mergeStudentParticipationMock.mockReturnValue([newParticipation]); participationWebsockerBehaviourSubject.next({ ...newParticipation, exercise: programmingExercise, results: [] }); - - tick(); - - expect(getActivatedExerciseHintsMock).toHaveBeenCalledOnce(); - expect(getAvailableExerciseHintsMock).toHaveBeenCalledOnce(); })); it.each<[string[]]>([[[]], [[PROFILE_IRIS]]])( diff --git a/src/test/javascript/spec/component/programming-diff-report-detail.component.spec.ts b/src/test/javascript/spec/component/programming-diff-report-detail.component.spec.ts index 9f1940740640..9c0ede8c5141 100644 --- a/src/test/javascript/spec/component/programming-diff-report-detail.component.spec.ts +++ b/src/test/javascript/spec/component/programming-diff-report-detail.component.spec.ts @@ -1,7 +1,7 @@ import { ComponentFixture, TestBed } from '@angular/core/testing'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; import { MockNgbModalService } from '../helpers/mocks/service/mock-ngb-modal.service'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from '../../../../main/webapp/app/entities/programming-exercise-git-diff-report.model'; import { ProgrammingDiffReportDetailComponent } from 'app/detail-overview-list/components/programming-diff-report-detail/programming-diff-report-detail.component'; import { TranslateService } from '@ngx-translate/core'; import { MockTranslateService } from '../helpers/mocks/service/mock-translate.service'; diff --git a/src/test/javascript/spec/component/programming-exercise/programming-exercise-custom-aeolus-build-plan.component.spec.ts b/src/test/javascript/spec/component/programming-exercise/programming-exercise-custom-aeolus-build-plan.component.spec.ts index 005fa8fdd41a..c1882b341c37 100644 --- a/src/test/javascript/spec/component/programming-exercise/programming-exercise-custom-aeolus-build-plan.component.spec.ts +++ b/src/test/javascript/spec/component/programming-exercise/programming-exercise-custom-aeolus-build-plan.component.spec.ts @@ -179,7 +179,6 @@ describe('ProgrammingExercise Aeolus Custom Build Plan', () => { comp.projectType = programmingExercise.projectType; comp.sequentialTestRuns = programmingExercise.buildConfig?.sequentialTestRuns; comp.staticCodeAnalysisEnabled = programmingExercise.staticCodeAnalysisEnabled; - comp.testwiseCoverageEnabled = programmingExercise.buildConfig?.testwiseCoverageEnabled; expect(comp.shouldReloadTemplate()).toBeFalse(); }); @@ -188,7 +187,6 @@ describe('ProgrammingExercise Aeolus Custom Build Plan', () => { comp.projectType = ProjectType.PLAIN_GRADLE; comp.sequentialTestRuns = programmingExercise.buildConfig?.sequentialTestRuns; comp.staticCodeAnalysisEnabled = true; - comp.testwiseCoverageEnabled = true; expect(comp.shouldReloadTemplate()).toBeTrue(); }); @@ -214,7 +212,6 @@ describe('ProgrammingExercise Aeolus Custom Build Plan', () => { comp.projectType = undefined; comp.sequentialTestRuns = undefined; comp.staticCodeAnalysisEnabled = undefined; - comp.testwiseCoverageEnabled = undefined; comp.programmingExerciseCreationConfig = programmingExerciseCreationConfigMock; comp.programmingExerciseCreationConfig.customBuildPlansSupported = PROFILE_AEOLUS; comp.loadAeolusTemplate(); @@ -222,7 +219,6 @@ describe('ProgrammingExercise Aeolus Custom Build Plan', () => { expect(comp.projectType).toBe(programmingExercise.projectType); expect(comp.sequentialTestRuns).toBe(programmingExercise.buildConfig?.sequentialTestRuns); expect(comp.staticCodeAnalysisEnabled).toBe(programmingExercise.staticCodeAnalysisEnabled); - expect(comp.testwiseCoverageEnabled).toBe(programmingExercise.buildConfig?.testwiseCoverageEnabled); }); it('should not call loadAeolusTemplate', () => { diff --git a/src/test/javascript/spec/component/programming-exercise/programming-exercise-custom-build-plan.component.spec.ts b/src/test/javascript/spec/component/programming-exercise/programming-exercise-custom-build-plan.component.spec.ts index a02862b32a59..2e9e9b841e50 100644 --- a/src/test/javascript/spec/component/programming-exercise/programming-exercise-custom-build-plan.component.spec.ts +++ b/src/test/javascript/spec/component/programming-exercise/programming-exercise-custom-build-plan.component.spec.ts @@ -98,7 +98,6 @@ describe('ProgrammingExercise Custom Build Plan', () => { comp.projectType = programmingExercise.projectType; comp.sequentialTestRuns = programmingExercise.buildConfig?.sequentialTestRuns; comp.staticCodeAnalysisEnabled = programmingExercise.staticCodeAnalysisEnabled; - comp.testwiseCoverageEnabled = programmingExercise.buildConfig?.testwiseCoverageEnabled; expect(comp.shouldReloadTemplate()).toBeFalse(); }); @@ -107,7 +106,6 @@ describe('ProgrammingExercise Custom Build Plan', () => { comp.projectType = ProjectType.PLAIN_GRADLE; comp.sequentialTestRuns = programmingExercise.buildConfig?.sequentialTestRuns; comp.staticCodeAnalysisEnabled = true; - comp.testwiseCoverageEnabled = true; expect(comp.shouldReloadTemplate()).toBeTrue(); }); @@ -133,7 +131,6 @@ describe('ProgrammingExercise Custom Build Plan', () => { comp.projectType = undefined; comp.sequentialTestRuns = undefined; comp.staticCodeAnalysisEnabled = undefined; - comp.testwiseCoverageEnabled = undefined; comp.programmingExerciseCreationConfig = programmingExerciseCreationConfigMock; comp.programmingExerciseCreationConfig.customBuildPlansSupported = PROFILE_LOCALCI; comp.loadAeolusTemplate(); @@ -141,7 +138,6 @@ describe('ProgrammingExercise Custom Build Plan', () => { expect(comp.projectType).toBe(programmingExercise.projectType); expect(comp.sequentialTestRuns).toBe(programmingExercise.buildConfig?.sequentialTestRuns); expect(comp.staticCodeAnalysisEnabled).toBe(programmingExercise.staticCodeAnalysisEnabled); - expect(comp.testwiseCoverageEnabled).toBe(programmingExercise.buildConfig?.testwiseCoverageEnabled); }); it('should not call loadAeolusTemplate', () => { diff --git a/src/test/javascript/spec/component/programming-exercise/programming-exercise-detail.component.spec.ts b/src/test/javascript/spec/component/programming-exercise/programming-exercise-detail.component.spec.ts index dba7c6b0a9ca..062554dd9c5f 100644 --- a/src/test/javascript/spec/component/programming-exercise/programming-exercise-detail.component.spec.ts +++ b/src/test/javascript/spec/component/programming-exercise/programming-exercise-detail.component.spec.ts @@ -22,7 +22,6 @@ import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; import { MockSyncStorage } from '../../helpers/mocks/service/mock-sync-storage.service'; import { LocalStorageService, SessionStorageService } from 'ngx-webstorage'; import { MockProgrammingExerciseGradingService } from '../../helpers/mocks/service/mock-programming-exercise-grading.service'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; import { TemplateProgrammingExerciseParticipation } from 'app/entities/participation/template-programming-exercise-participation.model'; import { SolutionProgrammingExerciseParticipation } from 'app/entities/participation/solution-programming-exercise-participation.model'; import { HttpErrorResponse, HttpResponse } from '@angular/common/http'; @@ -32,8 +31,7 @@ import { ProgrammingLanguageFeatureService, } from 'app/exercises/programming/shared/service/programming-language-feature/programming-language-feature.service'; import { MockRouter } from '../../helpers/mocks/mock-router'; -import { BuildConfig } from '../../../../../main/webapp/app/entities/programming/build-config.model'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffReport } from '../../../../../main/webapp/app/entities/programming-exercise-git-diff-report.model'; import { BuildLogStatisticsDTO } from 'app/entities/programming/build-log-statistics-dto'; import { SubmissionPolicyService } from '../../../../../main/webapp/app/exercises/programming/manage/services/submission-policy.service'; @@ -64,9 +62,6 @@ describe('ProgrammingExerciseDetailComponent', () => { solutionParticipation: { id: 2, } as SolutionProgrammingExerciseParticipation, - buildConfig: { - testwiseCoverageEnabled: true, - } as BuildConfig, } as ProgrammingExercise; const exerciseStatistics = { @@ -163,14 +158,12 @@ describe('ProgrammingExerciseDetailComponent', () => { it('should reload on participation change', fakeAsync(() => { const loadDiffSpy = jest.spyOn(comp, 'loadGitDiffReport'); jest.spyOn(exerciseService, 'getLatestResult').mockReturnValue({ successful: true }); - jest.spyOn(exerciseService, 'getLatestFullTestwiseCoverageReport').mockReturnValue(of({ coveredLineRatio: 0.5 })); comp.programmingExercise = mockProgrammingExercise; comp.programmingExerciseBuildConfig = mockProgrammingExercise.buildConfig; comp.onParticipationChange(); tick(); expect(loadDiffSpy).toHaveBeenCalledOnce(); expect(gitDiffReportStub).toHaveBeenCalledOnce(); - expect(comp.programmingExercise.coveredLinesRatio).toBe(0.5); })); describe('onInit for course exercise', () => { @@ -275,42 +268,6 @@ describe('ProgrammingExerciseDetailComponent', () => { expect(sections).toBeDefined(); }); - it('should create structural solution entries', () => { - const programmingExercise = new ProgrammingExercise(new Course(), undefined); - programmingExercise.id = 123; - comp.programmingExercise = programmingExercise; - - jest.spyOn(exerciseService, 'createStructuralSolutionEntries').mockReturnValue(of([] as ProgrammingExerciseSolutionEntry[])); - jest.spyOn(alertService, 'addAlert'); - - comp.createStructuralSolutionEntries(); - - expect(exerciseService.createStructuralSolutionEntries).toHaveBeenCalledOnce(); - expect(alertService.addAlert).toHaveBeenCalledOnce(); - expect(alertService.addAlert).toHaveBeenCalledWith({ - type: AlertType.SUCCESS, - message: 'artemisApp.programmingExercise.createStructuralSolutionEntriesSuccess', - }); - }); - - it('should create behavioral solution entries', () => { - const programmingExercise = new ProgrammingExercise(new Course(), undefined); - programmingExercise.id = 123; - comp.programmingExercise = programmingExercise; - - jest.spyOn(exerciseService, 'createBehavioralSolutionEntries').mockReturnValue(of([] as ProgrammingExerciseSolutionEntry[])); - jest.spyOn(alertService, 'addAlert'); - - comp.createBehavioralSolutionEntries(); - - expect(exerciseService.createBehavioralSolutionEntries).toHaveBeenCalledOnce(); - expect(alertService.addAlert).toHaveBeenCalledOnce(); - expect(alertService.addAlert).toHaveBeenCalledWith({ - type: AlertType.SUCCESS, - message: 'artemisApp.programmingExercise.createBehavioralSolutionEntriesSuccess', - }); - }); - it.each([ ['jenkins', true], ['gitlabci', true], diff --git a/src/test/javascript/spec/component/programming-exercise/programming-exercise-update.component.spec.ts b/src/test/javascript/spec/component/programming-exercise/programming-exercise-update.component.spec.ts index c4cbdf912f8a..3de1c9fe3eb5 100644 --- a/src/test/javascript/spec/component/programming-exercise/programming-exercise-update.component.spec.ts +++ b/src/test/javascript/spec/component/programming-exercise/programming-exercise-update.component.spec.ts @@ -197,7 +197,6 @@ describe('ProgrammingExerciseUpdateComponent', () => { packageNameRequired: true, checkoutSolutionRepositoryAllowed: true, projectTypes: [ProjectType.PLAIN_MAVEN, ProjectType.MAVEN_MAVEN], - testwiseCoverageAnalysisSupported: true, auxiliaryRepositoriesSupported: true, } as ProgrammingLanguageFeature); }); @@ -1195,7 +1194,6 @@ describe('ProgrammingExerciseUpdateComponent', () => { tick(); expect(comp.programmingExercise.staticCodeAnalysisEnabled).toBeFalse(); - expect(comp.programmingExercise.buildConfig?.testwiseCoverageEnabled).toBeFalse(); })); it('should disable options for java dejagnu project type and re-enable them after changing back to maven or gradle', fakeAsync(() => { @@ -1203,19 +1201,15 @@ describe('ProgrammingExerciseUpdateComponent', () => { getFeaturesStub.mockImplementation((language: ProgrammingLanguage) => getProgrammingLanguageFeature(language)); comp.selectedProjectType = ProjectType.MAVEN_BLACKBOX; expect(comp.sequentialTestRunsAllowed).toBeFalse(); - expect(comp.testwiseCoverageAnalysisSupported).toBeFalse(); comp.selectedProjectType = ProjectType.MAVEN_MAVEN; expect(comp.sequentialTestRunsAllowed).toBeTrue(); - expect(comp.testwiseCoverageAnalysisSupported).toBeTrue(); comp.selectedProjectType = ProjectType.MAVEN_BLACKBOX; expect(comp.sequentialTestRunsAllowed).toBeFalse(); - expect(comp.testwiseCoverageAnalysisSupported).toBeFalse(); comp.selectedProjectType = ProjectType.GRADLE_GRADLE; expect(comp.sequentialTestRunsAllowed).toBeTrue(); - expect(comp.testwiseCoverageAnalysisSupported).toBeTrue(); })); }); @@ -1392,7 +1386,6 @@ const getProgrammingLanguageFeature = (programmingLanguage: ProgrammingLanguage) packageNameRequired: true, checkoutSolutionRepositoryAllowed: false, projectTypes: [ProjectType.PLAIN, ProjectType.XCODE], - testwiseCoverageAnalysisSupported: false, auxiliaryRepositoriesSupported: true, } as ProgrammingLanguageFeature; case ProgrammingLanguage.JAVA: @@ -1404,7 +1397,6 @@ const getProgrammingLanguageFeature = (programmingLanguage: ProgrammingLanguage) packageNameRequired: true, checkoutSolutionRepositoryAllowed: true, projectTypes: [ProjectType.PLAIN_MAVEN, ProjectType.MAVEN_MAVEN], - testwiseCoverageAnalysisSupported: true, auxiliaryRepositoriesSupported: true, } as ProgrammingLanguageFeature; case ProgrammingLanguage.HASKELL: @@ -1415,7 +1407,6 @@ const getProgrammingLanguageFeature = (programmingLanguage: ProgrammingLanguage) plagiarismCheckSupported: false, packageNameRequired: false, checkoutSolutionRepositoryAllowed: true, - testwiseCoverageAnalysisSupported: false, auxiliaryRepositoriesSupported: true, } as ProgrammingLanguageFeature; case ProgrammingLanguage.C: @@ -1427,7 +1418,6 @@ const getProgrammingLanguageFeature = (programmingLanguage: ProgrammingLanguage) packageNameRequired: false, checkoutSolutionRepositoryAllowed: true, projectTypes: [ProjectType.FACT, ProjectType.GCC], - testwiseCoverageAnalysisSupported: false, auxiliaryRepositoriesSupported: true, } as ProgrammingLanguageFeature; default: diff --git a/src/test/javascript/spec/component/programming-exercise/update-components/programming-exercise-creation-config-mock.ts b/src/test/javascript/spec/component/programming-exercise/update-components/programming-exercise-creation-config-mock.ts index d1f3ef607ed6..36987b972c7d 100644 --- a/src/test/javascript/spec/component/programming-exercise/update-components/programming-exercise-creation-config-mock.ts +++ b/src/test/javascript/spec/component/programming-exercise/update-components/programming-exercise-creation-config-mock.ts @@ -51,7 +51,6 @@ export const programmingExerciseCreationConfigMock: ProgrammingExerciseCreationC staticCodeAnalysisAllowed: false, supportedLanguages: [], templateParticipationResultLoaded: false, - testwiseCoverageAnalysisSupported: false, titleNamePattern: '', updateCategories(_categories: ExerciseCategory[]): void {}, updateCheckoutDirectory(_editedAuxiliaryRepository: AuxiliaryRepository): (newValue: any) => string | undefined { diff --git a/src/test/javascript/spec/component/shared/http/file.service.spec.ts b/src/test/javascript/spec/component/shared/http/file.service.spec.ts index 843fd61a12d3..c8657cd9a70f 100644 --- a/src/test/javascript/spec/component/shared/http/file.service.spec.ts +++ b/src/test/javascript/spec/component/shared/http/file.service.spec.ts @@ -136,51 +136,12 @@ describe('FileService', () => { }); }); - describe('getAeolusTemplateFile', () => { - it('should fetch the aeolus template file with all parameters', () => { - const language = ProgrammingLanguage.PYTHON; - const projectType = ProjectType.PLAIN; - const staticAnalysis = true; - const sequentialRuns = false; - const coverage = true; - const expectedUrl = `api/files/aeolus/templates/PYTHON/PLAIN?staticAnalysis=true&sequentialRuns=false&testCoverage=true`; - const response = 'aeolus template content'; - - fileService.getAeolusTemplateFile(language, projectType, staticAnalysis, sequentialRuns, coverage).subscribe((data) => { - expect(data).toEqual(response); - }); - - const req = httpMock.expectOne({ - url: expectedUrl, - method: 'GET', - }); - expect(req.request.responseType).toBe('text'); - req.flush(response); - }); - - it('should fetch the aeolus template file with missing optional parameters', () => { - const expectedUrl = `api/files/aeolus/templates/PYTHON?staticAnalysis=false&sequentialRuns=false&testCoverage=false`; - const response = 'aeolus template content'; - - fileService.getAeolusTemplateFile(ProgrammingLanguage.PYTHON).subscribe((data) => { - expect(data).toEqual(response); - }); - - const req = httpMock.expectOne({ - url: expectedUrl, - method: 'GET', - }); - expect(req.request.responseType).toBe('text'); - req.flush(response); - }); - }); - describe('getTemplateCodeOfConduct', () => { it('should fetch the template code of conduct', () => { const expectedUrl = `api/files/templates/code-of-conduct`; const response = 'code of conduct content'; - fileService.getTemplateCodeOfCondcut().subscribe((data) => { + fileService.getTemplateCodeOfConduct().subscribe((data) => { expect(data.body).toEqual(response); }); diff --git a/src/test/javascript/spec/component/shared/navbar.component.spec.ts b/src/test/javascript/spec/component/shared/navbar.component.spec.ts index bc2e7cb37b92..5de3b28f3e38 100644 --- a/src/test/javascript/spec/component/shared/navbar.component.spec.ts +++ b/src/test/javascript/spec/component/shared/navbar.component.spec.ts @@ -1,11 +1,10 @@ -import { HttpResponse, provideHttpClient } from '@angular/common/http'; +import { provideHttpClient } from '@angular/common/http'; import { ComponentFixture, TestBed, fakeAsync, tick } from '@angular/core/testing'; import { ActivatedRoute, Router, UrlSerializer } from '@angular/router'; import { TranslateService } from '@ngx-translate/core'; import { MockTranslateService } from '../../helpers/mocks/service/mock-translate.service'; import { ProfileService } from 'app/shared/layouts/profiles/profile.service'; import { ProfileInfo } from 'app/shared/layouts/profiles/profile-info.model'; -import { ExerciseService } from 'app/exercises/shared/exercise/exercise.service'; import { GuidedTourComponent } from 'app/guided-tour/guided-tour.component'; import { HasAnyAuthorityDirective } from 'app/shared/auth/has-any-authority.directive'; import { FindLanguageFromKeyPipe } from 'app/shared/language/find-language-from-key.pipe'; @@ -21,7 +20,6 @@ import { of } from 'rxjs'; import { MockRouter } from '../../helpers/mocks/mock-router'; import { MockSyncStorage } from '../../helpers/mocks/service/mock-sync-storage.service'; import { MockRouterLinkActiveOptionsDirective, MockRouterLinkDirective } from '../../helpers/mocks/directive/mock-router-link.directive'; -import { Exercise, ExerciseType } from 'app/entities/exercise.model'; import { JhiConnectionWarningComponent } from 'app/shared/connection-warning/connection-warning.component'; import { AccountService } from 'app/core/auth/account.service'; import { MockAccountService } from '../../helpers/mocks/service/mock-account.service'; @@ -52,7 +50,6 @@ describe('NavbarComponent', () => { let fixture: ComponentFixture; let component: NavbarComponent; let entityTitleServiceStub: jest.SpyInstance; - let exerciseService: ExerciseService; let entityTitleService: EntityTitleService; let examParticipationService: ExamParticipationService; @@ -154,8 +151,6 @@ describe('NavbarComponent', () => { entityTitleServiceStub = jest .spyOn(entityTitleService, 'getTitle') .mockImplementation((type) => of('Test ' + type.substring(0, 1) + type.substring(1).toLowerCase())); - - exerciseService = fixture.debugElement.injector.get(ExerciseService); }); }); @@ -443,57 +438,6 @@ describe('NavbarComponent', () => { expect(component.breadcrumbs[4]).toEqual(assessmentCrumb); }); - it('programming exercise hints', () => { - const testUrl = '/course-management/1/exercises/2/exercise-hints/3'; - router.setUrl(testUrl); - - const findStub = jest.spyOn(exerciseService, 'find').mockReturnValue( - of({ - body: { - title: 'Test Exercise', - type: ExerciseType.PROGRAMMING, - }, - } as HttpResponse), - ); - - fixture.detectChanges(); - - expect(entityTitleServiceStub).toHaveBeenCalledTimes(2); - expect(entityTitleServiceStub).toHaveBeenCalledWith(EntityType.COURSE, [1]); - expect(entityTitleServiceStub).toHaveBeenCalledWith(EntityType.HINT, [3, 2]); - expect(findStub).toHaveBeenCalledOnce(); - expect(findStub).toHaveBeenCalledWith(2); - - const hintsCrumb = { - label: 'artemisApp.exerciseHint.home.title', - translate: true, - uri: '/course-management/1/exercises/2/exercise-hints/', - } as MockBreadcrumb; - - const hintCrumb = { - label: 'Test Hint', - translate: false, - uri: '/course-management/1/exercises/2/exercise-hints/3/', - } as MockBreadcrumb; - - expect(component.breadcrumbs).toHaveLength(6); - - expect(component.breadcrumbs[0]).toEqual(courseManagementCrumb); - expect(component.breadcrumbs[1]).toEqual(testCourseCrumb); - expect(component.breadcrumbs[2]).toEqual({ - label: 'artemisApp.course.exercises', - translate: true, - uri: '/course-management/1/exercises/', - } as MockBreadcrumb); - expect(component.breadcrumbs[3]).toEqual({ - label: 'Test Exercise', - translate: false, - uri: '/course-management/1/programming-exercises/2/', - } as MockBreadcrumb); - expect(component.breadcrumbs[4]).toEqual(hintsCrumb); - expect(component.breadcrumbs[5]).toEqual(hintCrumb); - }); - it('exercise assessment dashboard', () => { const courseId = 1; const exerciseId = 2; diff --git a/src/test/javascript/spec/helpers/mocks/service/mock-exercise-hint.service.ts b/src/test/javascript/spec/helpers/mocks/service/mock-exercise-hint.service.ts deleted file mode 100644 index a3767d2437d0..000000000000 --- a/src/test/javascript/spec/helpers/mocks/service/mock-exercise-hint.service.ts +++ /dev/null @@ -1,49 +0,0 @@ -import { HttpResponse } from '@angular/common/http'; -import { Observable, of } from 'rxjs'; -import { ExerciseHintResponse, IExerciseHintService } from 'app/exercises/shared/exercise-hint/shared/exercise-hint.service'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; - -export class MockExerciseHintService implements IExerciseHintService { - private exerciseHintDummy = { id: 1 } as ExerciseHint; - private exerciseHintDummy2 = { id: 2 } as ExerciseHint; - - create(exerciseId: number, exerciseHint: ExerciseHint): Observable { - return of({ body: this.exerciseHintDummy }) as Observable; - } - - delete(exerciseId: number, exerciseHintId: number): Observable> { - return of(); - } - - find(exerciseId: number, exerciseHintId: number): Observable { - return of({ body: this.exerciseHintDummy }) as Observable; - } - - findByExerciseId(exerciseId: number): Observable> { - return of({ body: [this.exerciseHintDummy, this.exerciseHintDummy2] }) as Observable>; - } - - update(exerciseId: number, exerciseHint: ExerciseHint): Observable { - return of({ body: this.exerciseHintDummy }) as Observable; - } - - activateExerciseHint(exerciseId: number, exerciseHintId: number): Observable { - return of(); - } - - getActivatedExerciseHints(exerciseId: number): Observable> { - return of(); - } - - getAvailableExerciseHints(exerciseId: number): Observable> { - return of(); - } - - getTitle(exerciseId: number, exerciseHintId: number): Observable> { - return of(); - } - - rateExerciseHint(exerciseId: number, exerciseHintId: number, ratingValue: number): Observable> { - return of(); - } -} diff --git a/src/test/javascript/spec/helpers/mocks/service/mock-programming-exercise.service.ts b/src/test/javascript/spec/helpers/mocks/service/mock-programming-exercise.service.ts index 27c111ba7665..70860b29e7c1 100644 --- a/src/test/javascript/spec/helpers/mocks/service/mock-programming-exercise.service.ts +++ b/src/test/javascript/spec/helpers/mocks/service/mock-programming-exercise.service.ts @@ -2,7 +2,6 @@ import { of } from 'rxjs'; import { ProgrammingExerciseInstructorRepositoryType } from 'app/exercises/programming/manage/services/programming-exercise.service'; import { Participation } from 'app/entities/participation/participation.model'; import { ProgrammingLanguage } from 'app/entities/programming/programming-exercise.model'; -import { ProgrammingExerciseStudentParticipation } from 'app/entities/participation/programming-exercise-student-participation.model'; export class MockProgrammingExerciseService { updateProblemStatement = (exerciseId: number, problemStatement: string) => of(); @@ -16,14 +15,9 @@ export class MockProgrammingExerciseService { exportInstructorRepository = (exerciseId: number, repositoryType: ProgrammingExerciseInstructorRepositoryType) => of({ body: undefined }); exportStudentRepository = (exerciseId: number, participationId: number) => of({ body: undefined }); exportStudentRequestedRepository = (exerciseId: number, includeTests: boolean) => of({ body: undefined }); - getTasksAndTestsExtractedFromProblemStatement = (exerciseId: number) => of(); - deleteTasksWithSolutionEntries = (exerciseId: number) => of(); getDiffReport = (exerciseId: number) => of({}); getBuildLogStatistics = (exerciseId: number) => of({}); - createStructuralSolutionEntries = (exerciseId: number) => of({}); - createBehavioralSolutionEntries = (exerciseId: number) => of({}); getLatestResult = (participation: Participation) => of({}); - getLatestFullTestwiseCoverageReport = (exerciseId: number) => of({}); combineTemplateRepositoryCommits = (exerciseId: number) => of({}); delete = (programmingExerciseId: number, deleteStudentReposBuildPlans: boolean, deleteBaseReposBuildPlans: boolean) => of({}); generateStructureOracle = (exerciseId: number) => of({}); diff --git a/src/test/javascript/spec/integration/code-editor/code-editor-student.integration.spec.ts b/src/test/javascript/spec/integration/code-editor/code-editor-student.integration.spec.ts index 5df8e8e32ce3..6b3b8441f2e7 100644 --- a/src/test/javascript/spec/integration/code-editor/code-editor-student.integration.spec.ts +++ b/src/test/javascript/spec/integration/code-editor/code-editor-student.integration.spec.ts @@ -1,5 +1,4 @@ import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { ExerciseHintButtonOverlayComponent } from 'app/exercises/shared/exercise-hint/participate/exercise-hint-button-overlay.component'; import { LocalStorageService, SessionStorageService } from 'ngx-webstorage'; import dayjs from 'dayjs/esm'; import { JhiLanguageHelper } from 'app/core/language/language.helper'; @@ -107,7 +106,6 @@ describe('CodeEditorStudentIntegration', () => { MockComponent(CodeEditorFileBrowserFolderComponent), MockComponent(CodeEditorFileBrowserFileComponent), MockComponent(CodeEditorStatusComponent), - MockComponent(ExerciseHintButtonOverlayComponent), TreeviewComponent, ], providers: [ diff --git a/src/test/javascript/spec/service/entity-title.service.spec.ts b/src/test/javascript/spec/service/entity-title.service.spec.ts index a0d25c344ae6..7bba4142c05a 100644 --- a/src/test/javascript/spec/service/entity-title.service.spec.ts +++ b/src/test/javascript/spec/service/entity-title.service.spec.ts @@ -38,7 +38,6 @@ describe('EntityTitleService', () => { }); it.each([ - { type: EntityType.HINT, ids: [3, 2], url: 'programming-exercises/2/exercise-hints/3' }, { type: EntityType.EXERCISE, ids: [1], url: 'exercises/1' }, { type: EntityType.LECTURE, ids: [1], url: 'lectures/1' }, { type: EntityType.COURSE, ids: [1], url: 'courses/1' }, @@ -82,7 +81,6 @@ describe('EntityTitleService', () => { it.each([ { type: EntityType.COURSE, ids: [] }, { type: EntityType.COURSE, ids: [undefined] }, - { type: EntityType.HINT, ids: [1, undefined] }, { type: undefined, ids: [undefined] }, { type: undefined, ids: [] }, ])('captures an exception if invalid parameters are supplied to getTitle', ({ type, ids }) => { @@ -100,12 +98,10 @@ describe('EntityTitleService', () => { it.each([ { type: EntityType.COURSE, ids: [], title: 'Test' }, { type: EntityType.COURSE, ids: [undefined], title: 'Test' }, - { type: EntityType.HINT, ids: [1, undefined], title: 'Test' }, { type: undefined, ids: [undefined], title: 'Test' }, { type: undefined, ids: [], title: 'Test' }, { type: EntityType.COURSE, ids: [], title: undefined }, { type: EntityType.COURSE, ids: [undefined], title: undefined }, - { type: EntityType.HINT, ids: [1, undefined], title: undefined }, { type: undefined, ids: [undefined], title: undefined }, { type: undefined, ids: [], title: undefined }, ])('captures an exception if invalid parameters are supplied to setTitle', ({ type, ids, title }) => { diff --git a/src/test/javascript/spec/service/exercise-hint.service.spec.ts b/src/test/javascript/spec/service/exercise-hint.service.spec.ts deleted file mode 100644 index b988b61db162..000000000000 --- a/src/test/javascript/spec/service/exercise-hint.service.spec.ts +++ /dev/null @@ -1,111 +0,0 @@ -import { TestBed } from '@angular/core/testing'; -import { HttpTestingController, provideHttpClientTesting } from '@angular/common/http/testing'; -import { HttpResponse, provideHttpClient } from '@angular/common/http'; -import { take } from 'rxjs/operators'; -import { ExerciseService } from 'app/exercises/shared/exercise/exercise.service'; -import { ProgrammingExercise } from 'app/entities/programming/programming-exercise.model'; -import { ExerciseHint } from 'app/entities/hestia/exercise-hint.model'; -import { MockExerciseService } from '../helpers/mocks/service/mock-exercise.service'; -import { ExerciseHintService } from 'app/exercises/shared/exercise-hint/shared/exercise-hint.service'; - -describe('ExerciseHint Service', () => { - let service: ExerciseHintService; - let httpMock: HttpTestingController; - let elemDefault: ExerciseHint; - let exerciseHint: ExerciseHint; - let expectedResult: any; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [], - providers: [provideHttpClient(), provideHttpClientTesting(), { provide: ExerciseService, useClass: MockExerciseService }], - }); - expectedResult = {} as HttpResponse; - service = TestBed.inject(ExerciseHintService); - httpMock = TestBed.inject(HttpTestingController); - - const exercise = new ProgrammingExercise(undefined, undefined); - exercise.id = 1; - - elemDefault = new ExerciseHint(); - elemDefault.id = 0; - elemDefault.title = 'AAAAAAA'; - elemDefault.content = 'AAAAAAA'; - elemDefault.exercise = exercise; - - exerciseHint = new ExerciseHint(); - exerciseHint.title = 'AAAAA'; - exerciseHint.content = 'BBBBB'; - exerciseHint.exercise = exercise; - }); - - describe('Service methods', () => { - it('should find an element', () => { - const returnedFromService = Object.assign({}, elemDefault); - service - .find(1, 123) - .pipe(take(1)) - .subscribe((resp) => (expectedResult = resp)); - - const req = httpMock.expectOne({ method: 'GET' }); - req.flush(returnedFromService); - expect(expectedResult.body).toEqual(elemDefault); - }); - }); - - it('should create an ExerciseHint', () => { - const returnedFromService = Object.assign( - { - id: 0, - title: 'AAAAA', - content: 'BBBBBB', - exercise: { - id: 1, - }, - }, - elemDefault, - ); - const expected = Object.assign({}, returnedFromService); - service - .create(1, exerciseHint) - .pipe(take(1)) - .subscribe((resp) => (expectedResult = resp)); - const req = httpMock.expectOne({ method: 'POST' }); - req.flush(returnedFromService); - expect(expectedResult.body).toEqual(expected); - }); - - it('should update an ExerciseHint', () => { - const returnedFromService = Object.assign( - { - title: 'BBBBBB', - content: 'BBBBBB', - exercise: { - id: 1, - }, - }, - elemDefault, - ); - - const expected = Object.assign({}, returnedFromService); - service - .update(1, elemDefault) - .pipe(take(1)) - .subscribe((resp) => (expectedResult = resp)); - const req = httpMock.expectOne({ method: 'PUT' }); - req.flush(returnedFromService); - expect(expectedResult.body).toEqual(expected); - }); - - it('should delete an ExerciseHint', () => { - service.delete(1, 123).subscribe((resp) => (expectedResult = resp.ok)); - - const req = httpMock.expectOne({ method: 'DELETE' }); - req.flush({ status: 200 }); - expect(expectedResult).toBeTrue(); - }); - - afterEach(() => { - httpMock.verify(); - }); -}); diff --git a/src/test/javascript/spec/service/profile.service.spec.ts b/src/test/javascript/spec/service/profile.service.spec.ts index a210b70eeed0..aeb91a792501 100644 --- a/src/test/javascript/spec/service/profile.service.spec.ts +++ b/src/test/javascript/spec/service/profile.service.spec.ts @@ -76,7 +76,6 @@ describe('ProfileService', () => { packageNameRequired: true, checkoutSolutionRepositoryAllowed: false, projectTypes: [], - testwiseCoverageAnalysisSupported: true, auxiliaryRepositoriesSupported: true, }, { @@ -87,7 +86,6 @@ describe('ProfileService', () => { packageNameRequired: false, checkoutSolutionRepositoryAllowed: false, projectTypes: [], - testwiseCoverageAnalysisSupported: false, auxiliaryRepositoriesSupported: true, }, { @@ -98,7 +96,6 @@ describe('ProfileService', () => { packageNameRequired: true, checkoutSolutionRepositoryAllowed: false, projectTypes: ['PLAIN', 'XCODE'], - testwiseCoverageAnalysisSupported: false, auxiliaryRepositoriesSupported: true, }, { @@ -109,7 +106,6 @@ describe('ProfileService', () => { packageNameRequired: false, checkoutSolutionRepositoryAllowed: false, projectTypes: [], - testwiseCoverageAnalysisSupported: false, auxiliaryRepositoriesSupported: true, }, { @@ -120,7 +116,6 @@ describe('ProfileService', () => { packageNameRequired: true, checkoutSolutionRepositoryAllowed: false, projectTypes: ['PLAIN_MAVEN', 'MAVEN_MAVEN'], - testwiseCoverageAnalysisSupported: true, auxiliaryRepositoriesSupported: true, }, ], @@ -203,7 +198,6 @@ describe('ProfileService', () => { projectTypes: [], sequentialTestRuns: true, staticCodeAnalysis: false, - testwiseCoverageAnalysisSupported: true, auxiliaryRepositoriesSupported: true, }, { @@ -214,7 +208,6 @@ describe('ProfileService', () => { projectTypes: [], sequentialTestRuns: false, staticCodeAnalysis: false, - testwiseCoverageAnalysisSupported: false, auxiliaryRepositoriesSupported: true, }, { @@ -225,7 +218,6 @@ describe('ProfileService', () => { projectTypes: [ProjectType.PLAIN, ProjectType.XCODE], sequentialTestRuns: false, staticCodeAnalysis: true, - testwiseCoverageAnalysisSupported: false, auxiliaryRepositoriesSupported: true, }, { @@ -236,7 +228,6 @@ describe('ProfileService', () => { projectTypes: [], sequentialTestRuns: false, staticCodeAnalysis: false, - testwiseCoverageAnalysisSupported: false, auxiliaryRepositoriesSupported: true, }, { @@ -247,7 +238,6 @@ describe('ProfileService', () => { projectTypes: [ProjectType.PLAIN_MAVEN, ProjectType.MAVEN_MAVEN], sequentialTestRuns: true, staticCodeAnalysis: true, - testwiseCoverageAnalysisSupported: true, auxiliaryRepositoriesSupported: true, }, ], diff --git a/src/test/javascript/spec/service/programming-exercise-task.service.spec.ts b/src/test/javascript/spec/service/programming-exercise-task.service.spec.ts index a47cc762f73b..3b9b7b28e1ff 100644 --- a/src/test/javascript/spec/service/programming-exercise-task.service.spec.ts +++ b/src/test/javascript/spec/service/programming-exercise-task.service.spec.ts @@ -10,7 +10,7 @@ import { ProgrammingExerciseGradingStatistics, TestCaseStats } from 'app/entitie import { MockProvider } from 'ng-mocks'; import { MockProgrammingExerciseGradingService } from '../helpers/mocks/service/mock-programming-exercise-grading.service'; import { ProgrammingExerciseTask } from 'app/exercises/programming/manage/grading/tasks/programming-exercise-task'; -import { ProgrammingExerciseServerSideTask } from 'app/entities/hestia/programming-exercise-task.model'; +import { ProgrammingExerciseServerSideTask } from '../../../../main/webapp/app/entities/programming-exercise-task.model'; import { ProgrammingExerciseTestCase, ProgrammingExerciseTestCaseType, Visibility } from 'app/entities/programming/programming-exercise-test-case.model'; import { firstValueFrom, of } from 'rxjs'; import { provideHttpClient } from '@angular/common/http'; diff --git a/src/test/javascript/spec/service/programming-exercise.service.spec.ts b/src/test/javascript/spec/service/programming-exercise.service.spec.ts index bcc07eeecb3a..6473ce25204c 100644 --- a/src/test/javascript/spec/service/programming-exercise.service.spec.ts +++ b/src/test/javascript/spec/service/programming-exercise.service.spec.ts @@ -14,12 +14,11 @@ import { ProgrammingSubmission } from 'app/entities/programming/programming-subm import { Result } from 'app/entities/result.model'; import { AccountService } from 'app/core/auth/account.service'; import { MockAccountService } from '../helpers/mocks/service/mock-account.service'; -import { ProgrammingExerciseSolutionEntry } from 'app/entities/hestia/programming-exercise-solution-entry.model'; import { Course } from 'app/entities/course.model'; import { SolutionProgrammingExerciseParticipation } from 'app/entities/participation/solution-programming-exercise-participation.model'; import { Submission } from 'app/entities/submission.model'; -import { ProgrammingExerciseGitDiffReport } from 'app/entities/hestia/programming-exercise-git-diff-report.model'; -import { ProgrammingExerciseGitDiffEntry } from 'app/entities/hestia/programming-exercise-git-diff-entry.model'; +import { ProgrammingExerciseGitDiffReport } from '../../../../main/webapp/app/entities/programming-exercise-git-diff-report.model'; +import { ProgrammingExerciseGitDiffEntry } from '../../../../main/webapp/app/entities/programming-exercise-git-diff-entry.model'; import { AuxiliaryRepository } from 'app/entities/programming/programming-exercise-auxiliary-repository-model'; import { provideHttpClient } from '@angular/common/http'; @@ -283,24 +282,6 @@ describe('ProgrammingExercise Service', () => { })); }); - it('should make post request for structural solution entries', fakeAsync(() => { - const expected = [new ProgrammingExerciseSolutionEntry()]; - expected[0].filePath = 'src/test.java'; - service.createStructuralSolutionEntries(123).subscribe((resp) => expect(resp).toEqual(expected)); - const req = httpMock.expectOne({ method: 'POST', url: `${resourceUrl}/123/structural-solution-entries` }); - req.flush(expected); - tick(); - })); - - it('should make post request for behavioral solution entries', fakeAsync(() => { - const expected = [new ProgrammingExerciseSolutionEntry()]; - expected[0].filePath = 'src/test.java'; - service.createBehavioralSolutionEntries(123).subscribe((resp) => expect(resp).toEqual(expected)); - const req = httpMock.expectOne({ method: 'POST', url: `${resourceUrl}/123/behavioral-solution-entries` }); - req.flush(expected); - tick(); - })); - it('should make post request for import from file', fakeAsync(() => { const course = new Course(); course.id = 1; @@ -444,12 +425,7 @@ describe('ProgrammingExercise Service', () => { { uri: 'check-plagiarism', method: 'checkPlagiarism' }, { uri: 'plagiarism-result', method: 'getLatestPlagiarismResult' }, { uri: 'test-case-state', method: 'getProgrammingExerciseTestCaseState' }, - { uri: 'tasks', method: 'getTasksAndTestsExtractedFromProblemStatement' }, { uri: 'diff-report', method: 'getDiffReport' }, - { uri: 'full-testwise-coverage-report', method: 'getLatestFullTestwiseCoverageReport' }, - { uri: 'file-names', method: 'getSolutionFileNames' }, - { uri: 'exercise-hints', method: 'getCodeHintsForExercise' }, - { uri: 'test-cases', method: 'getAllTestCases' }, { uri: 'build-log-statistics', method: 'getBuildLogStatistics' }, ])('should call correct exercise endpoint', (test) => fakeAsync(() => { diff --git a/src/test/playwright/e2e/exercise/programming/ProgrammingExerciseAssessment.spec.ts b/src/test/playwright/e2e/exercise/programming/ProgrammingExerciseAssessment.spec.ts index 0299bd423774..fc1282c3fc36 100644 --- a/src/test/playwright/e2e/exercise/programming/ProgrammingExerciseAssessment.spec.ts +++ b/src/test/playwright/e2e/exercise/programming/ProgrammingExerciseAssessment.spec.ts @@ -33,7 +33,6 @@ test.describe('Programming exercise assessment', { tag: '@sequential' }, () => { assessmentDueDate = dueDate.add(20, 'seconds'); exercise = await exerciseAPIRequests.createProgrammingExercise({ course, - recordTestwiseCoverage: false, releaseDate: dayjs(), dueDate: dueDate, assessmentDate: assessmentDueDate, diff --git a/src/test/playwright/fixtures/exercise/programming/python/template.json b/src/test/playwright/fixtures/exercise/programming/python/template.json index fe3456326355..a9ffcf14c969 100644 --- a/src/test/playwright/fixtures/exercise/programming/python/template.json +++ b/src/test/playwright/fixtures/exercise/programming/python/template.json @@ -33,7 +33,6 @@ "templateParticipation": { "type": "template" }, - "testwiseCoverageEnabled": false, "title": "", "type": "programming", "buildConfig": { diff --git a/src/test/playwright/support/requests/ExerciseAPIRequests.ts b/src/test/playwright/support/requests/ExerciseAPIRequests.ts index 20d55b216fa8..e9c836fb3e1d 100644 --- a/src/test/playwright/support/requests/ExerciseAPIRequests.ts +++ b/src/test/playwright/support/requests/ExerciseAPIRequests.ts @@ -70,7 +70,6 @@ export class ExerciseAPIRequests { * - exerciseGroup: The exercise group the exercise will be added to * - scaMaxPenalty: The max percentage (0-100) static code analysis can reduce from the points * If sca should be disabled, pass null or omit this property - * - recordTestwiseCoverage: Enable testwise coverage analysis for this exercise * - releaseDate: When the programming exercise should be available * - dueDate: When the programming exercise should be due * - title: The title of the programming exercise @@ -85,7 +84,6 @@ export class ExerciseAPIRequests { course?: Course; exerciseGroup?: ExerciseGroup; scaMaxPenalty?: number | undefined; - recordTestwiseCoverage?: boolean; releaseDate?: dayjs.Dayjs; dueDate?: dayjs.Dayjs; title?: string; @@ -101,7 +99,6 @@ export class ExerciseAPIRequests { course, exerciseGroup, scaMaxPenalty = undefined, - recordTestwiseCoverage = false, releaseDate = dayjs(), dueDate = dayjs().add(1, 'day'), title = 'Programming ' + generateUUID(), @@ -147,7 +144,6 @@ export class ExerciseAPIRequests { } exercise.programmingLanguage = programmingLanguage; - exercise.buildConfig!.testwiseCoverageEnabled = recordTestwiseCoverage; exercise.mode = mode; exercise.teamAssignmentConfig = teamAssignmentConfig; diff --git a/supporting_scripts/course-scripts/quick-course-setup/manage_programming_exercise.py b/supporting_scripts/course-scripts/quick-course-setup/manage_programming_exercise.py index ab2dfc469d5d..b550fb565d36 100644 --- a/supporting_scripts/course-scripts/quick-course-setup/manage_programming_exercise.py +++ b/supporting_scripts/course-scripts/quick-course-setup/manage_programming_exercise.py @@ -29,7 +29,6 @@ def create_programming_exercise(session: Session, course_id: int, server_url: st "buildConfig": { "buildScript": "#!/usr/bin/env bash\nset -e\n\ngradle () {\n echo '⚙️ executing gradle'\n chmod +x ./gradlew\n ./gradlew clean test\n}\n\nmain () {\n gradle\n}\n\nmain \"${@}\"\n", "checkoutSolutionRepository": False, - "testwiseCoverageEnabled": False }, } From 4e562380afa496653508212fe686a1af7614762b Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Sat, 4 Jan 2025 11:20:28 +0100 Subject: [PATCH 06/34] Development: Implement small server code and log improvements --- .../core/repository/DataExportRepository.java | 3 ++- .../artemis/core/security/jwt/JWTFilter.java | 19 ++++++++++++------- .../core/security/jwt/TokenProvider.java | 17 +++++++---------- .../ContinuousPlagiarismControlService.java | 3 +-- .../service/PlagiarismDetectionService.java | 6 ++---- ...ProgrammingPlagiarismDetectionService.java | 4 +--- .../TextPlagiarismDetectionService.java | 2 -- .../service/BuildLogEntryService.java | 2 -- ...ProgrammingExercisePlagiarismResource.java | 2 +- src/main/resources/config/application.yml | 2 ++ ...InternalAuthenticationIntegrationTest.java | 2 +- .../jwt/TokenProviderSecurityMetersTest.java | 10 +++++----- .../core/security/jwt/TokenProviderTest.java | 10 +++++----- ...ontinuousPlagiarismControlServiceTest.java | 14 ++++++-------- 14 files changed, 45 insertions(+), 51 deletions(-) diff --git a/src/main/java/de/tum/cit/aet/artemis/core/repository/DataExportRepository.java b/src/main/java/de/tum/cit/aet/artemis/core/repository/DataExportRepository.java index fb87a2b0a8b0..33d9c104b5f6 100644 --- a/src/main/java/de/tum/cit/aet/artemis/core/repository/DataExportRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/core/repository/DataExportRepository.java @@ -37,7 +37,7 @@ public interface DataExportRepository extends ArtemisJpaRepository findAllToBeCreated(); /** - * Find all data exports that need to be deleted. This includes all data exports that have a creation date older than 7 days + * Find all data exports that need to be deleted. This includes all data exports that have a creation date older than 7 days and that have not been deleted or failed before * * @param thresholdDate the date to filter data exports, typically 7 days before today. * @return a set of data exports that need to be deleted @@ -47,6 +47,7 @@ public interface DataExportRepository extends ArtemisJpaRepository findAllToBeDeleted(@Param("thresholdDate") ZonedDateTime thresholdDate); diff --git a/src/main/java/de/tum/cit/aet/artemis/core/security/jwt/JWTFilter.java b/src/main/java/de/tum/cit/aet/artemis/core/security/jwt/JWTFilter.java index ff1ddcaaf3e3..d83fd44615b6 100644 --- a/src/main/java/de/tum/cit/aet/artemis/core/security/jwt/JWTFilter.java +++ b/src/main/java/de/tum/cit/aet/artemis/core/security/jwt/JWTFilter.java @@ -62,7 +62,8 @@ public void doFilter(ServletRequest servletRequest, ServletResponse servletRespo * @param tokenProvider the Artemis token provider used to generate and validate jwt's * @return the valid jwt or null if not found or invalid */ - public static @Nullable String extractValidJwt(HttpServletRequest httpServletRequest, TokenProvider tokenProvider) { + @Nullable + public static String extractValidJwt(HttpServletRequest httpServletRequest, TokenProvider tokenProvider) { var cookie = WebUtils.getCookie(httpServletRequest, JWT_COOKIE_NAME); var authHeader = httpServletRequest.getHeader(AUTHORIZATION_HEADER); @@ -76,8 +77,9 @@ public void doFilter(ServletRequest servletRequest, ServletResponse servletRespo } String jwtToken = cookie != null ? getJwtFromCookie(cookie) : getJwtFromBearer(authHeader); + String source = cookie != null ? "cookie" : "bearer"; - if (!isJwtValid(tokenProvider, jwtToken)) { + if (!isJwtValid(tokenProvider, jwtToken, source)) { return null; } @@ -90,7 +92,8 @@ public void doFilter(ServletRequest servletRequest, ServletResponse servletRespo * @param jwtCookie the cookie with Key "jwt" * @return the jwt or null if not found */ - private static @Nullable String getJwtFromCookie(@Nullable Cookie jwtCookie) { + @Nullable + private static String getJwtFromCookie(@Nullable Cookie jwtCookie) { if (jwtCookie == null) { return null; } @@ -103,7 +106,8 @@ public void doFilter(ServletRequest servletRequest, ServletResponse servletRespo * @param jwtBearer the content of the Authorization header * @return the jwt or null if not found */ - private static @Nullable String getJwtFromBearer(@Nullable String jwtBearer) { + @Nullable + private static String getJwtFromBearer(@Nullable String jwtBearer) { if (!StringUtils.hasText(jwtBearer) || !jwtBearer.startsWith(BEARER_PREFIX)) { return null; } @@ -116,10 +120,11 @@ public void doFilter(ServletRequest servletRequest, ServletResponse servletRespo * Checks if the jwt is valid * * @param tokenProvider the Artemis token provider used to generate and validate jwt's - * @param jwtToken the jwt + * @param jwtToken the jwt token which should be validated + * @param source the source of the jwt token * @return true if the jwt is valid, false if missing or invalid */ - private static boolean isJwtValid(TokenProvider tokenProvider, @Nullable String jwtToken) { - return StringUtils.hasText(jwtToken) && tokenProvider.validateTokenForAuthority(jwtToken); + private static boolean isJwtValid(TokenProvider tokenProvider, @Nullable String jwtToken, @Nullable String source) { + return StringUtils.hasText(jwtToken) && tokenProvider.validateTokenForAuthority(jwtToken, source); } } diff --git a/src/main/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProvider.java b/src/main/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProvider.java index de9a88d01dfa..98cb4560811d 100644 --- a/src/main/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProvider.java +++ b/src/main/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProvider.java @@ -8,6 +8,7 @@ import java.util.List; import java.util.stream.Collectors; +import jakarta.annotation.Nullable; import jakarta.annotation.PostConstruct; import javax.crypto.SecretKey; @@ -125,19 +126,21 @@ public Authentication getAuthentication(String token) { * Validate an JWT Authorization Token * * @param authToken JWT Authorization Token + * @param source the source of the token * @return boolean indicating if token is valid */ - public boolean validateTokenForAuthority(String authToken) { - return validateJwsToken(authToken); + public boolean validateTokenForAuthority(String authToken, @Nullable String source) { + return validateJwsToken(authToken, source); } /** * Validate an JWT Authorization Token * * @param authToken JWT Authorization Token + * @param source the source of the token * @return boolean indicating if token is valid */ - private boolean validateJwsToken(String authToken) { + private boolean validateJwsToken(String authToken, @Nullable String source) { try { parseClaims(authToken); return true; @@ -161,7 +164,7 @@ private boolean validateJwsToken(String authToken) { catch (IllegalArgumentException e) { log.error("Token validation error {}", e.getMessage()); } - log.info("Invalid JWT token: {}", authToken); + log.info("Invalid JWT token: {} from source {}", authToken, source); return false; } @@ -169,13 +172,7 @@ private Claims parseClaims(String authToken) { return Jwts.parser().verifyWith(key).build().parseSignedClaims(authToken).getPayload(); } - public T getClaim(String token, String claimName, Class claimType) { - Claims claims = parseClaims(token); - return claims.get(claimName, claimType); - } - public Date getExpirationDate(String authToken) { return parseClaims(authToken).getExpiration(); } - } diff --git a/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ContinuousPlagiarismControlService.java b/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ContinuousPlagiarismControlService.java index 0111e5129295..3c2933133a9b 100644 --- a/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ContinuousPlagiarismControlService.java +++ b/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ContinuousPlagiarismControlService.java @@ -74,9 +74,8 @@ public ContinuousPlagiarismControlService(ExerciseRepository exerciseRepository, */ @Scheduled(cron = "${artemis.scheduling.continuous-plagiarism-control-trigger-time:0 0 5 * * *}") public void executeChecks() { - log.info("Starting continuous plagiarism control..."); - var exercises = exerciseRepository.findAllExercisesWithDueDateOnOrAfterYesterdayAndContinuousPlagiarismControlEnabledIsTrue(); + log.info("Starting scheduled continuous plagiarism control for {} exercises: {}", exercises.size(), exercises.stream().map(Exercise::getId).toList()); exercises.stream().filter(isBeforeDueDateOrAfterWithPostDueDateChecksEnabled).forEach(exercise -> { log.info("Started continuous plagiarism control for exercise: exerciseId={}, type={}.", exercise.getId(), exercise.getExerciseType()); final long startTime = System.nanoTime(); diff --git a/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/PlagiarismDetectionService.java b/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/PlagiarismDetectionService.java index e8d439d26c2a..55150026c442 100644 --- a/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/PlagiarismDetectionService.java +++ b/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/PlagiarismDetectionService.java @@ -12,7 +12,6 @@ import org.springframework.context.annotation.Profile; import org.springframework.stereotype.Service; -import de.jplag.exceptions.ExitException; import de.tum.cit.aet.artemis.modeling.domain.ModelingExercise; import de.tum.cit.aet.artemis.plagiarism.domain.PlagiarismResult; import de.tum.cit.aet.artemis.plagiarism.domain.modeling.ModelingPlagiarismResult; @@ -60,7 +59,7 @@ public PlagiarismDetectionService(TextPlagiarismDetectionService textPlagiarismD * @param exercise exercise to check plagiarism * @return result of plagiarism checks */ - public TextPlagiarismResult checkTextExercise(TextExercise exercise) throws ExitException { + public TextPlagiarismResult checkTextExercise(TextExercise exercise) { var plagiarismResult = textPlagiarismDetectionService.checkPlagiarism(exercise, exercise.getPlagiarismDetectionConfig().getSimilarityThreshold(), exercise.getPlagiarismDetectionConfig().getMinimumScore(), exercise.getPlagiarismDetectionConfig().getMinimumSize()); log.info("Finished textPlagiarismDetectionService.checkPlagiarism for exercise {} with {} comparisons,", exercise.getId(), plagiarismResult.getComparisons().size()); @@ -75,8 +74,7 @@ public TextPlagiarismResult checkTextExercise(TextExercise exercise) throws Exit * @param exercise exercise to check plagiarism * @return result of plagiarism checks */ - public TextPlagiarismResult checkProgrammingExercise(ProgrammingExercise exercise) - throws ExitException, IOException, ProgrammingLanguageNotSupportedForPlagiarismDetectionException { + public TextPlagiarismResult checkProgrammingExercise(ProgrammingExercise exercise) throws IOException, ProgrammingLanguageNotSupportedForPlagiarismDetectionException { checkProgrammingLanguageSupport(exercise); var plagiarismResult = programmingPlagiarismDetectionService.checkPlagiarism(exercise.getId(), exercise.getPlagiarismDetectionConfig().getSimilarityThreshold(), diff --git a/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ProgrammingPlagiarismDetectionService.java b/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ProgrammingPlagiarismDetectionService.java index cec8919e81dd..3152d0612fa5 100644 --- a/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ProgrammingPlagiarismDetectionService.java +++ b/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/ProgrammingPlagiarismDetectionService.java @@ -32,7 +32,6 @@ import de.jplag.clustering.ClusteringOptions; import de.jplag.cpp.CPPLanguage; import de.jplag.csharp.CSharpLanguage; -import de.jplag.exceptions.ExitException; import de.jplag.golang.GoLanguage; import de.jplag.java.JavaLanguage; import de.jplag.javascript.JavaScriptLanguage; @@ -117,8 +116,7 @@ public ProgrammingPlagiarismDetectionService(FileService fileService, Programmin * @param minimumScore consider only submissions whose score is greater or equal to this value * @param minimumSize consider only submissions whose number of lines in diff to template is greater or equal to this value * @return the text plagiarism result container with up to 500 comparisons with the highest similarity values - * @throws ExitException is thrown if JPlag exits unexpectedly - * @throws IOException is thrown for file handling errors + * @throws IOException is thrown for file handling errors */ public TextPlagiarismResult checkPlagiarism(long programmingExerciseId, float similarityThreshold, int minimumScore, int minimumSize) throws IOException { long start = System.nanoTime(); diff --git a/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/TextPlagiarismDetectionService.java b/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/TextPlagiarismDetectionService.java index 6a5bbe7e945c..c69d979c3640 100644 --- a/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/TextPlagiarismDetectionService.java +++ b/src/main/java/de/tum/cit/aet/artemis/plagiarism/service/TextPlagiarismDetectionService.java @@ -21,7 +21,6 @@ import de.jplag.JPlagResult; import de.jplag.Language; import de.jplag.clustering.ClusteringOptions; -import de.jplag.exceptions.ExitException; import de.jplag.options.JPlagOptions; import de.jplag.text.NaturalLanguage; import de.tum.cit.aet.artemis.core.exception.BadRequestAlertException; @@ -84,7 +83,6 @@ public List textSubmissionsForComparison(TextExercise exerciseWi * @param minimumScore consider only submissions whose score is greater or equal to this value * @param minimumSize consider only submissions whose size is greater or equal to this value * @return a zip file that can be returned to the client - * @throws ExitException is thrown if JPlag exits unexpectedly */ public TextPlagiarismResult checkPlagiarism(TextExercise textExercise, float similarityThreshold, int minimumScore, int minimumSize) { // Only one plagiarism check per course allowed diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildLogEntryService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildLogEntryService.java index 039fce2c76bc..1ee0f9f1470a 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildLogEntryService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildLogEntryService.java @@ -399,8 +399,6 @@ private ProgrammingExercise retrieveProgrammingExerciseByBuildJobId(String build * In case of an error during file deletion, it logs the error and continues processing. *

* - * @throws IOException if an I/O error occurs while accessing the build log files directory or - * deleting files. */ @Scheduled(cron = "${artemis.continuous-integration.build-log.cleanup-schedule:0 0 3 * * ?}") public void deleteOldBuildLogsFiles() { diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExercisePlagiarismResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExercisePlagiarismResource.java index 95e60a0ec89c..1537d7e7cbe7 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExercisePlagiarismResource.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/web/ProgrammingExercisePlagiarismResource.java @@ -105,7 +105,7 @@ public ResponseEntity> checkPlagiarism log.info("Started manual plagiarism checks for programming exercise: exerciseId={}.", exerciseId); PlagiarismDetectionConfigHelper.updateWithTemporaryParameters(programmingExercise, similarityThreshold, minimumScore, minimumSize); try { - var plagiarismResult = (TextPlagiarismResult) plagiarismDetectionService.checkProgrammingExercise(programmingExercise); + var plagiarismResult = plagiarismDetectionService.checkProgrammingExercise(programmingExercise); return buildPlagiarismResultResponse(plagiarismResult); } catch (ProgrammingLanguageNotSupportedForPlagiarismDetectionException e) { diff --git a/src/main/resources/config/application.yml b/src/main/resources/config/application.yml index 5a2441c2f919..82611f9ceac2 100644 --- a/src/main/resources/config/application.yml +++ b/src/main/resources/config/application.yml @@ -22,6 +22,8 @@ logging: org.springframework.web.socket.config: INFO liquibase: INFO de.jplag.Submission: ERROR + de.jplag.logging: WARN + de.jplag.reporting: WARN logback: rollingpolicy: max-history: 90 diff --git a/src/test/java/de/tum/cit/aet/artemis/core/authentication/InternalAuthenticationIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/core/authentication/InternalAuthenticationIntegrationTest.java index 99e424d35022..087ad4d8dfaf 100644 --- a/src/test/java/de/tum/cit/aet/artemis/core/authentication/InternalAuthenticationIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/core/authentication/InternalAuthenticationIntegrationTest.java @@ -234,7 +234,7 @@ void testJWTAuthentication() throws Exception { var responseBody = new ObjectMapper().readValue(response.getContentAsString(), new TypeReference>() { }); - assertThat(tokenProvider.validateTokenForAuthority(responseBody.get("access_token").toString())).isTrue(); + assertThat(tokenProvider.validateTokenForAuthority(responseBody.get("access_token").toString(), null)).isTrue(); } @Test diff --git a/src/test/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProviderSecurityMetersTest.java b/src/test/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProviderSecurityMetersTest.java index 40e8abfc01f8..9dd6339f6c6b 100644 --- a/src/test/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProviderSecurityMetersTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProviderSecurityMetersTest.java @@ -62,7 +62,7 @@ void testValidTokenShouldNotCountAnything() { String validToken = createValidToken(); - tokenProvider.validateTokenForAuthority(validToken); + tokenProvider.validateTokenForAuthority(validToken, null); assertThat(aggregate(counters)).isZero(); } @@ -73,7 +73,7 @@ void testTokenExpiredCount() { String expiredToken = createExpiredToken(); - tokenProvider.validateTokenForAuthority(expiredToken); + tokenProvider.validateTokenForAuthority(expiredToken, null); assertThat(meterRegistry.get(INVALID_TOKENS_METER_EXPECTED_NAME).tag("cause", "expired").counter().count()).isEqualTo(1); } @@ -84,7 +84,7 @@ void testTokenUnsupportedCount() { String unsupportedToken = createUnsupportedToken(); - tokenProvider.validateTokenForAuthority(unsupportedToken); + tokenProvider.validateTokenForAuthority(unsupportedToken, null); assertThat(meterRegistry.get(INVALID_TOKENS_METER_EXPECTED_NAME).tag("cause", "unsupported").counter().count()).isEqualTo(1); } @@ -95,7 +95,7 @@ void testTokenSignatureInvalidCount() { String tokenWithDifferentSignature = createTokenWithDifferentSignature(); - tokenProvider.validateTokenForAuthority(tokenWithDifferentSignature); + tokenProvider.validateTokenForAuthority(tokenWithDifferentSignature, null); assertThat(meterRegistry.get(INVALID_TOKENS_METER_EXPECTED_NAME).tag("cause", "invalid-signature").counter().count()).isEqualTo(1); } @@ -106,7 +106,7 @@ void testTokenMalformedCount() { String malformedToken = createMalformedToken(); - tokenProvider.validateTokenForAuthority(malformedToken); + tokenProvider.validateTokenForAuthority(malformedToken, null); assertThat(meterRegistry.get(INVALID_TOKENS_METER_EXPECTED_NAME).tag("cause", "malformed").counter().count()).isEqualTo(1); } diff --git a/src/test/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProviderTest.java b/src/test/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProviderTest.java index 11001e768351..320121b09fa9 100644 --- a/src/test/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProviderTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/core/security/jwt/TokenProviderTest.java @@ -55,7 +55,7 @@ void setup() { @Test void testReturnFalseWhenJWThasInvalidSignature() { - boolean isTokenValid = tokenProvider.validateTokenForAuthority(createTokenWithDifferentSignature()); + boolean isTokenValid = tokenProvider.validateTokenForAuthority(createTokenWithDifferentSignature(), null); assertThat(isTokenValid).isFalse(); } @@ -65,7 +65,7 @@ void testReturnFalseWhenJWTisMalformed() { Authentication authentication = createAuthentication(); String token = tokenProvider.createToken(authentication, false); String invalidToken = token.substring(1); - boolean isTokenValid = tokenProvider.validateTokenForAuthority(invalidToken); + boolean isTokenValid = tokenProvider.validateTokenForAuthority(invalidToken, null); assertThat(isTokenValid).isFalse(); } @@ -77,7 +77,7 @@ void testReturnFalseWhenJWTisExpired() { Authentication authentication = createAuthentication(); String token = tokenProvider.createToken(authentication, false); - boolean isTokenValid = tokenProvider.validateTokenForAuthority(token); + boolean isTokenValid = tokenProvider.validateTokenForAuthority(token, null); assertThat(isTokenValid).isFalse(); } @@ -86,14 +86,14 @@ void testReturnFalseWhenJWTisExpired() { void testReturnFalseWhenJWTisUnsupported() { String unsupportedToken = createUnsupportedToken(); - boolean isTokenValid = tokenProvider.validateTokenForAuthority(unsupportedToken); + boolean isTokenValid = tokenProvider.validateTokenForAuthority(unsupportedToken, null); assertThat(isTokenValid).isFalse(); } @Test void testReturnFalseWhenJWTisInvalid() { - boolean isTokenValid = tokenProvider.validateTokenForAuthority(""); + boolean isTokenValid = tokenProvider.validateTokenForAuthority("", null); assertThat(isTokenValid).isFalse(); } diff --git a/src/test/java/de/tum/cit/aet/artemis/plagiarism/ContinuousPlagiarismControlServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/plagiarism/ContinuousPlagiarismControlServiceTest.java index 1986d792856d..ed7947627b4f 100644 --- a/src/test/java/de/tum/cit/aet/artemis/plagiarism/ContinuousPlagiarismControlServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/plagiarism/ContinuousPlagiarismControlServiceTest.java @@ -19,8 +19,6 @@ import org.junit.jupiter.api.Test; -import de.jplag.exceptions.BasecodeException; -import de.jplag.exceptions.ExitException; import de.tum.cit.aet.artemis.core.domain.Course; import de.tum.cit.aet.artemis.core.domain.User; import de.tum.cit.aet.artemis.exercise.domain.Exercise; @@ -69,7 +67,7 @@ class ContinuousPlagiarismControlServiceTest { plagiarismCaseService, plagiarismCaseRepository, plagiarismPostService, plagiarismResultRepository); @Test - void shouldExecuteChecks() throws ExitException, IOException, ProgrammingLanguageNotSupportedForPlagiarismDetectionException { + void shouldExecuteChecks() throws IOException, ProgrammingLanguageNotSupportedForPlagiarismDetectionException { // given: text exercise with cpc enabled var textExercise = new TextExercise(); textExercise.setId(101L); @@ -114,7 +112,7 @@ void shouldExecuteChecks() throws ExitException, IOException, ProgrammingLanguag } @Test - void shouldAddSubmissionsToPlagiarismCase() throws ExitException { + void shouldAddSubmissionsToPlagiarismCase() { // given: text exercise with cpc enabled var exercise = new TextExercise(); exercise.setId(99L); @@ -152,7 +150,7 @@ void shouldAddSubmissionsToPlagiarismCase() throws ExitException { } @Test - void shouldRemoveStalePlagiarismCase() throws ExitException { + void shouldRemoveStalePlagiarismCase() { // given: text exercise with cpc enabled var exercise = new TextExercise(); exercise.setId(99L); @@ -211,12 +209,12 @@ void shouldDoNothingForFileUploadAndQuizExercises() { } @Test - void shouldSilentAnyJPlagExceptionsThrown() throws Exception { + void shouldSilentAnyJPlagExceptionsThrown() { // given var textExercise = new TextExercise(); textExercise.setId(123L); when(exerciseRepository.findAllExercisesWithDueDateOnOrAfterYesterdayAndContinuousPlagiarismControlEnabledIsTrue()).thenReturn(singleton(textExercise)); - when(plagiarismChecksService.checkTextExercise(textExercise)).thenThrow(new BasecodeException("msg")); + when(plagiarismChecksService.checkTextExercise(textExercise)).thenThrow(new NullPointerException("null")); // then assertThatNoException().isThrownBy(service::executeChecks); @@ -224,7 +222,7 @@ void shouldSilentAnyJPlagExceptionsThrown() throws Exception { } @Test - void shouldSilentAnyUnknownExceptionsThrown() throws Exception { + void shouldSilentAnyUnknownExceptionsThrown() { // given var textExercise = new TextExercise(); textExercise.setId(101L); From ec8e01092bdabf2da597e4a08ec5ae4cc59e26f5 Mon Sep 17 00:00:00 2001 From: Enea Gore <73840596+EneaGore@users.noreply.github.com> Date: Sat, 4 Jan 2025 12:02:15 +0100 Subject: [PATCH 07/34] Text exercises: Highlight selected result in result history timeline (#9845) --- .../participate/text-editor.component.html | 12 ++-- .../text/participate/text-editor.component.ts | 35 +++++------ .../result-history.component.html | 13 +++- .../result-history.component.ts | 19 +++--- .../result-history.component.spec.ts | 62 +++++++++++-------- .../text-editor/text-editor.component.spec.ts | 10 +-- 6 files changed, 86 insertions(+), 65 deletions(-) diff --git a/src/main/webapp/app/exercises/text/participate/text-editor.component.html b/src/main/webapp/app/exercises/text/participate/text-editor.component.html index c070d57e29ea..c240142e10c2 100644 --- a/src/main/webapp/app/exercises/text/participate/text-editor.component.html +++ b/src/main/webapp/app/exercises/text/participate/text-editor.component.html @@ -1,5 +1,5 @@
- @if (displayHeader) { + @if (displayHeader()) { {{ 'artemisApp.textSubmission.textEditor' | artemisTranslate }}: @@ -54,7 +54,7 @@ @if (isReadOnlyWithShowResult) { @if (showHistory) {
- +
} } @@ -64,7 +64,7 @@ @if (textExercise) { @@ -79,7 +79,7 @@ @@ -87,7 +87,7 @@ @@ -104,7 +104,7 @@ [disabled]="!isActive || !submission || !isOwnerOfParticipation" (keydown.tab)="onTextEditorTab(textEditor, $event)" (input)="onTextEditorInput($event)" - [hidden]="submission && !submission.submitted && isExamSummary" + [hidden]="submission && !submission.submitted && isExamSummary()" > @if (textExercise?.teamMode) { (); + displayHeader = input(true); + expandProblemStatement = input(true); + inputExercise = input(); + inputSubmission = input(); + inputParticipation = input(); + isExamSummary = input(false); textExercise: TextExercise; participation: StudentParticipation; @@ -109,7 +108,7 @@ export class TextEditorComponent implements OnInit, OnDestroy, ComponentCanDeact if (this.inputValuesArePresent()) { this.setupComponentWithInputValues(); } else { - const participationId = this.participationId !== undefined ? this.participationId : Number(this.route.snapshot.paramMap.get('participationId')); + const participationId = this.participationId() !== undefined ? this.participationId() : Number(this.route.snapshot.paramMap.get('participationId')); this.submissionId = Number(this.route.snapshot.paramMap.get('submissionId')) || undefined; if (Number.isNaN(participationId)) { @@ -121,7 +120,7 @@ export class TextEditorComponent implements OnInit, OnDestroy, ComponentCanDeact this.updateParticipation(this.participation, this.submissionId); }); - this.textService.get(participationId).subscribe({ + this.textService.get(participationId!).subscribe({ next: (data: StudentParticipation) => this.updateParticipation(data, this.submissionId), error: (error: HttpErrorResponse) => onError(this.alertService, error), }); @@ -164,7 +163,7 @@ export class TextEditorComponent implements OnInit, OnDestroy, ComponentCanDeact } private inputValuesArePresent(): boolean { - return !!(this.inputExercise || this.inputSubmission || this.inputParticipation); + return !!(this.inputExercise() || this.inputSubmission() || this.inputParticipation()); } /** @@ -175,14 +174,14 @@ export class TextEditorComponent implements OnInit, OnDestroy, ComponentCanDeact * @private */ private setupComponentWithInputValues() { - if (this.inputExercise) { - this.textExercise = this.inputExercise; + if (this.inputExercise() !== undefined) { + this.textExercise = this.inputExercise()!; } - if (this.inputSubmission) { - this.submission = this.inputSubmission; + if (this.inputSubmission() !== undefined) { + this.submission = this.inputSubmission()!; } - if (this.inputParticipation) { - this.participation = this.inputParticipation; + if (this.inputParticipation() !== undefined) { + this.participation = this.inputParticipation()!; } if (this.submission?.text) { diff --git a/src/main/webapp/app/overview/result-history/result-history.component.html b/src/main/webapp/app/overview/result-history/result-history.component.html index cfa6cdbb897e..3748b2755790 100644 --- a/src/main/webapp/app/overview/result-history/result-history.component.html +++ b/src/main/webapp/app/overview/result-history/result-history.component.html @@ -5,12 +5,19 @@ @for (result of displayedResults; track result; let i = $index) {
-
- +
+
(); + exercise = input(); + selectedResultId = input(); showPreviousDivider = false; displayedResults: Result[]; movedLastRatedResult: boolean; ngOnChanges(): void { - this.showPreviousDivider = this.results.length > MAX_RESULT_HISTORY_LENGTH; - if (this.exercise?.type === ExerciseType.TEXT || this.exercise?.type === ExerciseType.MODELING) { - this.displayedResults = this.results.filter((result) => result.successful !== undefined); + this.showPreviousDivider = this.results().length > MAX_RESULT_HISTORY_LENGTH; + if (this.exercise()?.type === ExerciseType.TEXT || this.exercise()?.type === ExerciseType.MODELING) { + this.displayedResults = this.results().filter((result) => result.successful !== undefined); } else { - this.displayedResults = this.results; + this.displayedResults = this.results(); } const successfulResultsLength = this.displayedResults.length; if (successfulResultsLength > MAX_RESULT_HISTORY_LENGTH) { this.displayedResults = this.displayedResults.slice(successfulResultsLength - MAX_RESULT_HISTORY_LENGTH); - const lastRatedResult = this.results.filter((result) => result.rated).last(); + const lastRatedResult = this.results() + .filter((result) => result.rated) + .last(); if (!this.displayedResults.first()?.rated && lastRatedResult) { this.displayedResults[0] = lastRatedResult; this.movedLastRatedResult = true; diff --git a/src/test/javascript/spec/component/overview/result-history/result-history.component.spec.ts b/src/test/javascript/spec/component/overview/result-history/result-history.component.spec.ts index 8c5333322db6..b5746f24e492 100644 --- a/src/test/javascript/spec/component/overview/result-history/result-history.component.spec.ts +++ b/src/test/javascript/spec/component/overview/result-history/result-history.component.spec.ts @@ -1,8 +1,10 @@ +import { input, runInInjectionContext } from '@angular/core'; import { ComponentFixture, TestBed } from '@angular/core/testing'; import { ResultHistoryComponent } from 'app/overview/result-history/result-history.component'; import { MockPipe } from 'ng-mocks'; import { ArtemisDatePipe } from 'app/shared/pipes/artemis-date.pipe'; import { ArtemisTestModule } from '../../../test.module'; +import { Result } from 'app/entities/result.model'; describe('ResultHistoryComponent', () => { let component: ResultHistoryComponent; @@ -25,11 +27,13 @@ describe('ResultHistoryComponent', () => { }); it('should initialize with same rated results', () => { - component.results = [ - { rated: true, id: 1 }, - { rated: true, id: 2 }, - { rated: true, id: 3 }, - ]; + runInInjectionContext(TestBed, () => { + component.results = input([ + { rated: true, id: 1 }, + { rated: true, id: 2 }, + { rated: true, id: 3 }, + ]); + }); component.ngOnChanges(); expect(component.displayedResults).toEqual([ { rated: true, id: 1 }, @@ -39,14 +43,16 @@ describe('ResultHistoryComponent', () => { expect(component.showPreviousDivider).toBeFalse(); expect(component.movedLastRatedResult).toBeFalsy(); - component.results = [ - { rated: false, id: 1 }, - { rated: false, id: 2 }, - { rated: false, id: 3 }, - { rated: false, id: 4 }, - { rated: false, id: 5 }, - { rated: false, id: 6 }, - ]; + runInInjectionContext(TestBed, () => { + component.results = input([ + { rated: false, id: 1 }, + { rated: false, id: 2 }, + { rated: false, id: 3 }, + { rated: false, id: 4 }, + { rated: false, id: 5 }, + { rated: false, id: 6 }, + ]); + }); component.ngOnChanges(); expect(component.displayedResults).toEqual([ { rated: false, id: 2 }, @@ -60,11 +66,13 @@ describe('ResultHistoryComponent', () => { }); it('should initialize with mixed rated results', () => { - component.results = [ - { rated: true, id: 1 }, - { rated: false, id: 2 }, - { rated: false, id: 3 }, - ]; + runInInjectionContext(TestBed, () => { + component.results = input([ + { rated: true, id: 1 }, + { rated: false, id: 2 }, + { rated: false, id: 3 }, + ]); + }); component.ngOnChanges(); expect(component.displayedResults).toEqual([ { rated: true, id: 1 }, @@ -74,14 +82,16 @@ describe('ResultHistoryComponent', () => { expect(component.showPreviousDivider).toBeFalse(); expect(component.movedLastRatedResult).toBeFalsy(); - component.results = [ - { rated: true, id: 1 }, - { rated: false, id: 2 }, - { rated: false, id: 3 }, - { rated: false, id: 4 }, - { rated: false, id: 5 }, - { rated: false, id: 6 }, - ]; + runInInjectionContext(TestBed, () => { + component.results = input([ + { rated: true, id: 1 }, + { rated: false, id: 2 }, + { rated: false, id: 3 }, + { rated: false, id: 4 }, + { rated: false, id: 5 }, + { rated: false, id: 6 }, + ]); + }); component.ngOnChanges(); expect(component.displayedResults).toEqual([ { rated: true, id: 1 }, diff --git a/src/test/javascript/spec/component/text-editor/text-editor.component.spec.ts b/src/test/javascript/spec/component/text-editor/text-editor.component.spec.ts index 0f65d8bc2ef3..47902e658500 100644 --- a/src/test/javascript/spec/component/text-editor/text-editor.component.spec.ts +++ b/src/test/javascript/spec/component/text-editor/text-editor.component.spec.ts @@ -1,4 +1,4 @@ -import { DebugElement } from '@angular/core'; +import { DebugElement, input, runInInjectionContext } from '@angular/core'; import dayjs from 'dayjs/esm'; import { ActivatedRoute, RouterModule, convertToParamMap } from '@angular/router'; import { ComponentFixture, TestBed, fakeAsync, flush, tick } from '@angular/core/testing'; @@ -110,9 +110,11 @@ describe('TextEditorComponent', () => { }); it('should use inputValues if present instead of loading new details', fakeAsync(() => { - comp.inputExercise = textExercise; - comp.inputParticipation = participation; - comp.inputSubmission = { id: 1, text: 'test' }; + runInInjectionContext(TestBed, () => { + comp.inputExercise = input(textExercise); + comp.inputParticipation = input(participation); + comp.inputSubmission = input({ id: 1, text: 'test' }); + }); // @ts-ignore updateParticipation is private const updateParticipationSpy = jest.spyOn(comp, 'updateParticipation'); // @ts-ignore setupComponentWithInputValuesSpy is private From e2195994acf68517836a2d8dc7d4c58b7d77806e Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Sat, 4 Jan 2025 12:44:25 +0100 Subject: [PATCH 08/34] Development: Prevent exception in Iris for inactive exercises (#10099) --- .../assessment/service/ResultService.java | 4 +- .../web/ResultWebsocketService.java | 3 +- .../tum/cit/aet/artemis/core/domain/User.java | 4 ++ .../IrisExerciseChatSessionRepository.java | 3 - .../IrisExerciseSettingsRepository.java | 24 +++++++ .../iris/service/pyris/PyrisEventService.java | 6 +- .../lti/service/LtiNewResultService.java | 2 + .../service/ProgrammingMessagingService.java | 30 +++++--- .../service/BuildAgentDockerServiceTest.java | 3 +- ...ParticipationTeamWebsocketServiceTest.java | 9 +-- .../iris/PyrisEventSystemIntegrationTest.java | 68 +++++++------------ ...ctSpringIntegrationLocalCILocalVCTest.java | 38 +++++------ 12 files changed, 110 insertions(+), 84 deletions(-) create mode 100644 src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseSettingsRepository.java diff --git a/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java b/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java index 80d564e4695a..92b1f6e19030 100644 --- a/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java +++ b/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java @@ -198,8 +198,8 @@ public Result createNewManualResult(Result result, boolean ratedResult) { return savedResult; } - public Result createNewRatedManualResult(Result result) { - return createNewManualResult(result, true); + public void createNewRatedManualResult(Result result) { + createNewManualResult(result, true); } /** diff --git a/src/main/java/de/tum/cit/aet/artemis/assessment/web/ResultWebsocketService.java b/src/main/java/de/tum/cit/aet/artemis/assessment/web/ResultWebsocketService.java index 927cf64f7999..99089f035083 100644 --- a/src/main/java/de/tum/cit/aet/artemis/assessment/web/ResultWebsocketService.java +++ b/src/main/java/de/tum/cit/aet/artemis/assessment/web/ResultWebsocketService.java @@ -18,7 +18,6 @@ import de.tum.cit.aet.artemis.communication.service.WebsocketMessagingService; import de.tum.cit.aet.artemis.core.service.AuthorizationCheckService; import de.tum.cit.aet.artemis.exam.service.ExamDateService; -import de.tum.cit.aet.artemis.exercise.domain.Exercise; import de.tum.cit.aet.artemis.exercise.domain.Team; import de.tum.cit.aet.artemis.exercise.domain.participation.Participation; import de.tum.cit.aet.artemis.exercise.domain.participation.StudentParticipation; @@ -73,7 +72,7 @@ public void broadcastNewResult(Participation participation, Result result) { } private void broadcastNewResultToParticipants(StudentParticipation studentParticipation, Result result) { - final Exercise exercise = studentParticipation.getExercise(); + final var exercise = studentParticipation.getExercise(); boolean isWorkingPeriodOver; if (exercise.isExamExercise()) { isWorkingPeriodOver = examDateService.isIndividualExerciseWorkingPeriodOver(exercise.getExam(), studentParticipation); diff --git a/src/main/java/de/tum/cit/aet/artemis/core/domain/User.java b/src/main/java/de/tum/cit/aet/artemis/core/domain/User.java index c435194cdf40..cebc42aafde9 100644 --- a/src/main/java/de/tum/cit/aet/artemis/core/domain/User.java +++ b/src/main/java/de/tum/cit/aet/artemis/core/domain/User.java @@ -534,6 +534,10 @@ public void setIrisAcceptedTimestamp(@Nullable ZonedDateTime irisAccepted) { this.irisAccepted = irisAccepted; } + public boolean hasAcceptedIris() { + return irisAccepted != null; + } + /** * Checks if the user has accepted the Iris privacy policy. * If not, an {@link AccessForbiddenException} is thrown. diff --git a/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseChatSessionRepository.java b/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseChatSessionRepository.java index 514be1f1d575..c6da437ac731 100644 --- a/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseChatSessionRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseChatSessionRepository.java @@ -36,7 +36,6 @@ public interface IrisExerciseChatSessionRepository extends ArtemisJpaRepository< * @return A list of chat sessions sorted by creation date in descending order. */ @Query(""" - SELECT s FROM IrisExerciseChatSession s WHERE s.exercise.id = :exerciseId @@ -68,11 +67,9 @@ public interface IrisExerciseChatSessionRepository extends ArtemisJpaRepository< */ default List findLatestByExerciseIdAndUserIdWithMessages(Long exerciseId, Long userId, Pageable pageable) { List ids = findSessionsByExerciseIdAndUserId(exerciseId, userId, pageable).stream().map(DomainObject::getId).toList(); - if (ids.isEmpty()) { return Collections.emptyList(); } - return findSessionsWithMessagesByIdIn(ids); } diff --git a/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseSettingsRepository.java b/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseSettingsRepository.java new file mode 100644 index 000000000000..8035ec965a2b --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseSettingsRepository.java @@ -0,0 +1,24 @@ +package de.tum.cit.aet.artemis.iris.repository; + +import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_IRIS; + +import org.springframework.context.annotation.Profile; +import org.springframework.data.jpa.repository.Query; +import org.springframework.data.repository.query.Param; +import org.springframework.stereotype.Repository; + +import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; +import de.tum.cit.aet.artemis.iris.domain.settings.IrisExerciseSettings; + +@Repository +@Profile(PROFILE_IRIS) +public interface IrisExerciseSettingsRepository extends ArtemisJpaRepository { + + @Query(""" + SELECT EXISTS(s) + FROM IrisExerciseSettings s + WHERE s.exercise.id = :exerciseId + AND s.irisTextExerciseChatSettings.enabled = TRUE + """) + boolean isExerciseChatEnabled(@Param("exerciseId") long exerciseId); +} diff --git a/src/main/java/de/tum/cit/aet/artemis/iris/service/pyris/PyrisEventService.java b/src/main/java/de/tum/cit/aet/artemis/iris/service/pyris/PyrisEventService.java index 58f0b8a069b8..e2a8a785fc77 100644 --- a/src/main/java/de/tum/cit/aet/artemis/iris/service/pyris/PyrisEventService.java +++ b/src/main/java/de/tum/cit/aet/artemis/iris/service/pyris/PyrisEventService.java @@ -5,6 +5,7 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.context.annotation.Profile; +import org.springframework.scheduling.annotation.Async; import org.springframework.stereotype.Service; import de.tum.cit.aet.artemis.iris.domain.session.IrisChatSession; @@ -41,17 +42,18 @@ public PyrisEventService(IrisCourseChatSessionService irisCourseChatSessionServi * * @see PyrisEvent */ + @Async public void trigger(PyrisEvent, ?> event) { log.debug("Starting to process event of type: {}", event.getClass().getSimpleName()); try { switch (event) { case CompetencyJolSetEvent competencyJolSetEvent -> { - log.info("Processing CompetencyJolSetEvent: {}", competencyJolSetEvent); + log.debug("Processing CompetencyJolSetEvent: {}", competencyJolSetEvent); competencyJolSetEvent.handleEvent(irisCourseChatSessionService); log.debug("Successfully processed CompetencyJolSetEvent"); } case NewResultEvent newResultEvent -> { - log.info("Processing NewResultEvent: {}", newResultEvent); + log.debug("Processing NewResultEvent: {}", newResultEvent); newResultEvent.handleEvent(irisExerciseChatSessionService); log.debug("Successfully processed NewResultEvent"); } diff --git a/src/main/java/de/tum/cit/aet/artemis/lti/service/LtiNewResultService.java b/src/main/java/de/tum/cit/aet/artemis/lti/service/LtiNewResultService.java index 7a64018a6969..a1dc241afa40 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lti/service/LtiNewResultService.java +++ b/src/main/java/de/tum/cit/aet/artemis/lti/service/LtiNewResultService.java @@ -3,6 +3,7 @@ import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_LTI; import org.springframework.context.annotation.Profile; +import org.springframework.scheduling.annotation.Async; import org.springframework.stereotype.Service; import de.tum.cit.aet.artemis.exercise.domain.participation.StudentParticipation; @@ -22,6 +23,7 @@ public LtiNewResultService(Lti13Service lti13Service) { * * @param participation The exercise participation for which a new build result is available */ + @Async public void onNewResult(StudentParticipation participation) { if (!participation.getExercise().getCourseViaExerciseGroupOrCourseMember().isOnlineCourse()) { return; diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingMessagingService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingMessagingService.java index 349aa53bc89e..7a7861cbcbcc 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingMessagingService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/ProgrammingMessagingService.java @@ -28,6 +28,7 @@ import de.tum.cit.aet.artemis.exercise.dto.SubmissionDTO; import de.tum.cit.aet.artemis.exercise.repository.ParticipationRepository; import de.tum.cit.aet.artemis.exercise.repository.TeamRepository; +import de.tum.cit.aet.artemis.iris.repository.IrisExerciseSettingsRepository; import de.tum.cit.aet.artemis.iris.service.pyris.PyrisEventService; import de.tum.cit.aet.artemis.iris.service.pyris.event.NewResultEvent; import de.tum.cit.aet.artemis.lti.service.LtiNewResultService; @@ -57,16 +58,20 @@ public class ProgrammingMessagingService { private final Optional pyrisEventService; + private final Optional irisExerciseSettingsRepository; + private final ParticipationRepository participationRepository; public ProgrammingMessagingService(GroupNotificationService groupNotificationService, WebsocketMessagingService websocketMessagingService, ResultWebsocketService resultWebsocketService, Optional ltiNewResultService, TeamRepository teamRepository, - Optional pyrisEventService, ParticipationRepository participationRepository) { + Optional pyrisEventService, Optional irisExerciseSettingsRepository, + ParticipationRepository participationRepository) { this.groupNotificationService = groupNotificationService; this.websocketMessagingService = websocketMessagingService; this.resultWebsocketService = resultWebsocketService; this.ltiNewResultService = ltiNewResultService; this.teamRepository = teamRepository; + this.irisExerciseSettingsRepository = irisExerciseSettingsRepository; this.participationRepository = participationRepository; this.pyrisEventService = pyrisEventService; } @@ -188,13 +193,16 @@ public void notifyUserAboutNewResult(Result result, ProgrammingExerciseParticipa if (participation instanceof ProgrammingExerciseStudentParticipation studentParticipation) { // do not try to report results for template or solution participations ltiNewResultService.ifPresent(newResultService -> newResultService.onNewResult(studentParticipation)); - // Inform Iris about the submission status + // Inform Iris about the submission status (when certain conditions are met) notifyIrisAboutSubmissionStatus(result, studentParticipation); } } /** * Notify Iris about the submission status for the given result and student participation. + * Only notifies if the user has accepted Iris, the exercise is not an exam exercise, and the exercise chat is enabled in the exercise settings + * NOTE: we check those settings early to prevent unnecessary database queries and exceptions later on in most cases. More sophisticated checks are done in the Iris service. + *

* If the submission was successful, Iris will be informed about the successful submission. * If the submission failed, Iris will be informed about the submission failure. * Iris will only be informed about the submission status if the participant is a user. @@ -203,14 +211,18 @@ public void notifyUserAboutNewResult(Result result, ProgrammingExerciseParticipa * @param studentParticipation the student participation for which Iris should be informed about the submission status */ private void notifyIrisAboutSubmissionStatus(Result result, ProgrammingExerciseStudentParticipation studentParticipation) { - if (studentParticipation.getParticipant() instanceof User) { + if (studentParticipation.getParticipant() instanceof User user) { pyrisEventService.ifPresent(eventService -> { - // Inform event service about the new result - try { - eventService.trigger(new NewResultEvent(result)); - } - catch (Exception e) { - log.error("Could not trigger service for result {}", result.getId(), e); + final var exercise = studentParticipation.getExercise(); + if (user.hasAcceptedIris() && !exercise.isExamExercise() && irisExerciseSettingsRepository.get().isExerciseChatEnabled(exercise.getId())) { + // Inform event service about the new result + try { + // This is done asynchronously to prevent blocking the current thread + eventService.trigger(new NewResultEvent(result)); + } + catch (Exception e) { + log.error("Could not trigger service for result {}", result.getId(), e); + } } }); } diff --git a/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildAgentDockerServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildAgentDockerServiceTest.java index cdc2af3ec1b1..6b13982262f5 100644 --- a/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildAgentDockerServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/buildagent/service/BuildAgentDockerServiceTest.java @@ -4,6 +4,7 @@ import static org.mockito.Mockito.doReturn; import static org.mockito.Mockito.doThrow; import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.never; import static org.mockito.Mockito.times; import static org.mockito.Mockito.verify; @@ -83,7 +84,7 @@ void testDeleteOldDockerImages_NoOutdatedImages() { buildAgentDockerService.deleteOldDockerImages(); // Verify that removeImageCmd() was not called. - verify(dockerClient, times(0)).removeImageCmd(anyString()); + verify(dockerClient, never()).removeImageCmd(anyString()); } @Test diff --git a/src/test/java/de/tum/cit/aet/artemis/exercise/service/ParticipationTeamWebsocketServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/exercise/service/ParticipationTeamWebsocketServiceTest.java index 37bf559d6b73..8c74aff1d9ff 100644 --- a/src/test/java/de/tum/cit/aet/artemis/exercise/service/ParticipationTeamWebsocketServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/exercise/service/ParticipationTeamWebsocketServiceTest.java @@ -2,6 +2,7 @@ import static org.assertj.core.api.Assertions.assertThat; import static org.mockito.Mockito.RETURNS_MOCKS; +import static org.mockito.Mockito.after; import static org.mockito.Mockito.any; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.timeout; @@ -139,7 +140,7 @@ void testPatchModelingSubmissionWithWrongPrincipal() { // when we submit a patch, but with the wrong user ... participationTeamWebsocketService.patchModelingSubmission(participation.getId(), patch, getPrincipalMock("student2")); // the patch should not be broadcast. - verify(websocketMessagingService, timeout(2000).times(0)).sendMessage(websocketTopic(participation), List.of()); + verify(websocketMessagingService, after(1000).never()).sendMessage(websocketTopic(participation), List.of()); } @Test @@ -152,7 +153,7 @@ void testUpdateModelingSubmission() { // the submission should be handled by the service (i.e. saved), ... verify(modelingSubmissionService, timeout(2000).times(1)).handleModelingSubmission(any(), any(), any()); // but it should NOT be broadcast (sync is handled with patches only). - verify(websocketMessagingService, timeout(2000).times(0)).sendMessage(websocketTopic(participation), List.of()); + verify(websocketMessagingService, after(1000).never()).sendMessage(websocketTopic(participation), List.of()); } @Test @@ -163,9 +164,9 @@ void testUpdateModelingSubmissionWithWrongPrincipal() { // when we submit a new modeling submission with the wrong user ... participationTeamWebsocketService.updateModelingSubmission(participation.getId(), submission, getPrincipalMock("student2")); // the submission is NOT saved ... - verify(modelingSubmissionService, timeout(2000).times(0)).handleModelingSubmission(any(), any(), any()); + verify(modelingSubmissionService, after(1000).never()).handleModelingSubmission(any(), any(), any()); // it is also not broadcast. - verify(websocketMessagingService, timeout(2000).times(0)).sendMessage(websocketTopic(participation), List.of()); + verify(websocketMessagingService, after(1000).never()).sendMessage(websocketTopic(participation), List.of()); } @Test diff --git a/src/test/java/de/tum/cit/aet/artemis/iris/PyrisEventSystemIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/iris/PyrisEventSystemIntegrationTest.java index 95b620850ddd..44aa312cc340 100644 --- a/src/test/java/de/tum/cit/aet/artemis/iris/PyrisEventSystemIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/iris/PyrisEventSystemIntegrationTest.java @@ -5,6 +5,7 @@ import static org.awaitility.Awaitility.await; import static org.mockito.ArgumentMatchers.any; import static org.mockito.ArgumentMatchers.eq; +import static org.mockito.Mockito.never; import static org.mockito.Mockito.times; import static org.mockito.Mockito.verify; @@ -30,7 +31,6 @@ import de.tum.cit.aet.artemis.atlas.domain.competency.CompetencyJol; import de.tum.cit.aet.artemis.core.domain.Course; import de.tum.cit.aet.artemis.core.domain.User; -import de.tum.cit.aet.artemis.core.exception.AccessForbiddenAlertException; import de.tum.cit.aet.artemis.core.user.util.UserUtilService; import de.tum.cit.aet.artemis.exercise.domain.SubmissionType; import de.tum.cit.aet.artemis.exercise.participation.util.ParticipationFactory; @@ -40,13 +40,7 @@ import de.tum.cit.aet.artemis.iris.domain.settings.event.IrisEventType; import de.tum.cit.aet.artemis.iris.repository.IrisSettingsRepository; import de.tum.cit.aet.artemis.iris.service.pyris.PyrisEventProcessingException; -import de.tum.cit.aet.artemis.iris.service.pyris.PyrisEventService; -import de.tum.cit.aet.artemis.iris.service.pyris.PyrisJobService; -import de.tum.cit.aet.artemis.iris.service.pyris.PyrisStatusUpdateService; -import de.tum.cit.aet.artemis.iris.service.pyris.UnsupportedPyrisEventException; import de.tum.cit.aet.artemis.iris.service.pyris.event.NewResultEvent; -import de.tum.cit.aet.artemis.iris.service.pyris.event.PyrisEvent; -import de.tum.cit.aet.artemis.iris.service.session.IrisExerciseChatSessionService; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExercise; import de.tum.cit.aet.artemis.programming.domain.ProgrammingExerciseStudentParticipation; import de.tum.cit.aet.artemis.programming.domain.ProgrammingSubmission; @@ -59,12 +53,6 @@ class PyrisEventSystemIntegrationTest extends AbstractIrisIntegrationTest { private static final String TEST_PREFIX = "pyriseventsystemintegration"; - @Autowired - protected PyrisStatusUpdateService pyrisStatusUpdateService; - - @Autowired - protected PyrisJobService pyrisJobService; - @Autowired protected IrisSettingsRepository irisSettingsRepository; @@ -74,9 +62,6 @@ class PyrisEventSystemIntegrationTest extends AbstractIrisIntegrationTest { @Autowired private SubmissionTestRepository submissionRepository; - @Autowired - private PyrisEventService pyrisEventService; - @Autowired private ParticipationUtilService participationUtilService; @@ -103,6 +88,13 @@ class PyrisEventSystemIntegrationTest extends AbstractIrisIntegrationTest { void initTestCase() throws GitAPIException, IOException, URISyntaxException { userUtilService.addUsers(TEST_PREFIX, 2, 0, 0, 1); + var student1 = userUtilService.getUserByLogin(TEST_PREFIX + "student1"); + student1.setIrisAcceptedTimestamp(ZonedDateTime.now().minusDays(1)); + userTestRepository.save(student1); + var student2 = userUtilService.getUserByLogin(TEST_PREFIX + "student2"); + student2.setIrisAcceptedTimestamp(ZonedDateTime.now().minusDays(1)); + userTestRepository.save(student2); + course = programmingExerciseUtilService.addCourseWithOneProgrammingExercise(); competency = competencyUtilService.createCompetency(course); exercise = exerciseUtilService.getFirstExerciseWithType(course, ProgrammingExercise.class); @@ -194,7 +186,9 @@ void testShouldFireProgressStalledEvent() { }); pyrisEventService.trigger(new NewResultEvent(result)); - verify(irisExerciseChatSessionService, times(1)).onNewResult(eq(result)); + // Wrap the following code into await() to ensure that the pipeline is executed before the test finishes. + + await().atMost(2, TimeUnit.SECONDS).untilAsserted(() -> verify(irisExerciseChatSessionService, times(1)).onNewResult(eq(result))); await().atMost(2, TimeUnit.SECONDS).until(() -> pipelineDone.get()); @@ -214,12 +208,13 @@ void testShouldFireBuildFailedEvent() { }); pyrisEventService.trigger(new NewResultEvent(result)); - verify(irisExerciseChatSessionService, times(1)).onBuildFailure(eq(result)); + + await().atMost(2, TimeUnit.SECONDS).untilAsserted(() -> verify(irisExerciseChatSessionService, times(1)).onBuildFailure(eq(result))); await().atMost(2, TimeUnit.SECONDS).until(() -> pipelineDone.get()); - verify(pyrisPipelineService, times(1)).executeExerciseChatPipeline(eq("default"), eq(Optional.ofNullable((ProgrammingSubmission) result.getSubmission())), eq(exercise), - eq(irisSession), eq(Optional.of("build_failed"))); + await().atMost(2, TimeUnit.SECONDS).untilAsserted(() -> verify(pyrisPipelineService, times(1)).executeExerciseChatPipeline(eq("default"), + eq(Optional.ofNullable((ProgrammingSubmission) result.getSubmission())), eq(exercise), eq(irisSession), eq(Optional.of("build_failed")))); } @Test @@ -237,20 +232,6 @@ void testShouldFireJolEvent() { verify(irisCourseChatSessionService, times(1)).onJudgementOfLearningSet(any(CompetencyJol.class)); verify(pyrisPipelineService, times(1)).executeCourseChatPipeline(eq("default"), eq(irisSession), any(CompetencyJol.class)); - - } - - @Test - @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") - void testShouldThrowUnsupportedEventException() { - assertThatExceptionOfType(UnsupportedPyrisEventException.class).isThrownBy(() -> pyrisEventService.trigger(new PyrisEvent() { - - @Override - public void handleEvent(IrisExerciseChatSessionService service) { - // Do nothing - } - })).withMessageStartingWith("Unsupported event"); - } @Test @@ -264,7 +245,7 @@ void testShouldNotFireProgressStalledEventWithEventDisabled() { createSubmissionWithScore(studentParticipation, 40); createSubmissionWithScore(studentParticipation, 40); var result = createSubmissionWithScore(studentParticipation, 40); - assertThatExceptionOfType(AccessForbiddenAlertException.class).isThrownBy(() -> pyrisEventService.trigger(new NewResultEvent(result))); + verify(pyrisEventService, never()).trigger(new NewResultEvent(result)); } @Test @@ -278,7 +259,8 @@ void testShouldNotFireBuildFailedEventWhenEventSettingDisabled() { irisExerciseChatSessionService.createChatSessionForProgrammingExercise(exercise, userUtilService.getUserByLogin(TEST_PREFIX + "student1")); // Create a failing submission for the student. var result = createFailingSubmission(studentParticipation); - assertThatExceptionOfType(AccessForbiddenAlertException.class).isThrownBy(() -> pyrisEventService.trigger(new NewResultEvent(result))); + // very that the event is not fired + verify(pyrisEventService, never()).trigger(new NewResultEvent(result)); } @Test @@ -301,8 +283,10 @@ void testShouldShouldNotFireProgressStalledEventWithExistingSuccessfulSubmission pyrisEventService.trigger(new NewResultEvent(result)); await().atMost(2, TimeUnit.SECONDS); - verify(irisExerciseChatSessionService, times(2)).onNewResult(any(Result.class)); - verify(pyrisPipelineService, times(0)).executeExerciseChatPipeline(any(), any(), any(), any(), any()); + await().atMost(2, TimeUnit.SECONDS).untilAsserted(() -> { + verify(irisExerciseChatSessionService, times(2)).onNewResult(any(Result.class)); + verify(pyrisPipelineService, never()).executeExerciseChatPipeline(any(), any(), any(), any(), any()); + }); } @Test @@ -315,8 +299,8 @@ void testShouldNotFireProgressStalledEventWithLessThanThreeSubmissions() { pyrisEventService.trigger(new NewResultEvent(result)); - verify(irisExerciseChatSessionService, times(1)).onNewResult(any(Result.class)); - verify(pyrisPipelineService, times(0)).executeExerciseChatPipeline(any(), any(), any(), any(), any()); + verify(irisExerciseChatSessionService, never()).onNewResult(any(Result.class)); + verify(pyrisPipelineService, never()).executeExerciseChatPipeline(any(), any(), any(), any(), any()); } @Test @@ -330,8 +314,8 @@ void testShouldNotFireProgressStalledEventWithIncreasingScores() { pyrisEventService.trigger(new NewResultEvent(result)); - verify(irisExerciseChatSessionService, times(1)).onNewResult(any(Result.class)); - verify(pyrisPipelineService, times(0)).executeExerciseChatPipeline(any(), any(), any(), any(), any()); + verify(irisExerciseChatSessionService, never()).onNewResult(any(Result.class)); + verify(pyrisPipelineService, never()).executeExerciseChatPipeline(any(), any(), any(), any(), any()); } @Test diff --git a/src/test/java/de/tum/cit/aet/artemis/shared/base/AbstractSpringIntegrationLocalCILocalVCTest.java b/src/test/java/de/tum/cit/aet/artemis/shared/base/AbstractSpringIntegrationLocalCILocalVCTest.java index e05cf257fbf7..90e651dfbac8 100644 --- a/src/test/java/de/tum/cit/aet/artemis/shared/base/AbstractSpringIntegrationLocalCILocalVCTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/shared/base/AbstractSpringIntegrationLocalCILocalVCTest.java @@ -66,7 +66,6 @@ import de.tum.cit.aet.artemis.atlas.service.competency.CompetencyJolService; import de.tum.cit.aet.artemis.buildagent.BuildAgentConfiguration; import de.tum.cit.aet.artemis.buildagent.service.BuildAgentDockerService; -import de.tum.cit.aet.artemis.communication.service.notifications.GroupNotificationScheduleService; import de.tum.cit.aet.artemis.core.domain.Course; import de.tum.cit.aet.artemis.core.domain.User; import de.tum.cit.aet.artemis.core.service.ResourceLoaderService; @@ -74,6 +73,7 @@ import de.tum.cit.aet.artemis.core.user.util.UserUtilService; import de.tum.cit.aet.artemis.exam.service.ExamLiveEventsService; import de.tum.cit.aet.artemis.exercise.domain.Team; +import de.tum.cit.aet.artemis.iris.service.pyris.PyrisEventService; import de.tum.cit.aet.artemis.iris.service.pyris.PyrisPipelineService; import de.tum.cit.aet.artemis.iris.service.session.IrisCourseChatSessionService; import de.tum.cit.aet.artemis.iris.service.session.IrisExerciseChatSessionService; @@ -120,21 +120,6 @@ public abstract class AbstractSpringIntegrationLocalCILocalVCTest extends Abstra @Autowired protected LocalVCLocalCITestService localVCLocalCITestService; - @MockitoSpyBean - protected LdapUserService ldapUserService; - - @MockitoSpyBean - protected SpringSecurityLdapTemplate ldapTemplate; - - @MockitoSpyBean - protected LocalVCService versionControlService; - - @MockitoSpyBean - protected LocalCIService continuousIntegrationService; - - @MockitoSpyBean - protected BuildAgentConfiguration buildAgentConfiguration; - @Autowired protected ProgrammingExerciseTestRepository programmingExerciseRepository; @@ -159,6 +144,21 @@ public abstract class AbstractSpringIntegrationLocalCILocalVCTest extends Abstra @Autowired protected BuildJobTestRepository buildJobRepository; + @MockitoSpyBean + protected LdapUserService ldapUserService; + + @MockitoSpyBean + protected SpringSecurityLdapTemplate ldapTemplate; + + @MockitoSpyBean + protected LocalVCService versionControlService; + + @MockitoSpyBean + protected LocalCIService continuousIntegrationService; + + @MockitoSpyBean + protected BuildAgentConfiguration buildAgentConfiguration; + /** * This is the mock(DockerClient.class). * Subclasses can use this to dynamically mock methods of the DockerClient. @@ -174,9 +174,6 @@ public abstract class AbstractSpringIntegrationLocalCILocalVCTest extends Abstra @MockitoSpyBean protected ExamLiveEventsService examLiveEventsService; - @MockitoSpyBean - protected GroupNotificationScheduleService groupNotificationScheduleService; - @MockitoSpyBean protected IrisCourseChatSessionService irisCourseChatSessionService; @@ -189,6 +186,9 @@ public abstract class AbstractSpringIntegrationLocalCILocalVCTest extends Abstra @MockitoSpyBean protected IrisExerciseChatSessionService irisExerciseChatSessionService; + @MockitoSpyBean + protected PyrisEventService pyrisEventService; + @Value("${artemis.version-control.url}") protected URL localVCBaseUrl; From 5a6ea025764a8bf309a0a9947898b4b517e90d6a Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Sat, 4 Jan 2025 18:37:52 +0100 Subject: [PATCH 09/34] Development: Update ngx-graph to 9.0.1 (#9717) --- package-lock.json | 2323 ++++++++++++----- package.json | 12 +- patches/@swimlane+ngx-graph+9.0.1.patch | 26 + .../competency-graph.component.html | 2 +- .../competency-graph.component.ts | 1 - 5 files changed, 1657 insertions(+), 707 deletions(-) create mode 100644 patches/@swimlane+ngx-graph+9.0.1.patch diff --git a/package-lock.json b/package-lock.json index 3f71e3b6a204..011442994489 100644 --- a/package-lock.json +++ b/package-lock.json @@ -36,7 +36,7 @@ "@sentry/angular": "8.47.0", "@siemens/ngx-datatable": "22.4.1", "@swimlane/ngx-charts": "21.1.2", - "@swimlane/ngx-graph": "8.4.0", + "@swimlane/ngx-graph": "9.0.1", "@vscode/codicons": "0.0.36", "@vscode/markdown-it-katex": "1.1.1", "bootstrap": "5.3.3", @@ -126,6 +126,7 @@ "lint-staged": "15.2.11", "ng-mocks": "14.13.1", "ngxtension": "4.2.0", + "patch-package": "8.0.0", "prettier": "3.4.2", "rimraf": "6.0.1", "sass": "1.83.0", @@ -180,53 +181,6 @@ } } }, - "node_modules/@analogjs/vite-plugin-angular/node_modules/@ts-morph/common": { - "version": "0.22.0", - "resolved": "https://registry.npmjs.org/@ts-morph/common/-/common-0.22.0.tgz", - "integrity": "sha512-HqNBuV/oIlMKdkLshXd1zKBqNQCsuPEsgQOkfFQ/eUKjRlwndXW1AjN9LVkBEIukm00gGXSRmfkl0Wv5VXLnlw==", - "dev": true, - "license": "MIT", - "dependencies": { - "fast-glob": "^3.3.2", - "minimatch": "^9.0.3", - "mkdirp": "^3.0.1", - "path-browserify": "^1.0.1" - } - }, - "node_modules/@analogjs/vite-plugin-angular/node_modules/code-block-writer": { - "version": "12.0.0", - "resolved": "https://registry.npmjs.org/code-block-writer/-/code-block-writer-12.0.0.tgz", - "integrity": "sha512-q4dMFMlXtKR3XNBHyMHt/3pwYNA69EDk00lloMOaaUMKPUXBw6lpXtbu3MMVG6/uOihGnRDOlkyqsONEUj60+w==", - "dev": true, - "license": "MIT" - }, - "node_modules/@analogjs/vite-plugin-angular/node_modules/mkdirp": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-3.0.1.tgz", - "integrity": "sha512-+NsyUUAZDmo6YVHzL/stxSu3t9YS1iljliy3BSDrXJ/dkn1KYdmtZODGGjLcc9XLgVVpH4KshHB8XmZgMhaBXg==", - "dev": true, - "license": "MIT", - "bin": { - "mkdirp": "dist/cjs/src/bin.js" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/@analogjs/vite-plugin-angular/node_modules/ts-morph": { - "version": "21.0.1", - "resolved": "https://registry.npmjs.org/ts-morph/-/ts-morph-21.0.1.tgz", - "integrity": "sha512-dbDtVdEAncKctzrVZ+Nr7kHpHkv+0JDJb2MjjpBaj8bFeCkePU9rHfMklmhuLFnpeq/EJZk2IhStY6NzqgjOkg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@ts-morph/common": "~0.22.0", - "code-block-writer": "^12.0.0" - } - }, "node_modules/@angular-builders/common": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/@angular-builders/common/-/common-2.0.0.tgz", @@ -266,6 +220,19 @@ "jest": ">=29" } }, + "node_modules/@angular-builders/jest/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, "node_modules/@angular-builders/jest/node_modules/jest-preset-angular": { "version": "14.1.0", "resolved": "https://registry.npmjs.org/jest-preset-angular/-/jest-preset-angular-14.1.0.tgz", @@ -295,6 +262,28 @@ "typescript": ">=4.8" } }, + "node_modules/@angular-builders/jest/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@angular-builders/jest/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/@angular-devkit/architect": { "version": "0.1802.12", "resolved": "https://registry.npmjs.org/@angular-devkit/architect/-/architect-0.1802.12.tgz", @@ -440,6 +429,44 @@ } } }, + "node_modules/@angular-devkit/build-angular/node_modules/chokidar": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", + "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/@angular-devkit/build-angular/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, "node_modules/@angular-devkit/build-angular/node_modules/immutable": { "version": "4.3.7", "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.3.7.tgz", @@ -447,6 +474,32 @@ "dev": true, "license": "MIT" }, + "node_modules/@angular-devkit/build-angular/node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/@angular-devkit/build-angular/node_modules/readdirp/node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, "node_modules/@angular-devkit/build-angular/node_modules/sass": { "version": "1.77.6", "resolved": "https://registry.npmjs.org/sass/-/sass-1.77.6.tgz", @@ -724,6 +777,44 @@ } } }, + "node_modules/@angular/build/node_modules/chokidar": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", + "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/@angular/build/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, "node_modules/@angular/build/node_modules/immutable": { "version": "4.3.7", "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.3.7.tgz", @@ -731,6 +822,32 @@ "dev": true, "license": "MIT" }, + "node_modules/@angular/build/node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/@angular/build/node_modules/readdirp/node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, "node_modules/@angular/build/node_modules/sass": { "version": "1.77.6", "resolved": "https://registry.npmjs.org/sass/-/sass-1.77.6.tgz", @@ -864,34 +981,6 @@ "typescript": ">=5.4 <5.6" } }, - "node_modules/@angular/compiler-cli/node_modules/chokidar": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-4.0.1.tgz", - "integrity": "sha512-n8enUVCED/KVRQlab1hr3MVpcVMvxtZjmEa956u+4YijlmQED223XMSYj2tLuKvr4jcCTzNNMpQDUer72MMmzA==", - "license": "MIT", - "dependencies": { - "readdirp": "^4.0.1" - }, - "engines": { - "node": ">= 14.16.0" - }, - "funding": { - "url": "https://paulmillr.com/funding/" - } - }, - "node_modules/@angular/compiler-cli/node_modules/readdirp": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-4.0.2.tgz", - "integrity": "sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==", - "license": "MIT", - "engines": { - "node": ">= 14.16.0" - }, - "funding": { - "type": "individual", - "url": "https://paulmillr.com/funding/" - } - }, "node_modules/@angular/core": { "version": "18.2.13", "resolved": "https://registry.npmjs.org/@angular/core/-/core-18.2.13.tgz", @@ -1070,9 +1159,9 @@ } }, "node_modules/@babel/compat-data": { - "version": "7.26.2", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.26.2.tgz", - "integrity": "sha512-Z0WgzSEa+aUcdiJuCIqgujCshpMWgUpgOxXotrYPSA53hA3qopNaqcJpyr0hVb1FeWdnqFA35/fUtXgBK8srQg==", + "version": "7.26.3", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.26.3.tgz", + "integrity": "sha512-nHIxvKPniQXpmQLb0vhY3VaFb3S0YrTAwpOWJZh1wn3oJPjJk9Asva204PsBdmAE8vpzfHudT8DB0scYvy9q0g==", "license": "MIT", "engines": { "node": ">=6.9.0" @@ -1141,20 +1230,6 @@ "node": ">=6.9.0" } }, - "node_modules/@babel/helper-builder-binary-assignment-operator-visitor": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.25.9.tgz", - "integrity": "sha512-C47lC7LIDCnz0h4vai/tpNOI95tCd5ZT3iBt/DBH5lXKHZsyNQv18yf1wIIg2ntiQNgmAvA+DgZ82iW8Qdym8g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/traverse": "^7.25.9", - "@babel/types": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - } - }, "node_modules/@babel/helper-compilation-targets": { "version": "7.25.9", "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.25.9.tgz", @@ -1207,14 +1282,14 @@ } }, "node_modules/@babel/helper-create-regexp-features-plugin": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.25.9.tgz", - "integrity": "sha512-ORPNZ3h6ZRkOyAa/SaHU+XsLZr0UQzRwuDQ0cczIA17nAzZ+85G5cVkOJIj7QavLZGSe8QXUmNFxSZzjcZF9bw==", + "version": "7.26.3", + "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.26.3.tgz", + "integrity": "sha512-G7ZRb40uUgdKOQqPLjfD12ZmGA54PzqDFUv2BKImnC9QIfGhIHKvVML0oN8IUiDq4iRqpq74ABpvOaerfWdong==", "dev": true, "license": "MIT", "dependencies": { "@babel/helper-annotate-as-pure": "^7.25.9", - "regexpu-core": "^6.1.1", + "regexpu-core": "^6.2.0", "semver": "^6.3.1" }, "engines": { @@ -1369,20 +1444,6 @@ "@babel/core": "^7.0.0" } }, - "node_modules/@babel/helper-simple-access": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.25.9.tgz", - "integrity": "sha512-c6WHXuiaRsJTyHYLJV75t9IqsmTbItYfdj99PnzYGQZkYKvan5/2jKJ7gu31J3/BJ/A18grImSPModuyG/Eo0Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/traverse": "^7.25.9", - "@babel/types": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - } - }, "node_modules/@babel/helper-skip-transparent-expression-wrappers": { "version": "7.25.9", "resolved": "https://registry.npmjs.org/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.25.9.tgz", @@ -1466,12 +1527,12 @@ } }, "node_modules/@babel/parser": { - "version": "7.26.2", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.26.2.tgz", - "integrity": "sha512-DWMCZH9WA4Maitz2q21SRKHo9QXZxkDsbNZoVD62gusNtNBBqDg9i7uOhASfTfIGNzW+O+r7+jAlM8dwphcJKQ==", + "version": "7.26.3", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.26.3.tgz", + "integrity": "sha512-WJ/CvmY8Mea8iDXo6a7RK2wbmJITT5fN3BEkRuFlxVyNx8jOKIIhmC4fSkTcPcf8JyavbBwIe6OpiCOBXt/IcA==", "license": "MIT", "dependencies": { - "@babel/types": "^7.26.0" + "@babel/types": "^7.26.3" }, "bin": { "parser": "bin/babel-parser.js" @@ -2127,13 +2188,12 @@ } }, "node_modules/@babel/plugin-transform-exponentiation-operator": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.25.9.tgz", - "integrity": "sha512-KRhdhlVk2nObA5AYa7QMgTMTVJdfHprfpAk4DjZVtllqRg9qarilstTKEhpVjyt+Npi8ThRyiV8176Am3CodPA==", + "version": "7.26.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.26.3.tgz", + "integrity": "sha512-7CAHcQ58z2chuXPWblnn1K6rLDnDWieghSOEmqQsrBenH0P9InCUtOJYD89pvngljmZlJcz3fcmgYsXFNGa1ZQ==", "dev": true, "license": "MIT", "dependencies": { - "@babel/helper-builder-binary-assignment-operator-visitor": "^7.25.9", "@babel/helper-plugin-utils": "^7.25.9" }, "engines": { @@ -2276,15 +2336,14 @@ } }, "node_modules/@babel/plugin-transform-modules-commonjs": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.25.9.tgz", - "integrity": "sha512-dwh2Ol1jWwL2MgkCzUSOvfmKElqQcuswAZypBSUsScMXvgdT8Ekq5YA6TtqpTVWH+4903NmboMuH1o9i8Rxlyg==", + "version": "7.26.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.26.3.tgz", + "integrity": "sha512-MgR55l4q9KddUDITEzEFYn5ZsGDXMSsU9E+kh7fjRXTIC3RHqfCo8RPRbyReYJh44HQ/yomFkqbOFohXvDCiIQ==", "dev": true, "license": "MIT", "dependencies": { - "@babel/helper-module-transforms": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/helper-simple-access": "^7.25.9" + "@babel/helper-module-transforms": "^7.26.0", + "@babel/helper-plugin-utils": "^7.25.9" }, "engines": { "node": ">=6.9.0" @@ -2884,16 +2943,16 @@ } }, "node_modules/@babel/traverse": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.25.9.tgz", - "integrity": "sha512-ZCuvfwOwlz/bawvAuvcj8rrithP2/N55Tzz342AkTvq4qaWbGfmCk/tKhNaV2cthijKrPAA8SRJV5WWe7IBMJw==", + "version": "7.26.4", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.26.4.tgz", + "integrity": "sha512-fH+b7Y4p3yqvApJALCPJcwb0/XaOSgtK4pzV6WVjPR5GLFQBRI7pfoX2V2iM48NXvX07NUxxm1Vw98YjqTcU5w==", "license": "MIT", "dependencies": { - "@babel/code-frame": "^7.25.9", - "@babel/generator": "^7.25.9", - "@babel/parser": "^7.25.9", + "@babel/code-frame": "^7.26.2", + "@babel/generator": "^7.26.3", + "@babel/parser": "^7.26.3", "@babel/template": "^7.25.9", - "@babel/types": "^7.25.9", + "@babel/types": "^7.26.3", "debug": "^4.3.1", "globals": "^11.1.0" }, @@ -2902,13 +2961,13 @@ } }, "node_modules/@babel/traverse/node_modules/@babel/generator": { - "version": "7.26.2", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.26.2.tgz", - "integrity": "sha512-zevQbhbau95nkoxSq3f/DC/SC+EEOUZd3DYqfSkMhY2/wfSeaHV1Ew4vk8e+x8lja31IbyuUa2uQ3JONqKbysw==", + "version": "7.26.3", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.26.3.tgz", + "integrity": "sha512-6FF/urZvD0sTeO7k6/B15pMLC4CHUv1426lzr3N01aHJTl046uCAh9LXW/fzeXXjPNCJ6iABW5XaWOsIZB93aQ==", "license": "MIT", "dependencies": { - "@babel/parser": "^7.26.2", - "@babel/types": "^7.26.0", + "@babel/parser": "^7.26.3", + "@babel/types": "^7.26.3", "@jridgewell/gen-mapping": "^0.3.5", "@jridgewell/trace-mapping": "^0.3.25", "jsesc": "^3.0.2" @@ -2918,9 +2977,9 @@ } }, "node_modules/@babel/traverse/node_modules/jsesc": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.0.2.tgz", - "integrity": "sha512-xKqzzWXDttJuOcawBt4KnKHHIf5oQ/Cxax+0PWFG+DFDgHNAdi+TXECADI+RYiFUMmx8792xsMbbgXj4CwnP4g==", + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", + "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", "license": "MIT", "bin": { "jsesc": "bin/jsesc" @@ -2930,9 +2989,9 @@ } }, "node_modules/@babel/types": { - "version": "7.26.0", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.26.0.tgz", - "integrity": "sha512-Z/yiTPj+lDVnF7lWeKCIJzaIkI0vYO87dMpZ4bg4TDrFe4XXLFWL1TbXU27gBP3QccxV9mZICCrnjnYlJjXHOA==", + "version": "7.26.3", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.26.3.tgz", + "integrity": "sha512-vN5p+1kl59GVKMvTHt55NzzmYVxprfJD+ql7U9NFIfKCBkYE55LYtS+WtPlaYOyzydrKI8Nezd+aZextrd+FMA==", "license": "MIT", "dependencies": { "@babel/helper-string-parser": "^7.25.9", @@ -3505,13 +3564,13 @@ } }, "node_modules/@eslint/config-array": { - "version": "0.19.0", - "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.19.0.tgz", - "integrity": "sha512-zdHg2FPIFNKPdcHWtiNT+jEFCHYVplAXRDlQDyqy0zGx/q2parwh7brGJSiTxRk/TSMkbM//zt/f5CHgyTyaSQ==", + "version": "0.19.1", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.19.1.tgz", + "integrity": "sha512-fo6Mtm5mWyKjA/Chy1BYTdn5mGJoDNjC7C64ug20ADsRDGrA85bN3uK3MaKbeRkRuuIEAR5N33Jr1pbm411/PA==", "dev": true, "license": "Apache-2.0", "dependencies": { - "@eslint/object-schema": "^2.1.4", + "@eslint/object-schema": "^2.1.5", "debug": "^4.3.1", "minimatch": "^3.1.2" }, @@ -3544,11 +3603,14 @@ } }, "node_modules/@eslint/core": { - "version": "0.9.0", - "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.9.0.tgz", - "integrity": "sha512-7ATR9F0e4W85D/0w7cU0SNj7qkAexMG+bAHEZOjo9akvGuhHE2m7umzWzfnpa0XAg5Kxc1BWmtPMV67jJ+9VUg==", + "version": "0.9.1", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.9.1.tgz", + "integrity": "sha512-GuUdqkyyzQI5RMIWkHhvTWLCyLo1jNK3vzkSyaExH5kHPDHcuL2VOpHjmMY+y3+NC69qAKToBqldTBgYeLSr9Q==", "dev": true, "license": "Apache-2.0", + "dependencies": { + "@types/json-schema": "^7.0.15" + }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } @@ -3659,9 +3721,9 @@ } }, "node_modules/@eslint/object-schema": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.4.tgz", - "integrity": "sha512-BsWiH1yFGjXXS2yvrf5LyuoSIIbPrGUWob917o+BTKuZ7qJdxX8aJLRxs1fS9n6r7vESrq1OUqb68dANcFXuQQ==", + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.5.tgz", + "integrity": "sha512-o0bhxnL89h5Bae5T318nFoFzGy+YE5i/gGkoPAgkmTVdRKTiv3p8JHevPiPaMwoloKfEiiaHlawCqaZMqRm+XQ==", "dev": true, "license": "Apache-2.0", "engines": { @@ -3669,9 +3731,9 @@ } }, "node_modules/@eslint/plugin-kit": { - "version": "0.2.3", - "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.3.tgz", - "integrity": "sha512-2b/g5hRmpbb1o4GnTZax9N9m0FXzz9OV42ZzI4rDDMDuHUqigAiQCEWChBWCY4ztAGVRjoWT19v0yMmc5/L5kA==", + "version": "0.2.4", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.4.tgz", + "integrity": "sha512-zSkKow6H5Kdm0ZUQUB2kV5JIXqoG0+uH5YADhaEHswm664N9Db8dXSi0nMJpacpMf+MyyglF1vnZohpEg5yUtg==", "dev": true, "license": "Apache-2.0", "dependencies": { @@ -3922,9 +3984,9 @@ } }, "node_modules/@inquirer/figures": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/@inquirer/figures/-/figures-1.0.8.tgz", - "integrity": "sha512-tKd+jsmhq21AP1LhexC0pPwsCxEhGgAkg28byjJAd+xhmIs8LUX8JbUc3vBf3PhLxWiB5EvyBE5X7JSPAqMAqg==", + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/@inquirer/figures/-/figures-1.0.9.tgz", + "integrity": "sha512-BXvGj0ehzrngHTPTDqUoDT3NXL8U0RxUk2zJm2A66RhCEIWdtU1v6GuUqNAgArW4PQ9CinqIWyHdQgdwOj06zQ==", "dev": true, "license": "MIT", "engines": { @@ -4366,6 +4428,41 @@ } } }, + "node_modules/@jest/core/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@jest/core/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/core/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/@jest/core/node_modules/slash": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", @@ -4640,9 +4737,9 @@ } }, "node_modules/@jridgewell/gen-mapping": { - "version": "0.3.5", - "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz", - "integrity": "sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==", + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.8.tgz", + "integrity": "sha512-imAbBGkb+ebQyxKgzv5Hu2nmROxoDOXHh80evxdoXNOrvAnVx7zimzc1Oo5h9RlfV4vPXaE2iM5pOFbvOCClWA==", "license": "MIT", "dependencies": { "@jridgewell/set-array": "^1.2.1", @@ -4716,9 +4813,9 @@ } }, "node_modules/@jsonjoy.com/json-pack": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@jsonjoy.com/json-pack/-/json-pack-1.1.0.tgz", - "integrity": "sha512-zlQONA+msXPPwHWZMKFVS78ewFczIll5lXiVPwFPCZUsrOKdxc2AvxU1HoNBmMRhqDZUR9HkC3UOm+6pME6Xsg==", + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@jsonjoy.com/json-pack/-/json-pack-1.1.1.tgz", + "integrity": "sha512-osjeBqMJ2lb/j/M8NCPjs1ylqWIcTRTycIhVB5pt6LgzgeRSb0YRZ7j9RfA8wIUrsr/medIuhVyonXRZWLyfdw==", "dev": true, "license": "Apache-2.0", "dependencies": { @@ -5538,9 +5635,9 @@ } }, "node_modules/@nx/devkit": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/devkit/-/devkit-20.1.4.tgz", - "integrity": "sha512-Opz7eRPmpt3e4SGkbwZbE9Bg3MhKeivh1QTNCj4tQVAB4gucz0lW/F3mdtRDFdj6gUbqIc5rRrbO/DGlNaEzYw==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/devkit/-/devkit-20.3.0.tgz", + "integrity": "sha512-u9oRd2F33DLNWPbzpYGW7xuMEYUAOwO9DLP9nGYpxbZXy6Z4AdoKeqhN+KBTyg8+DyQGuKUSEXcWriDyLLgcHw==", "dev": true, "license": "MIT", "dependencies": { @@ -5594,9 +5691,9 @@ } }, "node_modules/@nx/nx-darwin-arm64": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/nx-darwin-arm64/-/nx-darwin-arm64-20.1.4.tgz", - "integrity": "sha512-afyDOZbIyHi6BgKk+Bb4RI1t8dZ6/oIbOY89z4mBPNNevZkbGqUfMwO2vjKnaOoThcjT93SEMJfCLGL8i857ww==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/nx-darwin-arm64/-/nx-darwin-arm64-20.3.0.tgz", + "integrity": "sha512-9PqSe1Sh7qNqA4GL0cZH0t3S0EZzb2Xn14XY9au7yf0+eoxyag1oETjjULrxLeUmSoXW2hDxzNtoqKFE9zF07Q==", "cpu": [ "arm64" ], @@ -5611,9 +5708,9 @@ } }, "node_modules/@nx/nx-darwin-x64": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/nx-darwin-x64/-/nx-darwin-x64-20.1.4.tgz", - "integrity": "sha512-aiYklAt95aX0EinepJRryMna8K53G52ngYOFuac1G8iLlguinJvg/YgSKCf7GOAzec8b7Hm7KauPjSJE/P3/iw==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/nx-darwin-x64/-/nx-darwin-x64-20.3.0.tgz", + "integrity": "sha512-gsGGhJVvi5QZVVTZie5sNMo1zOAU+A2edm6DGegObdFRLV41Ju/Yrm/gTaSp4yUtywd3UU4S/30C/nI2c55adA==", "cpu": [ "x64" ], @@ -5628,9 +5725,9 @@ } }, "node_modules/@nx/nx-freebsd-x64": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/nx-freebsd-x64/-/nx-freebsd-x64-20.1.4.tgz", - "integrity": "sha512-WUh4bsLK+e7wuN3lE3ZQUj+xQKdWU4P4RymutfLQQnPYiilCMtFwITcvDmazmOHFWI2vPhzSyYJRbOu+YMIR3A==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/nx-freebsd-x64/-/nx-freebsd-x64-20.3.0.tgz", + "integrity": "sha512-DiymYZBBu0upbiskdfn9KRyoXdyvKohezJiV3j4VkeRE8KR2p04NgwRQviDFbeD1cjWrDy9wk8y+G5PabLlqAA==", "cpu": [ "x64" ], @@ -5645,9 +5742,9 @@ } }, "node_modules/@nx/nx-linux-arm-gnueabihf": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/nx-linux-arm-gnueabihf/-/nx-linux-arm-gnueabihf-20.1.4.tgz", - "integrity": "sha512-9vPMw5s89v3od7aw3enTWjdMSCAmQ0tIA89Uz7xbbjB2kX2mAdihSzAKd9woi/cj+ROnY+ynNXzU9UjqhfxdBg==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/nx-linux-arm-gnueabihf/-/nx-linux-arm-gnueabihf-20.3.0.tgz", + "integrity": "sha512-Aksx66e8jmt/4rGJ/5z34SWXbPcYr9Ht52UonEeuCdQdoEvAOs7yBUbllYOjIcUsfZikEyZgvqfiQslsggSJdQ==", "cpu": [ "arm" ], @@ -5662,9 +5759,9 @@ } }, "node_modules/@nx/nx-linux-arm64-gnu": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/nx-linux-arm64-gnu/-/nx-linux-arm64-gnu-20.1.4.tgz", - "integrity": "sha512-JUE4l8utr9KmQSG9tO2Qw5R5i/bZ16s1+J5xnEar7UfcSOfOLqxGHS7HCBUZcfr46dmtv6KjIC83uHMs19AwDQ==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/nx-linux-arm64-gnu/-/nx-linux-arm64-gnu-20.3.0.tgz", + "integrity": "sha512-Y5wmYEwF1bl014Ps8QjagI911VbViQSFHSTVOCNSObdAzig9E5o6NOkoWe+doT1UZLrrInnlkrggQUsbtdKjOg==", "cpu": [ "arm64" ], @@ -5679,9 +5776,9 @@ } }, "node_modules/@nx/nx-linux-arm64-musl": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/nx-linux-arm64-musl/-/nx-linux-arm64-musl-20.1.4.tgz", - "integrity": "sha512-EaPUDqXvnPc/ure0x7N+5lRYvk5zqOQ3LzFOTRPWdqnFXejyTkGjZEYWbLFIJTFrvyEdpfaPTHyNmCHUrEz9TQ==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/nx-linux-arm64-musl/-/nx-linux-arm64-musl-20.3.0.tgz", + "integrity": "sha512-yGcIkmImyOMfPkQSYH2EVjPmFE0VkLcO71Bbkpr3RlJ1N/vjYxsGbdnqPiBb8Wshib/hmwpiMHf/yzQtKH0SQw==", "cpu": [ "arm64" ], @@ -5696,9 +5793,9 @@ } }, "node_modules/@nx/nx-linux-x64-gnu": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/nx-linux-x64-gnu/-/nx-linux-x64-gnu-20.1.4.tgz", - "integrity": "sha512-vaWV37ZayfyckVI/faWdQWIV9XQb06ZT8jHQnwgSd9tKbGz37vN30eYtgZlFL0P4bHfhjtmMXnLvADmfyO/KOw==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/nx-linux-x64-gnu/-/nx-linux-x64-gnu-20.3.0.tgz", + "integrity": "sha512-nkA2DLI+rpmiuiy7dyXP4l9s7dgHkQWDX7lG1XltiT41RzAReJF1h8qBE6XrsAYE1CtI76DRWVphnc93+iZr+A==", "cpu": [ "x64" ], @@ -5713,9 +5810,9 @@ } }, "node_modules/@nx/nx-linux-x64-musl": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/nx-linux-x64-musl/-/nx-linux-x64-musl-20.1.4.tgz", - "integrity": "sha512-wjq4Ea1oweBsIA9jq+jDT6BALxv/uac0aFykwoN23dOiwwSMFWMxbXUuBrxp0LjMFGV49S62kVDoRezukvkiZA==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/nx-linux-x64-musl/-/nx-linux-x64-musl-20.3.0.tgz", + "integrity": "sha512-sPMtTt9iTrCmFEIp9Qv27UX9PeL1aqKck2dz2TAFbXKVtF6+djOdTcNnTYw45KIP6izcUcOXXAq4G0QSQE7CLg==", "cpu": [ "x64" ], @@ -5730,9 +5827,9 @@ } }, "node_modules/@nx/nx-win32-arm64-msvc": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/nx-win32-arm64-msvc/-/nx-win32-arm64-msvc-20.1.4.tgz", - "integrity": "sha512-d9jN8biyEJh4Mjdc3RU1j/+WIOjrO9mCDxYuERXP2ELaNsOk0tJgcXE1xsa9AF88AHGpOkCOS2rxy61DKBtFKg==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/nx-win32-arm64-msvc/-/nx-win32-arm64-msvc-20.3.0.tgz", + "integrity": "sha512-ppfNa/8OfpWA9o26Pz3vArN4ulAC+Hx70/ghPRCP7ed1Mb3Z6yR2Ry9KfBRImbqajvuAExM0TePKMGq9LCdXmg==", "cpu": [ "arm64" ], @@ -5747,9 +5844,9 @@ } }, "node_modules/@nx/nx-win32-x64-msvc": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/@nx/nx-win32-x64-msvc/-/nx-win32-x64-msvc-20.1.4.tgz", - "integrity": "sha512-s3RwOkkWKzOflbTmc5MRc4EH2mk1AkJ/V8Gu3Qi2QncF9r1GrR7hDxROpu0MEoHfIhRG+d+n8OGX31nC9GZWUg==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/@nx/nx-win32-x64-msvc/-/nx-win32-x64-msvc-20.3.0.tgz", + "integrity": "sha512-8FOejZ4emtLSVn3pYWs4PIc3n4//qMbwMDPVxmPE8us3ir91Qh0bzr5zRj7Q8sEdSgvneXRXqtBp2grY2KMJsw==", "cpu": [ "x64" ], @@ -6722,105 +6819,32 @@ } }, "node_modules/@swimlane/ngx-graph": { - "version": "8.4.0", - "resolved": "https://registry.npmjs.org/@swimlane/ngx-graph/-/ngx-graph-8.4.0.tgz", - "integrity": "sha512-/A4FKyHU4ijnx8tGbTV5zBsMS+CH0/dA50Ee9oKPG8UCV3lJfUYr8+/UIwusrWC2YWXCgp/2A1VHFszCErrIaQ==", + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/@swimlane/ngx-graph/-/ngx-graph-9.0.1.tgz", + "integrity": "sha512-kymJLcAicmw30tbcnEIobIz5Cv3f7RZvoe2MJVbBjVhRxEqFVs5SONILwcdQqGIrqJC9E2PUM9VFwm9Qpgq8Rg==", "license": "MIT", "dependencies": { - "d3-dispatch": "^1.0.3", - "d3-ease": "^1.0.5", - "d3-force": "^1.1.0", - "d3-scale": "^3.2.3", - "d3-selection": "^1.2.0", - "d3-shape": "^1.2.0", - "d3-timer": "^1.0.7", + "d3-dispatch": "^3.0.1", + "d3-ease": "^3.0.1", + "d3-force": "^3.0.0", + "d3-scale": "^4.0.2", + "d3-selection": "^3.0.0", + "d3-shape": "^3.2.0", + "d3-timer": "^3.0.1", "d3-transition": "^3.0.1", - "dagre": "^0.8.4", - "transformation-matrix": "^1.15.3", - "tslib": "^2.0.0", - "webcola": "^3.3.8" + "dagre": "^0.8.5", + "transformation-matrix": "^2.16.1", + "tslib": "^2.3.1", + "webcola": "^3.4.0" }, "peerDependencies": { - "@angular/animations": ">=10.0.0", - "@angular/cdk": ">=10.0.0", - "@angular/common": ">=10.0.0", - "@angular/core": ">=10.0.0", - "rxjs": ">=6.0.0" - } - }, - "node_modules/@swimlane/ngx-graph/node_modules/d3-array": { - "version": "2.12.1", - "resolved": "https://registry.npmjs.org/d3-array/-/d3-array-2.12.1.tgz", - "integrity": "sha512-B0ErZK/66mHtEsR1TkPEEkwdy+WDesimkM5gpZr5Dsg54BiTA5RXtYW5qTLIAcekaS9xfZrzBLF/OAkB3Qn1YQ==", - "license": "BSD-3-Clause", - "dependencies": { - "internmap": "^1.0.0" - } - }, - "node_modules/@swimlane/ngx-graph/node_modules/d3-ease": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/d3-ease/-/d3-ease-1.0.7.tgz", - "integrity": "sha512-lx14ZPYkhNx0s/2HX5sLFUI3mbasHjSSpwO/KaaNACweVwxUruKyWVcb293wMv1RqTPZyZ8kSZ2NogUZNcLOFQ==", - "license": "BSD-3-Clause" - }, - "node_modules/@swimlane/ngx-graph/node_modules/d3-format": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/d3-format/-/d3-format-2.0.0.tgz", - "integrity": "sha512-Ab3S6XuE/Q+flY96HXT0jOXcM4EAClYFnRGY5zsjRGNy6qCYrQsMffs7cV5Q9xejb35zxW5hf/guKw34kvIKsA==", - "license": "BSD-3-Clause" - }, - "node_modules/@swimlane/ngx-graph/node_modules/d3-path": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/d3-path/-/d3-path-1.0.9.tgz", - "integrity": "sha512-VLaYcn81dtHVTjEHd8B+pbe9yHWpXKZUC87PzoFmsFrJqgFwDe/qxfp5MlfsfM1V5E/iVt0MmEbWQ7FVIXh/bg==", - "license": "BSD-3-Clause" - }, - "node_modules/@swimlane/ngx-graph/node_modules/d3-scale": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/d3-scale/-/d3-scale-3.3.0.tgz", - "integrity": "sha512-1JGp44NQCt5d1g+Yy+GeOnZP7xHo0ii8zsQp6PGzd+C1/dl0KGsp9A7Mxwp+1D1o4unbTTxVdU/ZOIEBoeZPbQ==", - "license": "BSD-3-Clause", - "dependencies": { - "d3-array": "^2.3.0", - "d3-format": "1 - 2", - "d3-interpolate": "1.2.0 - 2", - "d3-time": "^2.1.1", - "d3-time-format": "2 - 3" - } - }, - "node_modules/@swimlane/ngx-graph/node_modules/d3-shape": { - "version": "1.3.7", - "resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-1.3.7.tgz", - "integrity": "sha512-EUkvKjqPFUAZyOlhY5gzCxCeI0Aep04LwIRpsZ/mLFelJiUfnK56jo5JMDSE7yyP2kLSb6LtF+S5chMk7uqPqw==", - "license": "BSD-3-Clause", - "dependencies": { - "d3-path": "1" - } - }, - "node_modules/@swimlane/ngx-graph/node_modules/d3-time": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/d3-time/-/d3-time-2.1.1.tgz", - "integrity": "sha512-/eIQe/eR4kCQwq7yxi7z4c6qEXf2IYGcjoWB5OOQy4Tq9Uv39/947qlDcN2TLkiTzQWzvnsuYPB9TrWaNfipKQ==", - "license": "BSD-3-Clause", - "dependencies": { - "d3-array": "2" - } - }, - "node_modules/@swimlane/ngx-graph/node_modules/d3-time-format": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/d3-time-format/-/d3-time-format-3.0.0.tgz", - "integrity": "sha512-UXJh6EKsHBTjopVqZBhFysQcoXSv/5yLONZvkQ5Kk3qbwiUYkdX17Xa1PT6U1ZWXGGfB1ey5L8dKMlFq2DO0Ag==", - "license": "BSD-3-Clause", - "dependencies": { - "d3-time": "1 - 2" + "@angular/animations": "15.x || 16.x || 17.x || 18.x", + "@angular/cdk": "15.x || 16.x || 17.x || 18.x", + "@angular/common": "15.x || 16.x || 17.x || 18.x", + "@angular/core": "15.x || 16.x || 17.x || 18.x", + "rxjs": "7.x" } }, - "node_modules/@swimlane/ngx-graph/node_modules/internmap": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/internmap/-/internmap-1.0.1.tgz", - "integrity": "sha512-lDB5YccMydFBtasVtxnZ3MRBHuaoE8GKsppq+EchKL2U4nK/DmEpPHNH8MZe5HkMtpSiTSOZwfN0tzYjO/lJEw==", - "license": "ISC" - }, "node_modules/@testing-library/angular": { "version": "17.3.5", "resolved": "https://registry.npmjs.org/@testing-library/angular/-/angular-17.3.5.tgz", @@ -6859,20 +6883,6 @@ "node": ">=18" } }, - "node_modules/@testing-library/dom/node_modules/ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true, - "license": "MIT", - "peer": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, "node_modules/@testing-library/dom/node_modules/aria-query": { "version": "5.3.0", "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.0.tgz", @@ -6884,34 +6894,10 @@ "dequal": "^2.0.3" } }, - "node_modules/@testing-library/dom/node_modules/pretty-format": { - "version": "27.5.1", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", - "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "ansi-regex": "^5.0.1", - "ansi-styles": "^5.0.0", - "react-is": "^17.0.1" - }, - "engines": { - "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" - } - }, - "node_modules/@testing-library/dom/node_modules/react-is": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", - "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", - "dev": true, - "license": "MIT", - "peer": true - }, "node_modules/@ts-morph/common": { - "version": "0.23.0", - "resolved": "https://registry.npmjs.org/@ts-morph/common/-/common-0.23.0.tgz", - "integrity": "sha512-m7Lllj9n/S6sOkCkRftpM7L24uvmfXQFedlW/4hENcuJH1HHm9u5EgxZb9uVjQSCGrbBWBkOGgcTxNg36r6ywA==", + "version": "0.22.0", + "resolved": "https://registry.npmjs.org/@ts-morph/common/-/common-0.22.0.tgz", + "integrity": "sha512-HqNBuV/oIlMKdkLshXd1zKBqNQCsuPEsgQOkfFQ/eUKjRlwndXW1AjN9LVkBEIukm00gGXSRmfkl0Wv5VXLnlw==", "dev": true, "license": "MIT", "dependencies": { @@ -7152,9 +7138,9 @@ } }, "node_modules/@types/express-serve-static-core": { - "version": "5.0.2", - "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-5.0.2.tgz", - "integrity": "sha512-vluaspfvWEtE4vcSDlKRNer52DvOGrB2xv6diXy6UKyKW0lqZiWHGNApSyxOv+8DE5Z27IzVvE7hNkxg7EXIcg==", + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-5.0.3.tgz", + "integrity": "sha512-JEhMNwUJt7bw728CydvYzntD0XJeTmDnvwLlbfbAhE7Tbslm/ax6bdIiUwTgeVlZTsJQPwZwKpAkyDtIjsvx3g==", "dev": true, "license": "MIT", "dependencies": { @@ -7188,9 +7174,9 @@ } }, "node_modules/@types/hoist-non-react-statics": { - "version": "3.3.5", - "resolved": "https://registry.npmjs.org/@types/hoist-non-react-statics/-/hoist-non-react-statics-3.3.5.tgz", - "integrity": "sha512-SbcrWzkKBw2cdwRTwQAswfpB9g9LJWfjtUeW/jvNwbhC8cpmmNYVePa+ncbUe0rGTQ7G3Ff6mYUN2VMfLVr+Sg==", + "version": "3.3.6", + "resolved": "https://registry.npmjs.org/@types/hoist-non-react-statics/-/hoist-non-react-statics-3.3.6.tgz", + "integrity": "sha512-lPByRJUer/iN/xa4qpyL0qmL11DqNW81iU/IG1S3uvRUq4oKagz8VCxZjiWkumgt66YT3vOdDgZ0o32sGKtCEw==", "license": "MIT", "dependencies": { "@types/react": "*", @@ -7252,6 +7238,41 @@ "pretty-format": "^29.0.0" } }, + "node_modules/@types/jest/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@types/jest/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/@types/jsdom": { "version": "20.0.1", "resolved": "https://registry.npmjs.org/@types/jsdom/-/jsdom-20.0.1.tgz", @@ -7279,9 +7300,9 @@ "license": "MIT" }, "node_modules/@types/lodash": { - "version": "4.17.13", - "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.17.13.tgz", - "integrity": "sha512-lfx+dftrEZcdBPczf9d0Qv0x+j/rfNCMuC6OcfXmO8gkfeNAY88PgKUbvG56whcN23gc27yenwF6oJZXGFpYxg==", + "version": "4.17.14", + "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.17.14.tgz", + "integrity": "sha512-jsxagdikDiDBeIRaPYtArcT8my4tN1og7MtMRquFT3XNA6axxyHDRUemqDz/taRDdOUn0GnGHRCuff4q48sW9A==", "dev": true, "license": "MIT" }, @@ -7360,12 +7381,6 @@ "@types/node": "*" } }, - "node_modules/@types/prop-types": { - "version": "15.7.13", - "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.13.tgz", - "integrity": "sha512-hCZTSvwbzWGvhqxp/RqVqwU999pBf2vp7hzIjiYOsl8wqOmUxkQ6ddw1cV3l8811+kdUFus/q4d1Y3E3SyEifA==", - "license": "MIT" - }, "node_modules/@types/qs": { "version": "6.9.17", "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.17.tgz", @@ -7381,12 +7396,11 @@ "license": "MIT" }, "node_modules/@types/react": { - "version": "18.3.12", - "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.12.tgz", - "integrity": "sha512-D2wOSq/d6Agt28q7rSI3jhU7G6aiuzljDGZ2hTZHIkrTLUI+AF3WMeKkEZ9nN2fkBAlcktT6vcZjDFiIhMYEQw==", + "version": "19.0.2", + "resolved": "https://registry.npmjs.org/@types/react/-/react-19.0.2.tgz", + "integrity": "sha512-USU8ZI/xyKJwFTpjSVIrSeHBVAGagkHQKPNbxeWwql/vDmnTIBgx+TJnhFnj1NXgz8XfprU0egV2dROLGpsBEg==", "license": "MIT", "dependencies": { - "@types/prop-types": "*", "csstype": "^3.0.2" } }, @@ -7579,6 +7593,44 @@ "typescript": ">=4.8.4 <5.8.0" } }, + "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/types": { + "version": "8.18.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", + "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/utils": { + "version": "8.18.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.18.1.tgz", + "integrity": "sha512-8vikiIj2ebrC4WRdcAdDcmnu9Q/MXXwg+STf40BVfT8exDqBCUPdypvzcUPxEqRGKg9ALagZ0UWcYCtn+4W2iQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.4.0", + "@typescript-eslint/scope-manager": "8.18.1", + "@typescript-eslint/types": "8.18.1", + "@typescript-eslint/typescript-estree": "8.18.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" + } + }, "node_modules/@typescript-eslint/eslint-plugin/node_modules/ignore": { "version": "5.3.2", "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", @@ -7614,6 +7666,20 @@ "typescript": ">=4.8.4 <5.8.0" } }, + "node_modules/@typescript-eslint/parser/node_modules/@typescript-eslint/types": { + "version": "8.18.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", + "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, "node_modules/@typescript-eslint/scope-manager": { "version": "8.18.1", "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.18.1.tgz", @@ -7632,6 +7698,20 @@ "url": "https://opencollective.com/typescript-eslint" } }, + "node_modules/@typescript-eslint/scope-manager/node_modules/@typescript-eslint/types": { + "version": "8.18.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", + "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, "node_modules/@typescript-eslint/type-utils": { "version": "8.18.1", "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.18.1.tgz", @@ -7656,7 +7736,7 @@ "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/@typescript-eslint/types": { + "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/types": { "version": "8.18.1", "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", @@ -7670,21 +7750,17 @@ "url": "https://opencollective.com/typescript-eslint" } }, - "node_modules/@typescript-eslint/typescript-estree": { + "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/utils": { "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.18.1.tgz", - "integrity": "sha512-z8U21WI5txzl2XYOW7i9hJhxoKKNG1kcU4RzyNvKrdZDmbjkmLBo8bgeiOJmA06kizLI76/CCBAAGlTlEeUfyg==", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.18.1.tgz", + "integrity": "sha512-8vikiIj2ebrC4WRdcAdDcmnu9Q/MXXwg+STf40BVfT8exDqBCUPdypvzcUPxEqRGKg9ALagZ0UWcYCtn+4W2iQ==", "dev": true, "license": "MIT", "dependencies": { + "@eslint-community/eslint-utils": "^4.4.0", + "@typescript-eslint/scope-manager": "8.18.1", "@typescript-eslint/types": "8.18.1", - "@typescript-eslint/visitor-keys": "8.18.1", - "debug": "^4.3.4", - "fast-glob": "^3.3.2", - "is-glob": "^4.0.3", - "minimatch": "^9.0.4", - "semver": "^7.6.0", - "ts-api-utils": "^1.3.0" + "@typescript-eslint/typescript-estree": "8.18.1" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -7694,20 +7770,76 @@ "url": "https://opencollective.com/typescript-eslint" }, "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/@typescript-eslint/utils": { + "node_modules/@typescript-eslint/types": { + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.19.0.tgz", + "integrity": "sha512-8XQ4Ss7G9WX8oaYvD4OOLCjIQYgRQxO+qCiR2V2s2GxI9AUpo7riNwo6jDhKtTcaJjT8PY54j2Yb33kWtSJsmA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/typescript-estree": { "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.18.1.tgz", - "integrity": "sha512-8vikiIj2ebrC4WRdcAdDcmnu9Q/MXXwg+STf40BVfT8exDqBCUPdypvzcUPxEqRGKg9ALagZ0UWcYCtn+4W2iQ==", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.18.1.tgz", + "integrity": "sha512-z8U21WI5txzl2XYOW7i9hJhxoKKNG1kcU4RzyNvKrdZDmbjkmLBo8bgeiOJmA06kizLI76/CCBAAGlTlEeUfyg==", "dev": true, "license": "MIT", "dependencies": { - "@eslint-community/eslint-utils": "^4.4.0", - "@typescript-eslint/scope-manager": "8.18.1", "@typescript-eslint/types": "8.18.1", - "@typescript-eslint/typescript-estree": "8.18.1" + "@typescript-eslint/visitor-keys": "8.18.1", + "debug": "^4.3.4", + "fast-glob": "^3.3.2", + "is-glob": "^4.0.3", + "minimatch": "^9.0.4", + "semver": "^7.6.0", + "ts-api-utils": "^1.3.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/@typescript-eslint/types": { + "version": "8.18.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", + "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/utils": { + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.19.0.tgz", + "integrity": "sha512-PTBG+0oEMPH9jCZlfg07LCB2nYI0I317yyvXGfxnvGvw4SHIOuRnQ3kadyyXY6tGdChusIHIbM5zfIbp4M6tCg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.4.0", + "@typescript-eslint/scope-manager": "8.19.0", + "@typescript-eslint/types": "8.19.0", + "@typescript-eslint/typescript-estree": "8.19.0" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -7721,6 +7853,82 @@ "typescript": ">=4.8.4 <5.8.0" } }, + "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/scope-manager": { + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.19.0.tgz", + "integrity": "sha512-hkoJiKQS3GQ13TSMEiuNmSCvhz7ujyqD1x3ShbaETATHrck+9RaDdUbt+osXaUuns9OFwrDTTrjtwsU8gJyyRA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.19.0", + "@typescript-eslint/visitor-keys": "8.19.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/typescript-estree": { + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.19.0.tgz", + "integrity": "sha512-WW9PpDaLIFW9LCbucMSdYUuGeFUz1OkWYS/5fwZwTA+l2RwlWFdJvReQqMUMBw4yJWJOfqd7An9uwut2Oj8sLw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.19.0", + "@typescript-eslint/visitor-keys": "8.19.0", + "debug": "^4.3.4", + "fast-glob": "^3.3.2", + "is-glob": "^4.0.3", + "minimatch": "^9.0.4", + "semver": "^7.6.0", + "ts-api-utils": "^1.3.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/visitor-keys": { + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.19.0.tgz", + "integrity": "sha512-mCFtBbFBJDCNCWUl5y6sZSCHXw1DEFEk3c/M3nRK2a4XUB8StGFtmcEMizdjKuBzB6e/smJAAWYug3VrdLMr1w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.19.0", + "eslint-visitor-keys": "^4.2.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/utils/node_modules/eslint-visitor-keys": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", + "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, "node_modules/@typescript-eslint/visitor-keys": { "version": "8.18.1", "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.18.1.tgz", @@ -7739,6 +7947,20 @@ "url": "https://opencollective.com/typescript-eslint" } }, + "node_modules/@typescript-eslint/visitor-keys/node_modules/@typescript-eslint/types": { + "version": "8.18.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", + "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, "node_modules/@typescript-eslint/visitor-keys/node_modules/eslint-visitor-keys": { "version": "4.2.0", "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", @@ -8286,14 +8508,11 @@ } }, "node_modules/agent-base": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.1.tgz", - "integrity": "sha512-H0TSyFNDMomMNJQBn8wFV5YC/2eJ+VXECwOadZJT554xP6cODZHPX3H9QMQECxvrgiSOP1pHjy1sMWQVYJOUOA==", + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.3.tgz", + "integrity": "sha512-jRR5wdylq8CkOe6hei19GGZnxM6rBGwFl3Bg0YItGDimvjGtAvdZk4Pu6Cl4u4Igsws4a1fd1Vq3ezrhn4KmFw==", "dev": true, "license": "MIT", - "dependencies": { - "debug": "^4.3.4" - }, "engines": { "node": ">= 14" } @@ -8514,6 +8733,16 @@ "dev": true, "license": "MIT" }, + "node_modules/at-least-node": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/at-least-node/-/at-least-node-1.0.0.tgz", + "integrity": "sha512-+q/t7Ekv1EDY2l6Gda6LLiX14rU9TV20Wa3ofeQmwPFZbOMo9DXrLbOjFaaclkXKWidIaopwAObQDqwWtGUjqg==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">= 4.0.0" + } + }, "node_modules/autoprefixer": { "version": "10.4.20", "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.20.tgz", @@ -8553,9 +8782,9 @@ } }, "node_modules/axios": { - "version": "1.7.8", - "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.8.tgz", - "integrity": "sha512-Uu0wb7KNqK2t5K+YQyVCLM76prD5sRFjKHbJYCP1J7JFGEQ6nN7HWn9+04LAeiJ3ji54lgS/gZCH1oxyrf1SPw==", + "version": "1.7.9", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.9.tgz", + "integrity": "sha512-LhLcE7Hbiryz8oMDdDptSrWowmB4Bl6RCt6sIJKpRB4XtVf0iEgewX3au/pJqm+Py1kCASkb/FFKjxQaLtxJvw==", "dev": true, "license": "MIT", "dependencies": { @@ -8992,9 +9221,9 @@ } }, "node_modules/browserslist": { - "version": "4.24.2", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.24.2.tgz", - "integrity": "sha512-ZIc+Q62revdMcqC6aChtW4jz3My3klmCO1fEmINZY/8J3EpBg5/A/D0AKmBveUh6pgoeycoMkVMko84tuYS+Gg==", + "version": "4.24.3", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.24.3.tgz", + "integrity": "sha512-1CPmv8iobE2fyRMV97dAcMVegvvWKxmq94hkLiAkUGwKVTyDLw33K+ZxiFrREKmmps4rIw6grcCFCnTMSZ/YiA==", "funding": [ { "type": "opencollective", @@ -9011,9 +9240,9 @@ ], "license": "MIT", "dependencies": { - "caniuse-lite": "^1.0.30001669", - "electron-to-chromium": "^1.5.41", - "node-releases": "^2.0.18", + "caniuse-lite": "^1.0.30001688", + "electron-to-chromium": "^1.5.73", + "node-releases": "^2.0.19", "update-browserslist-db": "^1.1.1" }, "bin": { @@ -9167,17 +9396,47 @@ "license": "ISC" }, "node_modules/call-bind": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz", - "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==", + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.8.tgz", + "integrity": "sha512-oKlSFMcMwpUg2ednkhQ454wfWiU/ul3CkJe/PEHcTKuiX6RpbehUiFMXu13HalGZxfUwCQzZG747YXBn1im9ww==", "dev": true, "license": "MIT", "dependencies": { + "call-bind-apply-helpers": "^1.0.0", "es-define-property": "^1.0.0", - "es-errors": "^1.3.0", - "function-bind": "^1.1.2", "get-intrinsic": "^1.2.4", - "set-function-length": "^1.2.1" + "set-function-length": "^1.2.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/call-bind-apply-helpers": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.1.tgz", + "integrity": "sha512-BhYE+WDaywFg2TBWYNXAE+8B1ATnThNBqXHP5nQu0jWJdVvY2hvkpyB3qOmtmDePiS5/BDQ8wASEWGMWRG148g==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/call-bound": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.3.tgz", + "integrity": "sha512-YTd+6wGlNlPxSuri7Y6X8tY2dmm12UMH66RpKMhiX6rsk5wXXnYgbUcOt8kiS31/AjfoTOvCsE+w8nZQLQnzHA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "get-intrinsic": "^1.2.6" }, "engines": { "node": ">= 0.4" @@ -9216,9 +9475,9 @@ } }, "node_modules/caniuse-lite": { - "version": "1.0.30001685", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001685.tgz", - "integrity": "sha512-e/kJN1EMyHQzgcMEEgoo+YTCO1NGCmIYHk5Qk8jT6AazWemS5QFKJ5ShCJlH3GZrNIdZofcNCEwZqbMjjKzmnA==", + "version": "1.0.30001690", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001690.tgz", + "integrity": "sha512-5ExiE3qQN6oF8Clf8ifIDcMRCRE/dMGcETG/XGMD8/XiXm6HXQgQTh1yZYLXXpSOsEUlJm1Xr7kGULZTuGtP/w==", "funding": [ { "type": "opencollective", @@ -9297,41 +9556,18 @@ } }, "node_modules/chokidar": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", - "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", - "dev": true, + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-4.0.3.tgz", + "integrity": "sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA==", "license": "MIT", "dependencies": { - "anymatch": "~3.1.2", - "braces": "~3.0.2", - "glob-parent": "~5.1.2", - "is-binary-path": "~2.1.0", - "is-glob": "~4.0.1", - "normalize-path": "~3.0.0", - "readdirp": "~3.6.0" + "readdirp": "^4.0.1" }, "engines": { - "node": ">= 8.10.0" + "node": ">= 14.16.0" }, "funding": { "url": "https://paulmillr.com/funding/" - }, - "optionalDependencies": { - "fsevents": "~2.3.2" - } - }, - "node_modules/chokidar/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" } }, "node_modules/chownr": { @@ -9553,9 +9789,9 @@ } }, "node_modules/code-block-writer": { - "version": "13.0.3", - "resolved": "https://registry.npmjs.org/code-block-writer/-/code-block-writer-13.0.3.tgz", - "integrity": "sha512-Oofo0pq3IKnsFtuHqSF7TqBfr71aeyZDVJ0HpmqB7FBM2qEigL0iPONSCZSO9pE9dZTAxANe5XHG9Uy0YMv8cg==", + "version": "12.0.0", + "resolved": "https://registry.npmjs.org/code-block-writer/-/code-block-writer-12.0.0.tgz", + "integrity": "sha512-q4dMFMlXtKR3XNBHyMHt/3pwYNA69EDk00lloMOaaUMKPUXBw6lpXtbu3MMVG6/uOihGnRDOlkyqsONEUj60+w==", "dev": true, "license": "MIT" }, @@ -10092,12 +10328,6 @@ "node": ">=12" } }, - "node_modules/d3-collection": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/d3-collection/-/d3-collection-1.0.7.tgz", - "integrity": "sha512-ii0/r5f4sjKNTfh84Di+DpztYwqKhEyUlKoPrzUFfeSkWxjW49xU2QzO9qrPrNkpdI0XJkfzvmTu8V2Zylln6A==", - "license": "BSD-3-Clause" - }, "node_modules/d3-color": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/d3-color/-/d3-color-3.1.0.tgz", @@ -10108,10 +10338,13 @@ } }, "node_modules/d3-dispatch": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/d3-dispatch/-/d3-dispatch-1.0.6.tgz", - "integrity": "sha512-fVjoElzjhCEy+Hbn8KygnmMS7Or0a9sI2UzGwoB7cCtvI1XpVN9GpoYlnb3xt2YV66oXYb1fLJ8GMvP4hdU1RA==", - "license": "BSD-3-Clause" + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-dispatch/-/d3-dispatch-3.0.1.tgz", + "integrity": "sha512-rzUyPU/S7rwUflMyLc1ETDeBj0NRuHKKAcvukozwhshr6g6c5d8zh4c2gQjY2bZ0dXeGLWc1PF174P2tVvKhfg==", + "license": "ISC", + "engines": { + "node": ">=12" + } }, "node_modules/d3-drag": { "version": "3.0.0", @@ -10136,15 +10369,17 @@ } }, "node_modules/d3-force": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/d3-force/-/d3-force-1.2.1.tgz", - "integrity": "sha512-HHvehyaiUlVo5CxBJ0yF/xny4xoaxFxDnBXNvNcfW9adORGZfyNF1dj6DGLKyk4Yh3brP/1h3rnDzdIAwL08zg==", - "license": "BSD-3-Clause", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-force/-/d3-force-3.0.0.tgz", + "integrity": "sha512-zxV/SsA+U4yte8051P4ECydjD/S+qeYtnaIyAs9tgHCqfguma/aAQDjo85A9Z6EKhBirHRJHXIgJUlffT4wdLg==", + "license": "ISC", "dependencies": { - "d3-collection": "1", - "d3-dispatch": "1", - "d3-quadtree": "1", - "d3-timer": "1" + "d3-dispatch": "1 - 3", + "d3-quadtree": "1 - 3", + "d3-timer": "1 - 3" + }, + "engines": { + "node": ">=12" } }, "node_modules/d3-format": { @@ -10187,10 +10422,13 @@ } }, "node_modules/d3-quadtree": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/d3-quadtree/-/d3-quadtree-1.0.7.tgz", - "integrity": "sha512-RKPAeXnkC59IDGD0Wu5mANy0Q2V28L+fNe65pOCXVdVuTJS3WPKaJlFHer32Rbh9gIo9qMuJXio8ra4+YmIymA==", - "license": "BSD-3-Clause" + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-quadtree/-/d3-quadtree-3.0.1.tgz", + "integrity": "sha512-04xDrxQTDTCFwP5H6hRhsRcb9xxv2RzkcsygFzmkSIOJy3PeRJP7sNk3VRIbKXcog561P9oU0/rVH6vDROAgUw==", + "license": "ISC", + "engines": { + "node": ">=12" + } }, "node_modules/d3-sankey": { "version": "0.12.3", @@ -10294,10 +10532,13 @@ } }, "node_modules/d3-timer": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/d3-timer/-/d3-timer-1.0.10.tgz", - "integrity": "sha512-B1JDm0XDaQC+uvo4DT79H0XmBskgS3l6Ve+1SBCfxgmtIb1AVrPIoqd+nPSv+loMX8szQ0sVUhGngL7D5QPiXw==", - "license": "BSD-3-Clause" + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-timer/-/d3-timer-3.0.1.tgz", + "integrity": "sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA==", + "license": "ISC", + "engines": { + "node": ">=12" + } }, "node_modules/d3-transition": { "version": "3.0.1", @@ -10681,9 +10922,9 @@ } }, "node_modules/domutils": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.1.0.tgz", - "integrity": "sha512-H78uMmQtI2AhgDJjWeQmHwJJ2bLPD3GMmO7Zja/ZZh84wkm+4ut+IUnUdRa8uCGX88DiVx1j6FRe1XfxEgjEZA==", + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.2.1.tgz", + "integrity": "sha512-xWXmuRnN9OMP6ptPd2+H0cCbcYBULa5YDTbMm/2lvkWvNA3O4wcW+GvzooqBuNM8yy6pl3VIAeJTUUWUbfI5Fw==", "dev": true, "license": "BSD-2-Clause", "dependencies": { @@ -10696,9 +10937,9 @@ } }, "node_modules/dotenv": { - "version": "16.4.5", - "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.5.tgz", - "integrity": "sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==", + "version": "16.4.7", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.7.tgz", + "integrity": "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ==", "dev": true, "license": "BSD-2-Clause", "engines": { @@ -10724,6 +10965,21 @@ "url": "https://dotenvx.com" } }, + "node_modules/dunder-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", + "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "es-errors": "^1.3.0", + "gopd": "^1.2.0" + }, + "engines": { + "node": ">= 0.4" + } + }, "node_modules/eastasianwidth": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", @@ -10755,9 +11011,9 @@ } }, "node_modules/electron-to-chromium": { - "version": "1.5.68", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.68.tgz", - "integrity": "sha512-FgMdJlma0OzUYlbrtZ4AeXjKxKPk6KT8WOP8BjcqxWtlg8qyJQjRzPJzUtUn5GBg1oQ26hFs7HOOHJMYiJRnvQ==", + "version": "1.5.76", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.76.tgz", + "integrity": "sha512-CjVQyG7n7Sr+eBXE86HIulnL5N8xZY1sgmOPGuq/F0Rr0FJq63lg0kEtOIDfZBk44FnDLf6FUJ+dsJcuiUDdDQ==", "license": "ISC" }, "node_modules/emittery": { @@ -10851,9 +11107,9 @@ } }, "node_modules/enhanced-resolve": { - "version": "5.17.1", - "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.17.1.tgz", - "integrity": "sha512-LMHl3dXhTcfv8gM4kEzIUeTQ+7fpdA0l2tUf34BddXPkz2A5xJ5L/Pchd5BL6rdccM9QGvu0sWZzK1Z1t4wwyg==", + "version": "5.18.0", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.18.0.tgz", + "integrity": "sha512-0/r0MySGYG8YqlayBZ6MuCfECmHFdJ5qyPh8s8wa5Hnm6SaFLSK1VYCbj+NKp090Nm1caZhD+QTnmxO7esYGyQ==", "dev": true, "license": "MIT", "dependencies": { @@ -10944,14 +11200,11 @@ } }, "node_modules/es-define-property": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.0.tgz", - "integrity": "sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==", + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", + "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", "dev": true, "license": "MIT", - "dependencies": { - "get-intrinsic": "^1.2.4" - }, "engines": { "node": ">= 0.4" } @@ -10967,12 +11220,25 @@ } }, "node_modules/es-module-lexer": { - "version": "1.5.4", - "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.5.4.tgz", - "integrity": "sha512-MVNK56NiMrOwitFB7cqDwq0CQutbw+0BvLshJSse0MUNU+y1FC3bUS/AQg7oUng+/wKrrki7JfmwtVHkVfPLlw==", + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.6.0.tgz", + "integrity": "sha512-qqnD1yMU6tk/jnaMosogGySTZP8YtUgAffA9nMN+E/rjxcfRQ6IEk7IiozUjgxKoFHBGjTLnrHB/YC45r/59EQ==", "dev": true, "license": "MIT" }, + "node_modules/es-object-atoms": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.0.0.tgz", + "integrity": "sha512-MZ4iQ6JwHOBQjahnjwaC1ZtIBH+2ohjamzAO3oaHcXYup7qxjF2fixyH+Q71voWHeOkI2q/TnJao/KfXYIZWbw==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + } + }, "node_modules/esbuild": { "version": "0.23.0", "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.23.0.tgz", @@ -12024,9 +12290,9 @@ "license": "BSD-3-Clause" }, "node_modules/fastq": { - "version": "1.17.1", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.17.1.tgz", - "integrity": "sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==", + "version": "1.18.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.18.0.tgz", + "integrity": "sha512-QKHXPW0hD8g4UET03SdOdunzSouc9N4AuHdsX8XNcTsuz+yYFILVNIX4l9yHABMhiEI9Db0JTTIpu0wB+Y1QQw==", "license": "ISC", "dependencies": { "reusify": "^1.0.4" @@ -12191,6 +12457,16 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/find-yarn-workspace-root": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/find-yarn-workspace-root/-/find-yarn-workspace-root-2.0.0.tgz", + "integrity": "sha512-1IMnbjt4KzsQfnhnzNd8wUEgXZ44IzZaZmnLYx7D5FZlaHt2gW20Cri8Q+E/t5tIj4+epTBub+2Zxu/vNILzqQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "micromatch": "^4.0.2" + } + }, "node_modules/flat": { "version": "5.0.2", "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz", @@ -12400,6 +12676,22 @@ "dev": true, "license": "MIT" }, + "node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/fs-minipass": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-3.0.3.tgz", @@ -12477,17 +12769,22 @@ } }, "node_modules/get-intrinsic": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz", - "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==", + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.7.tgz", + "integrity": "sha512-VW6Pxhsrk0KAOqs3WEd0klDiF/+V7gQOpAvY1jVU/LHmaD/kQO4523aiJuikX/QAKYiW6x8Jh+RJej1almdtCA==", "dev": true, "license": "MIT", "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "es-define-property": "^1.0.1", "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", "function-bind": "^1.1.2", - "has-proto": "^1.0.1", - "has-symbols": "^1.0.3", - "hasown": "^2.0.0" + "get-proto": "^1.0.0", + "gopd": "^1.2.0", + "has-symbols": "^1.1.0", + "hasown": "^2.0.2", + "math-intrinsics": "^1.1.0" }, "engines": { "node": ">= 0.4" @@ -12506,6 +12803,20 @@ "node": ">=8.0.0" } }, + "node_modules/get-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", + "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", + "dev": true, + "license": "MIT", + "dependencies": { + "dunder-proto": "^1.0.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, "node_modules/get-stream": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", @@ -12650,14 +12961,11 @@ "license": "MIT" }, "node_modules/gopd": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.1.0.tgz", - "integrity": "sha512-FQoVQnqcdk4hVM4JN1eromaun4iuS34oStkdlLENLdpULsuQcTyXj8w7ayhuUfPwEYZ1ZOooOTT6fdA9Vmx/RA==", + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", + "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", "dev": true, "license": "MIT", - "dependencies": { - "get-intrinsic": "^1.2.4" - }, "engines": { "node": ">= 0.4" }, @@ -12727,22 +13035,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/has-proto": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.1.0.tgz", - "integrity": "sha512-QLdzI9IIO1Jg7f9GT1gXpPpXArAn6cS31R1eEZqz08Gc+uQ8/XiqHWt17Fiw+2p6oTTIq5GXEpQkAlA88YRl/Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/has-symbols": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", @@ -12770,9 +13062,9 @@ } }, "node_modules/highlight.js": { - "version": "11.10.0", - "resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-11.10.0.tgz", - "integrity": "sha512-SYVnVFswQER+zu1laSya563s+F8VDGt7o35d4utbamowvUNLLMovFqwCLSocpZTz3MgaSRA1IbqRWZv97dtErQ==", + "version": "11.11.1", + "resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-11.11.1.tgz", + "integrity": "sha512-Xwwo44whKBVCYoliBQwaPvtd/2tYFkRQtXDWj1nackaV2JPXx3L0+Jvd8/qCJ2p+ML0/XVkJ2q+Mr+UVdpJK5w==", "license": "BSD-3-Clause", "engines": { "node": ">=12.0.0" @@ -13320,9 +13612,9 @@ } }, "node_modules/is-core-module": { - "version": "2.15.1", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.15.1.tgz", - "integrity": "sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==", + "version": "2.16.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", + "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", "dev": true, "license": "MIT", "dependencies": { @@ -13546,9 +13838,10 @@ } }, "node_modules/isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", + "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "dev": true, "license": "MIT" }, "node_modules/isexe": { @@ -13799,6 +14092,41 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/jest-circus/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-circus/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-circus/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/jest-circus/node_modules/slash": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", @@ -13889,6 +14217,41 @@ } } }, + "node_modules/jest-config/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-config/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-config/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/jest-config/node_modules/slash": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", @@ -13922,6 +14285,41 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/jest-diff/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-diff/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-diff/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/jest-docblock": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-29.7.0.tgz", @@ -13952,6 +14350,41 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/jest-each/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-each/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-each/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/jest-environment-jsdom": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-environment-jsdom/-/jest-environment-jsdom-29.7.0.tgz", @@ -14103,6 +14536,41 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/jest-leak-detector/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-leak-detector/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-leak-detector/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/jest-matcher-utils": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-29.7.0.tgz", @@ -14119,6 +14587,41 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/jest-matcher-utils/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-matcher-utils/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-matcher-utils/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/jest-message-util": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-29.7.0.tgz", @@ -14140,6 +14643,41 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/jest-message-util/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-message-util/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-message-util/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/jest-message-util/node_modules/slash": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", @@ -14211,6 +14749,41 @@ "typescript": ">=4.8" } }, + "node_modules/jest-preset-angular/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-preset-angular/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-preset-angular/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/jest-regex-util": { "version": "29.6.3", "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-29.6.3.tgz", @@ -14396,6 +14969,41 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/jest-snapshot/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-snapshot/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-snapshot/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/jest-util": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-29.7.0.tgz", @@ -14445,6 +15053,19 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/jest-validate/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, "node_modules/jest-validate/node_modules/camelcase": { "version": "6.3.0", "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", @@ -14458,6 +15079,28 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/jest-validate/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-validate/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, "node_modules/jest-watcher": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-29.7.0.tgz", @@ -14511,9 +15154,9 @@ } }, "node_modules/jiti": { - "version": "1.21.6", - "resolved": "https://registry.npmjs.org/jiti/-/jiti-1.21.6.tgz", - "integrity": "sha512-2yTgeWTWzMWkHu6Jp9NKgePDaYHbntiwvYuuJLbbN9vl7DC9DvXKOB2BC3ZZ92D3cvV/aflH0osDfwpHepQ53w==", + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/jiti/-/jiti-1.21.7.tgz", + "integrity": "sha512-/imKNG4EbWNrVjoNC/1H5/9GFy+tqjGBHCaSsN+P2RnPqjsLmv6UD3Ej+Kj8nBWaRAwyk7kK5ZUc+OEatnTR3A==", "dev": true, "license": "MIT", "bin": { @@ -14629,6 +15272,26 @@ "dev": true, "license": "MIT" }, + "node_modules/json-stable-stringify": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify/-/json-stable-stringify-1.2.1.tgz", + "integrity": "sha512-Lp6HbbBgosLmJbjx0pBLbgvx68FaFU1sdkmBuckmhhJ88kL13OA51CDtR2yJB50eCNMH9wRqtQNNiAqQH4YXnA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.3", + "isarray": "^2.0.5", + "jsonify": "^0.0.1", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/json-stable-stringify-without-jsonify": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", @@ -14655,6 +15318,29 @@ "dev": true, "license": "MIT" }, + "node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/jsonify": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/jsonify/-/jsonify-0.0.1.tgz", + "integrity": "sha512-2/Ki0GcmuqSrgFyelQq9M05y7PS0mEwuIzrf3f1fPqkVDVRvZrPZtVSMHxdgo8Aq0sxAOb/cr2aqqA3LeWHVPg==", + "dev": true, + "license": "Public Domain", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/jsonparse": { "version": "1.3.1", "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz", @@ -14688,9 +15374,9 @@ } }, "node_modules/katex": { - "version": "0.16.11", - "resolved": "https://registry.npmjs.org/katex/-/katex-0.16.11.tgz", - "integrity": "sha512-RQrI8rlHY92OLf3rho/Ts8i/XvjgguEjOkO1BEXcU3N8BqPpSzBNwV/G0Ukr+P/l3ivvJUE/Fa/CwbS6HesGNQ==", + "version": "0.16.19", + "resolved": "https://registry.npmjs.org/katex/-/katex-0.16.19.tgz", + "integrity": "sha512-3IA6DYVhxhBabjSLTNO9S4+OliA3Qvb8pBQXMfC4WxXJgLwZgnfDl0BmB4z6nBMdznBsZ+CGM8DrGZ5hcguDZg==", "funding": [ "https://opencollective.com/katex", "https://github.com/sponsors/katex" @@ -14723,6 +15409,16 @@ "node": ">=0.10.0" } }, + "node_modules/klaw-sync": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/klaw-sync/-/klaw-sync-6.0.0.tgz", + "integrity": "sha512-nIeuVSzdCCs6TDPTqI8w1Yre34sSq7AkZ4B3sfOBbI2CgVSB4Du4aLQijFU2+lhAFCwt9+42Hel6lQNIv6AntQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "graceful-fs": "^4.1.11" + } + }, "node_modules/kleur": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", @@ -15644,6 +16340,16 @@ "integrity": "sha512-6qE4B9deFBIa9YSpOc9O0Sgc43zTeVYbgDT5veRKSlB2+ZuHNoVVxA1L/ckMUayV9Ay9y7Z/SZCLcGteW9i7bg==", "license": "ISC" }, + "node_modules/math-intrinsics": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", + "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, "node_modules/mdurl": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/mdurl/-/mdurl-2.0.0.tgz", @@ -15661,9 +16367,9 @@ } }, "node_modules/memfs": { - "version": "4.14.1", - "resolved": "https://registry.npmjs.org/memfs/-/memfs-4.14.1.tgz", - "integrity": "sha512-Fq5CMEth+2iprLJ5mNizRcWuiwRZYjNkUD0zKk224jZunE9CRacTRDK8QLALbMBlNX2y3nY6lKZbesCwDwacig==", + "version": "4.15.3", + "resolved": "https://registry.npmjs.org/memfs/-/memfs-4.15.3.tgz", + "integrity": "sha512-vR/g1SgqvKJgAyYla+06G4p/EOcEmwhYuVb1yc1ixcKf8o/sh7Zngv63957ZSNd1xrZJoinmNyDf2LzuP8WJXw==", "dev": true, "license": "Apache-2.0", "dependencies": { @@ -16295,6 +17001,53 @@ } } }, + "node_modules/ngxtension/node_modules/@ts-morph/common": { + "version": "0.23.0", + "resolved": "https://registry.npmjs.org/@ts-morph/common/-/common-0.23.0.tgz", + "integrity": "sha512-m7Lllj9n/S6sOkCkRftpM7L24uvmfXQFedlW/4hENcuJH1HHm9u5EgxZb9uVjQSCGrbBWBkOGgcTxNg36r6ywA==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-glob": "^3.3.2", + "minimatch": "^9.0.3", + "mkdirp": "^3.0.1", + "path-browserify": "^1.0.1" + } + }, + "node_modules/ngxtension/node_modules/code-block-writer": { + "version": "13.0.3", + "resolved": "https://registry.npmjs.org/code-block-writer/-/code-block-writer-13.0.3.tgz", + "integrity": "sha512-Oofo0pq3IKnsFtuHqSF7TqBfr71aeyZDVJ0HpmqB7FBM2qEigL0iPONSCZSO9pE9dZTAxANe5XHG9Uy0YMv8cg==", + "dev": true, + "license": "MIT" + }, + "node_modules/ngxtension/node_modules/mkdirp": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-3.0.1.tgz", + "integrity": "sha512-+NsyUUAZDmo6YVHzL/stxSu3t9YS1iljliy3BSDrXJ/dkn1KYdmtZODGGjLcc9XLgVVpH4KshHB8XmZgMhaBXg==", + "dev": true, + "license": "MIT", + "bin": { + "mkdirp": "dist/cjs/src/bin.js" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/ngxtension/node_modules/ts-morph": { + "version": "22.0.0", + "resolved": "https://registry.npmjs.org/ts-morph/-/ts-morph-22.0.0.tgz", + "integrity": "sha512-M9MqFGZREyeb5fTl6gNHKZLqBQA0TjA1lea+CR48R8EBTDuWrNqW6ccC5QvjNR4s6wDumD3LTCjOFSp9iwlzaw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@ts-morph/common": "~0.23.0", + "code-block-writer": "^13.0.1" + } + }, "node_modules/nice-napi": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/nice-napi/-/nice-napi-1.0.2.tgz", @@ -16450,9 +17203,9 @@ "license": "MIT" }, "node_modules/node-releases": { - "version": "2.0.18", - "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.18.tgz", - "integrity": "sha512-d9VeXT4SJ7ZeOqGX6R5EM022wpL+eWPooLI+5UpWn2jCT1aosUQEhQP214x33Wkwx3JQMvIm+tIoVOdodFS40g==", + "version": "2.0.19", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.19.tgz", + "integrity": "sha512-xxOWJsBKtzAq7DY0J+DTzuz58K8e7sJbdgwkbMWQe8UYB6ekmsQ45q0M/tJDsGaZmbC+l7n57UV8Hl5tHxO9uw==", "license": "MIT" }, "node_modules/nopt": { @@ -16641,9 +17394,9 @@ "license": "MIT" }, "node_modules/nx": { - "version": "20.1.4", - "resolved": "https://registry.npmjs.org/nx/-/nx-20.1.4.tgz", - "integrity": "sha512-hyvGYxTzBkPxSXAB2tuqdv9TpVde5xOdGalsIdhF7j7PI3nwPpqtc3y28YTgRgpxtOE1Y6BfDNkXMO1SW0xu2w==", + "version": "20.3.0", + "resolved": "https://registry.npmjs.org/nx/-/nx-20.3.0.tgz", + "integrity": "sha512-Nzi4k7tV22zwO2iBLk+pHxorLEWPJpPrVCACtz0SQ63j/LiAgfhoqruJO+VU+V+E9qdyPsvmqIL/Iaf/GRQlqA==", "dev": true, "hasInstallScript": true, "license": "MIT", @@ -16672,12 +17425,14 @@ "npm-run-path": "^4.0.1", "open": "^8.4.0", "ora": "5.3.0", + "resolve.exports": "2.0.3", "semver": "^7.5.3", "string-width": "^4.2.3", "tar-stream": "~2.2.0", "tmp": "~0.2.1", "tsconfig-paths": "^4.1.2", "tslib": "^2.3.0", + "yaml": "^2.6.0", "yargs": "^17.6.2", "yargs-parser": "21.1.1" }, @@ -16686,16 +17441,16 @@ "nx-cloud": "bin/nx-cloud.js" }, "optionalDependencies": { - "@nx/nx-darwin-arm64": "20.1.4", - "@nx/nx-darwin-x64": "20.1.4", - "@nx/nx-freebsd-x64": "20.1.4", - "@nx/nx-linux-arm-gnueabihf": "20.1.4", - "@nx/nx-linux-arm64-gnu": "20.1.4", - "@nx/nx-linux-arm64-musl": "20.1.4", - "@nx/nx-linux-x64-gnu": "20.1.4", - "@nx/nx-linux-x64-musl": "20.1.4", - "@nx/nx-win32-arm64-msvc": "20.1.4", - "@nx/nx-win32-x64-msvc": "20.1.4" + "@nx/nx-darwin-arm64": "20.3.0", + "@nx/nx-darwin-x64": "20.3.0", + "@nx/nx-freebsd-x64": "20.3.0", + "@nx/nx-linux-arm-gnueabihf": "20.3.0", + "@nx/nx-linux-arm64-gnu": "20.3.0", + "@nx/nx-linux-arm64-musl": "20.3.0", + "@nx/nx-linux-x64-gnu": "20.3.0", + "@nx/nx-linux-x64-musl": "20.3.0", + "@nx/nx-win32-arm64-msvc": "20.3.0", + "@nx/nx-win32-x64-msvc": "20.3.0" }, "peerDependencies": { "@swc-node/register": "^1.8.0", @@ -16921,6 +17676,16 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/object-keys": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, "node_modules/obuf": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/obuf/-/obuf-1.1.2.tgz", @@ -17333,6 +18098,93 @@ "node": ">= 0.8" } }, + "node_modules/patch-package": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/patch-package/-/patch-package-8.0.0.tgz", + "integrity": "sha512-da8BVIhzjtgScwDJ2TtKsfT5JFWz1hYoBl9rUQ1f38MC2HwnEIkK8VN3dKMKcP7P7bvvgzNDbfNHtx3MsQb5vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@yarnpkg/lockfile": "^1.1.0", + "chalk": "^4.1.2", + "ci-info": "^3.7.0", + "cross-spawn": "^7.0.3", + "find-yarn-workspace-root": "^2.0.0", + "fs-extra": "^9.0.0", + "json-stable-stringify": "^1.0.2", + "klaw-sync": "^6.0.0", + "minimist": "^1.2.6", + "open": "^7.4.2", + "rimraf": "^2.6.3", + "semver": "^7.5.3", + "slash": "^2.0.0", + "tmp": "^0.0.33", + "yaml": "^2.2.2" + }, + "bin": { + "patch-package": "index.js" + }, + "engines": { + "node": ">=14", + "npm": ">5" + } + }, + "node_modules/patch-package/node_modules/is-docker": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.2.1.tgz", + "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==", + "dev": true, + "license": "MIT", + "bin": { + "is-docker": "cli.js" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/patch-package/node_modules/is-wsl": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz", + "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-docker": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/patch-package/node_modules/open": { + "version": "7.4.2", + "resolved": "https://registry.npmjs.org/open/-/open-7.4.2.tgz", + "integrity": "sha512-MVHddDVweXZF3awtlAS+6pgKLlm/JgxZ90+/NBurBoQctVOOB/zDdVjcyPzQ+0laDGbsWgrRkflI65sQeOgT9Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-docker": "^2.0.0", + "is-wsl": "^2.1.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/patch-package/node_modules/slash": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-2.0.0.tgz", + "integrity": "sha512-ZYKh3Wh2z1PpEXWr0MpSBZ0V6mZHAQfYevttO11c51CaWjGTaadiKZ+wVt1PbMlDV5qhMFslpZCemhwOK7C89A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, "node_modules/path-browserify": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/path-browserify/-/path-browserify-1.0.1.tgz", @@ -17726,9 +18578,9 @@ } }, "node_modules/postcss-modules-local-by-default": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-4.1.0.tgz", - "integrity": "sha512-rm0bdSv4jC3BDma3s9H19ZddW0aHX6EoqwDYU2IfZhRN+53QrufTRo2IdkAbRqLx4R2IYbZnbjKKxg4VN5oU9Q==", + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-4.2.0.tgz", + "integrity": "sha512-5kcJm/zk+GJDSfw+V/42fJ5fhjL5YbFDl8nVdXkJPLLW+Vf9mTD5Xe0wqIaDnLuL2U6cDNpTr+UQ+v2HWIBhzw==", "dev": true, "license": "MIT", "dependencies": { @@ -17835,18 +18687,19 @@ } }, "node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", + "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", "dev": true, "license": "MIT", + "peer": true, "dependencies": { - "@jest/schemas": "^29.6.3", + "ansi-regex": "^5.0.1", "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" + "react-is": "^17.0.1" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" } }, "node_modules/pretty-format/node_modules/ansi-styles": { @@ -17855,6 +18708,7 @@ "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", "dev": true, "license": "MIT", + "peer": true, "engines": { "node": ">=10" }, @@ -17862,6 +18716,14 @@ "url": "https://github.com/chalk/ansi-styles?sponsor=1" } }, + "node_modules/pretty-format/node_modules/react-is": { + "version": "17.0.2", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", + "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", + "dev": true, + "license": "MIT", + "peer": true + }, "node_modules/proc-log": { "version": "4.2.0", "resolved": "https://registry.npmjs.org/proc-log/-/proc-log-4.2.0.tgz", @@ -18140,10 +19002,11 @@ } }, "node_modules/react-is": { - "version": "18.3.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", - "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", - "license": "MIT" + "version": "19.0.0", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-19.0.0.tgz", + "integrity": "sha512-H91OHcwjZsbq3ClIDHMzBShc1rotbfACdWENsmEf0IFvZ3FgGPtdHMcsv45bQ1hAbgdfiA8SnxTKfDS+x/8m2g==", + "license": "MIT", + "peer": true }, "node_modules/react-redux": { "version": "8.1.3", @@ -18184,6 +19047,12 @@ } } }, + "node_modules/react-redux/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "license": "MIT" + }, "node_modules/react-tooltip": { "version": "4.5.1", "resolved": "https://registry.npmjs.org/react-tooltip/-/react-tooltip-4.5.1.tgz", @@ -18234,30 +19103,23 @@ "util-deprecate": "~1.0.1" } }, - "node_modules/readdirp": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", - "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", - "dev": true, - "license": "MIT", - "dependencies": { - "picomatch": "^2.2.1" - }, - "engines": { - "node": ">=8.10.0" - } + "node_modules/readable-stream/node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", + "license": "MIT" }, - "node_modules/readdirp/node_modules/picomatch": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", - "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", - "dev": true, + "node_modules/readdirp": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-4.0.2.tgz", + "integrity": "sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==", "license": "MIT", "engines": { - "node": ">=8.6" + "node": ">= 14.16.0" }, "funding": { - "url": "https://github.com/sponsors/jonschlinkert" + "type": "individual", + "url": "https://paulmillr.com/funding/" } }, "node_modules/redux": { @@ -18867,36 +19729,6 @@ } } }, - "node_modules/sass/node_modules/chokidar": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-4.0.1.tgz", - "integrity": "sha512-n8enUVCED/KVRQlab1hr3MVpcVMvxtZjmEa956u+4YijlmQED223XMSYj2tLuKvr4jcCTzNNMpQDUer72MMmzA==", - "dev": true, - "license": "MIT", - "dependencies": { - "readdirp": "^4.0.1" - }, - "engines": { - "node": ">= 14.16.0" - }, - "funding": { - "url": "https://paulmillr.com/funding/" - } - }, - "node_modules/sass/node_modules/readdirp": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-4.0.2.tgz", - "integrity": "sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 14.16.0" - }, - "funding": { - "type": "individual", - "url": "https://paulmillr.com/funding/" - } - }, "node_modules/sax": { "version": "1.4.1", "resolved": "https://registry.npmjs.org/sax/-/sax-1.4.1.tgz", @@ -18928,9 +19760,9 @@ } }, "node_modules/schema-utils": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.2.0.tgz", - "integrity": "sha512-L0jRsrPpjdckP3oPug3/VxNKt2trR8TcabrM6FOAAlvC/9Phcmm+cuAgTlxBqdBR1WJx7Naj9WHw+aOmheSVbw==", + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.3.0.tgz", + "integrity": "sha512-Gf9qqc58SpCA/xdziiHz35F4GNIWYWZrEshUc/G/r5BnLph6xpKuLeoJoQuj5WfBIx/eQLf+hmVPYHaxJu7V2g==", "dev": true, "license": "MIT", "dependencies": { @@ -18940,7 +19772,7 @@ "ajv-keywords": "^5.1.0" }, "engines": { - "node": ">= 12.13.0" + "node": ">= 10.13.0" }, "funding": { "type": "opencollective", @@ -19249,16 +20081,73 @@ } }, "node_modules/side-channel": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.6.tgz", - "integrity": "sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==", + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", + "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", "dev": true, "license": "MIT", "dependencies": { - "call-bind": "^1.0.7", "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.4", - "object-inspect": "^1.13.1" + "object-inspect": "^1.13.3", + "side-channel-list": "^1.0.0", + "side-channel-map": "^1.0.1", + "side-channel-weakmap": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-list": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", + "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-map": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", + "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-weakmap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", + "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3", + "side-channel-map": "^1.0.1" }, "engines": { "node": ">= 0.4" @@ -19438,13 +20327,13 @@ } }, "node_modules/socks-proxy-agent": { - "version": "8.0.4", - "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-8.0.4.tgz", - "integrity": "sha512-GNAq/eg8Udq2x0eNiFkr9gRg5bA7PXEWagQdeRX4cPSG+X/8V38v637gim9bjFptMk1QWsCTr0ttrJEiXbNnRw==", + "version": "8.0.5", + "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-8.0.5.tgz", + "integrity": "sha512-HehCEsotFqbPW9sJ8WVYB6UbmIMv7kUUORIF2Nncq4VQvBfNBLibW9YZR5dlYCSUhwcD628pRllm7n+E+YTzJw==", "dev": true, "license": "MIT", "dependencies": { - "agent-base": "^7.1.1", + "agent-base": "^7.1.2", "debug": "^4.3.4", "socks": "^2.8.3" }, @@ -20089,17 +20978,17 @@ } }, "node_modules/terser-webpack-plugin": { - "version": "5.3.10", - "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-5.3.10.tgz", - "integrity": "sha512-BKFPWlPDndPs+NGGCr1U59t0XScL5317Y0UReNrHaw9/FwhPENlq6bfgs+4yPfyP51vqC1bQ4rp1EfXW5ZSH9w==", + "version": "5.3.11", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-5.3.11.tgz", + "integrity": "sha512-RVCsMfuD0+cTt3EwX8hSl2Ks56EbFHWmhluwcqoPKtBnfjiT6olaq7PRIRfhyU8nnC2MrnDrBLfrD/RGE+cVXQ==", "dev": true, "license": "MIT", "dependencies": { - "@jridgewell/trace-mapping": "^0.3.20", + "@jridgewell/trace-mapping": "^0.3.25", "jest-worker": "^27.4.5", - "schema-utils": "^3.1.1", - "serialize-javascript": "^6.0.1", - "terser": "^5.26.0" + "schema-utils": "^4.3.0", + "serialize-javascript": "^6.0.2", + "terser": "^5.31.1" }, "engines": { "node": ">= 10.13.0" @@ -20123,33 +21012,6 @@ } } }, - "node_modules/terser-webpack-plugin/node_modules/ajv": { - "version": "6.12.6", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", - "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", - "dev": true, - "license": "MIT", - "dependencies": { - "fast-deep-equal": "^3.1.1", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.4.1", - "uri-js": "^4.2.2" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/epoberezkin" - } - }, - "node_modules/terser-webpack-plugin/node_modules/ajv-keywords": { - "version": "3.5.2", - "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.5.2.tgz", - "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", - "dev": true, - "license": "MIT", - "peerDependencies": { - "ajv": "^6.9.1" - } - }, "node_modules/terser-webpack-plugin/node_modules/jest-worker": { "version": "27.5.1", "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.5.1.tgz", @@ -20165,32 +21027,6 @@ "node": ">= 10.13.0" } }, - "node_modules/terser-webpack-plugin/node_modules/json-schema-traverse": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", - "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true, - "license": "MIT" - }, - "node_modules/terser-webpack-plugin/node_modules/schema-utils": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.3.0.tgz", - "integrity": "sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/json-schema": "^7.0.8", - "ajv": "^6.12.5", - "ajv-keywords": "^3.5.2" - }, - "engines": { - "node": ">= 10.13.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/webpack" - } - }, "node_modules/terser-webpack-plugin/node_modules/supports-color": { "version": "8.1.1", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", @@ -20287,9 +21123,9 @@ "license": "MIT" }, "node_modules/tinyexec": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.1.tgz", - "integrity": "sha512-WiCJLEECkO18gwqIp6+hJg0//p23HXp4S+gGtAKu3mI2F2/sXC4FvHvXvB0zJVVaTPhx1/tOwdbRsa1sOBIKqQ==", + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.2.tgz", + "integrity": "sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==", "dev": true, "license": "MIT" }, @@ -20334,22 +21170,22 @@ } }, "node_modules/tldts": { - "version": "6.1.65", - "resolved": "https://registry.npmjs.org/tldts/-/tldts-6.1.65.tgz", - "integrity": "sha512-xU9gLTfAGsADQ2PcWee6Hg8RFAv0DnjMGVJmDnUmI8a9+nYmapMQix4afwrdaCtT+AqP4MaxEzu7cCrYmBPbzQ==", + "version": "6.1.70", + "resolved": "https://registry.npmjs.org/tldts/-/tldts-6.1.70.tgz", + "integrity": "sha512-/W1YVgYVJd9ZDjey5NXadNh0mJXkiUMUue9Zebd0vpdo1sU+H4zFFTaJ1RKD4N6KFoHfcXy6l+Vu7bh+bdWCzA==", "dev": true, "license": "MIT", "dependencies": { - "tldts-core": "^6.1.65" + "tldts-core": "^6.1.70" }, "bin": { "tldts": "bin/cli.js" } }, "node_modules/tldts-core": { - "version": "6.1.65", - "resolved": "https://registry.npmjs.org/tldts-core/-/tldts-core-6.1.65.tgz", - "integrity": "sha512-Uq5t0N0Oj4nQSbU8wFN1YYENvMthvwU13MQrMJRspYCGLSAZjAfoBOJki5IQpnBM/WFskxxC/gIOTwaedmHaSg==", + "version": "6.1.70", + "resolved": "https://registry.npmjs.org/tldts-core/-/tldts-core-6.1.70.tgz", + "integrity": "sha512-RNnIXDB1FD4T9cpQRErEqw6ZpjLlGdMOitdV+0xtbsnwr4YFka1zpc7D4KD+aAn8oSG5JyFrdasZTE04qDE9Yg==", "dev": true, "license": "MIT" }, @@ -20422,10 +21258,13 @@ } }, "node_modules/transformation-matrix": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/transformation-matrix/-/transformation-matrix-1.15.3.tgz", - "integrity": "sha512-ThJH58GNFKhCw3gIoOtwf3tNwuYjbyEeiGdeq4mNMYWdJctnI896KUqn6PVt7jmNVepqa1bcKQtnMB1HtjsDMA==", - "license": "MIT" + "version": "2.16.1", + "resolved": "https://registry.npmjs.org/transformation-matrix/-/transformation-matrix-2.16.1.tgz", + "integrity": "sha512-tdtC3wxVEuzU7X/ydL131Q3JU5cPMEn37oqVLITjRDSDsnSHVFzW2JiCLfZLIQEgWzZHdSy3J6bZzvKEN24jGA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/chrvadala" + } }, "node_modules/tree-dump": { "version": "1.0.2", @@ -20540,14 +21379,14 @@ } }, "node_modules/ts-morph": { - "version": "22.0.0", - "resolved": "https://registry.npmjs.org/ts-morph/-/ts-morph-22.0.0.tgz", - "integrity": "sha512-M9MqFGZREyeb5fTl6gNHKZLqBQA0TjA1lea+CR48R8EBTDuWrNqW6ccC5QvjNR4s6wDumD3LTCjOFSp9iwlzaw==", + "version": "21.0.1", + "resolved": "https://registry.npmjs.org/ts-morph/-/ts-morph-21.0.1.tgz", + "integrity": "sha512-dbDtVdEAncKctzrVZ+Nr7kHpHkv+0JDJb2MjjpBaj8bFeCkePU9rHfMklmhuLFnpeq/EJZk2IhStY6NzqgjOkg==", "dev": true, "license": "MIT", "dependencies": { - "@ts-morph/common": "~0.23.0", - "code-block-writer": "^13.0.1" + "@ts-morph/common": "~0.22.0", + "code-block-writer": "^12.0.0" } }, "node_modules/ts-node": { @@ -21147,6 +21986,16 @@ "url": "https://opencollective.com/unified" } }, + "node_modules/universalify": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz", + "integrity": "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 10.0.0" + } + }, "node_modules/unpipe": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", @@ -21208,12 +22057,12 @@ } }, "node_modules/use-sync-external-store": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.2.2.tgz", - "integrity": "sha512-PElTlVMwpblvbNqQ82d2n6RjStvdSoNe9FG28kNfz3WiXilJm4DdNkEzRhCZuIDwY8U08WVihhGR5iRqAwfDiw==", + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.4.0.tgz", + "integrity": "sha512-9WXSPC5fMv61vaupRkCKCxsPxBocVnwakBEkMIHHpkTTg6icbJtg6jzgtLDm4bl3cSHAca52rYWih0k4K3PfHw==", "license": "MIT", "peerDependencies": { - "react": "^16.8.0 || ^17.0.0 || ^18.0.0" + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "node_modules/util-deprecate": { @@ -22040,6 +22889,12 @@ "d3-timer": "^1.0.5" } }, + "node_modules/webcola/node_modules/d3-dispatch": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/d3-dispatch/-/d3-dispatch-1.0.6.tgz", + "integrity": "sha512-fVjoElzjhCEy+Hbn8KygnmMS7Or0a9sI2UzGwoB7cCtvI1XpVN9GpoYlnb3xt2YV66oXYb1fLJ8GMvP4hdU1RA==", + "license": "BSD-3-Clause" + }, "node_modules/webcola/node_modules/d3-drag": { "version": "1.2.5", "resolved": "https://registry.npmjs.org/d3-drag/-/d3-drag-1.2.5.tgz", @@ -22056,6 +22911,12 @@ "integrity": "sha512-VLaYcn81dtHVTjEHd8B+pbe9yHWpXKZUC87PzoFmsFrJqgFwDe/qxfp5MlfsfM1V5E/iVt0MmEbWQ7FVIXh/bg==", "license": "BSD-3-Clause" }, + "node_modules/webcola/node_modules/d3-selection": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/d3-selection/-/d3-selection-1.4.2.tgz", + "integrity": "sha512-SJ0BqYihzOjDnnlfyeHT0e30k0K1+5sR3d5fNueCNeuhZTnGw4M4o8mqJchSwgKMXCNFo+e2VTChiSJ0vYtXkg==", + "license": "BSD-3-Clause" + }, "node_modules/webcola/node_modules/d3-shape": { "version": "1.3.7", "resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-1.3.7.tgz", @@ -22065,6 +22926,12 @@ "d3-path": "1" } }, + "node_modules/webcola/node_modules/d3-timer": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/d3-timer/-/d3-timer-1.0.10.tgz", + "integrity": "sha512-B1JDm0XDaQC+uvo4DT79H0XmBskgS3l6Ve+1SBCfxgmtIb1AVrPIoqd+nPSv+loMX8szQ0sVUhGngL7D5QPiXw==", + "license": "BSD-3-Clause" + }, "node_modules/webidl-conversions": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz", @@ -22209,6 +23076,44 @@ } } }, + "node_modules/webpack-dev-server/node_modules/chokidar": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", + "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/webpack-dev-server/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, "node_modules/webpack-dev-server/node_modules/http-proxy-middleware": { "version": "2.0.7", "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-2.0.7.tgz", @@ -22234,6 +23139,32 @@ } } }, + "node_modules/webpack-dev-server/node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/webpack-dev-server/node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, "node_modules/webpack-merge": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-6.0.1.tgz", @@ -22431,9 +23362,9 @@ } }, "node_modules/whatwg-url": { - "version": "14.0.0", - "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-14.0.0.tgz", - "integrity": "sha512-1lfMEm2IEr7RIV+f4lUNPOqfFL+pO+Xw3fJSqmjX9AbXcXcYOkCe1P6+9VBZB6n94af16NfZf+sSk0JCBZC9aw==", + "version": "14.1.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-14.1.0.tgz", + "integrity": "sha512-jlf/foYIKywAt3x/XWKZ/3rz8OSJPiWktjmk891alJUEjiVxKX9LEO92qH3hv4aJ0mN3MWPvGMCy8jQi95xK4w==", "dev": true, "license": "MIT", "dependencies": { diff --git a/package.json b/package.json index 1248fce42f27..1d55723acd2d 100644 --- a/package.json +++ b/package.json @@ -39,7 +39,7 @@ "@sentry/angular": "8.47.0", "@siemens/ngx-datatable": "22.4.1", "@swimlane/ngx-charts": "21.1.2", - "@swimlane/ngx-graph": "8.4.0", + "@swimlane/ngx-graph": "9.0.1", "@vscode/codicons": "0.0.36", "@vscode/markdown-it-katex": "1.1.1", "bootstrap": "5.3.3", @@ -83,13 +83,6 @@ "zone.js": "0.14.10" }, "overrides": { - "@swimlane/ngx-graph": { - "d3-brush": "^3.0.0", - "d3-color": "^3.1.0", - "d3-interpolate": "^3.0.1", - "d3-selection": "^3.0.0", - "d3-transition": "^3.0.1" - }, "@typescript-eslint/utils": { "eslint": "^9.17.0" }, @@ -160,6 +153,7 @@ "lint-staged": "15.2.11", "ng-mocks": "14.13.1", "ngxtension": "4.2.0", + "patch-package": "8.0.0", "prettier": "3.4.2", "rimraf": "6.0.1", "sass": "1.83.0", @@ -180,7 +174,7 @@ "compile:ts:tests": "npm run prebuild && tsc --project tsconfig.spec.json", "lint": "eslint", "lint:fix": "eslint --fix", - "postinstall": "husky", + "postinstall": "husky && patch-package", "prebuild": "node prebuild.mjs", "prepare": "husky", "prettier:check": "prettier --check \"src/{main/webapp,test}/**/*.{json,ts,js,css,scss,html}\"", diff --git a/patches/@swimlane+ngx-graph+9.0.1.patch b/patches/@swimlane+ngx-graph+9.0.1.patch new file mode 100644 index 000000000000..2843ab5eb28d --- /dev/null +++ b/patches/@swimlane+ngx-graph+9.0.1.patch @@ -0,0 +1,26 @@ +diff --git a/node_modules/@swimlane/ngx-graph/esm2022/lib/graph/graph.component.mjs b/node_modules/@swimlane/ngx-graph/esm2022/lib/graph/graph.component.mjs +index fadcfa7..6a6ab52 100644 +--- a/node_modules/@swimlane/ngx-graph/esm2022/lib/graph/graph.component.mjs ++++ b/node_modules/@swimlane/ngx-graph/esm2022/lib/graph/graph.component.mjs +@@ -197,7 +197,7 @@ export class GraphComponent { + if (layoutSettings) { + this.setLayoutSettings(this.layoutSettings); + } +- if (this.layout && this.nodes.length && this.links.length) { ++ if (this.layout && this.nodes && this.links) { + this.update(); + } + } +diff --git a/node_modules/@swimlane/ngx-graph/fesm2022/swimlane-ngx-graph.mjs b/node_modules/@swimlane/ngx-graph/fesm2022/swimlane-ngx-graph.mjs +index 0e851a7..c8dce60 100644 +--- a/node_modules/@swimlane/ngx-graph/fesm2022/swimlane-ngx-graph.mjs ++++ b/node_modules/@swimlane/ngx-graph/fesm2022/swimlane-ngx-graph.mjs +@@ -1478,7 +1478,7 @@ class GraphComponent { + if (layoutSettings) { + this.setLayoutSettings(this.layoutSettings); + } +- if (this.layout && this.nodes.length && this.links.length) { ++ if (this.layout && this.nodes && this.links) { + this.update(); + } + } diff --git a/src/main/webapp/app/course/learning-paths/components/competency-graph/competency-graph.component.html b/src/main/webapp/app/course/learning-paths/components/competency-graph/competency-graph.component.html index 99b808753650..630f93e343e8 100644 --- a/src/main/webapp/app/course/learning-paths/components/competency-graph/competency-graph.component.html +++ b/src/main/webapp/app/course/learning-paths/components/competency-graph/competency-graph.component.html @@ -1,6 +1,6 @@

(); readonly center$ = new Subject(); readonly zoomToFit$ = new Subject(); From fca49a3202646ce2abc3f7b86618cddacc36ad44 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maximilian=20S=C3=B6lch?= Date: Sat, 4 Jan 2025 19:44:31 +0100 Subject: [PATCH 10/34] Assessment: Fix display of exercise title in assessment dashboard (#10103) --- .../tutor/exercise-assessment-dashboard.component.html | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/main/webapp/app/exercises/shared/dashboards/tutor/exercise-assessment-dashboard.component.html b/src/main/webapp/app/exercises/shared/dashboards/tutor/exercise-assessment-dashboard.component.html index 2ac0bbee41af..b1b2e9295b76 100644 --- a/src/main/webapp/app/exercises/shared/dashboards/tutor/exercise-assessment-dashboard.component.html +++ b/src/main/webapp/app/exercises/shared/dashboards/tutor/exercise-assessment-dashboard.component.html @@ -1,12 +1,12 @@ @if (exercise) { - + @if (!isTestRun) { {{ 'artemisApp.exerciseAssessmentDashboard.pageHeader' | artemisTranslate }} {{ exercise?.title }} } @else { {{ 'artemisApp.exerciseAssessmentDashboard.testRunPageHeader' | artemisTranslate }} {{ exercise?.title }} } - + } From fed02c36cfe451e770969edf01046902da8763fb Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Sun, 5 Jan 2025 09:34:13 +0100 Subject: [PATCH 11/34] Development: Bump version to 7.8.2 (bugfix update) --- README.md | 2 +- build.gradle | 2 +- package-lock.json | 4 ++-- package.json | 2 +- 4 files changed, 5 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index 14970443fb21..0c776b37df5c 100644 --- a/README.md +++ b/README.md @@ -193,7 +193,7 @@ Refer to [Using JHipster in production](http://www.jhipster.tech/production) for The following command can automate the deployment to a server. The example shows the deployment to the main Artemis test server (which runs a virtual machine): ```shell -./artemis-server-cli deploy username@artemis-test0.artemis.in.tum.de -w build/libs/Artemis-7.8.1.war +./artemis-server-cli deploy username@artemis-test0.artemis.in.tum.de -w build/libs/Artemis-7.8.2.war ``` ## Architecture diff --git a/build.gradle b/build.gradle index 310c4d32143e..fe1c9e2f9d8d 100644 --- a/build.gradle +++ b/build.gradle @@ -25,7 +25,7 @@ plugins { } group = "de.tum.cit.aet.artemis" -version = "7.8.1" +version = "7.8.2" description = "Interactive Learning with Individual Feedback" java { diff --git a/package-lock.json b/package-lock.json index 011442994489..3ce295642bc3 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "artemis", - "version": "7.8.1", + "version": "7.8.2", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "artemis", - "version": "7.8.1", + "version": "7.8.2", "hasInstallScript": true, "license": "MIT", "dependencies": { diff --git a/package.json b/package.json index 1d55723acd2d..1663425bd516 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "artemis", - "version": "7.8.1", + "version": "7.8.2", "description": "Interactive Learning with Individual Feedback", "private": true, "license": "MIT", From 0d71dfd0dee8b06c3eed0c2afec85f27d0e2eb87 Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Sun, 5 Jan 2025 09:44:41 +0100 Subject: [PATCH 12/34] Development: Update client dependencies --- package-lock.json | 570 +++++++--------------------------------------- package.json | 20 +- 2 files changed, 91 insertions(+), 499 deletions(-) diff --git a/package-lock.json b/package-lock.json index 3ce295642bc3..8ea357ad52d4 100644 --- a/package-lock.json +++ b/package-lock.json @@ -46,7 +46,7 @@ "dayjs": "1.11.13", "diff-match-patch-typescript": "1.1.0", "dompurify": "3.2.3", - "emoji-js": "3.8.0", + "emoji-js": "3.8.1", "export-to-csv": "1.4.0", "fast-json-patch": "3.1.1", "franc-min": "6.2.0", @@ -65,7 +65,7 @@ "ngx-webstorage": "18.0.0", "papaparse": "5.4.1", "pdf-lib": "1.17.1", - "pdfjs-dist": "4.9.155", + "pdfjs-dist": "4.10.38", "rxjs": "7.8.1", "simple-statistics": "7.8.7", "smoothscroll-polyfill": "0.4.4", @@ -80,7 +80,7 @@ "zone.js": "0.14.10" }, "devDependencies": { - "@analogjs/vite-plugin-angular": "1.10.3", + "@analogjs/vite-plugin-angular": "1.11.0", "@angular-builders/jest": "18.0.0", "@angular-devkit/build-angular": "18.2.12", "@angular-eslint/builder": "18.4.1", @@ -100,21 +100,21 @@ "@types/jest": "29.5.14", "@types/lodash-es": "4.17.12", "@types/markdown-it": "14.1.2", - "@types/node": "22.10.2", + "@types/node": "22.10.5", "@types/papaparse": "5.3.15", "@types/smoothscroll-polyfill": "0.3.4", "@types/sockjs-client": "1.5.4", "@types/turndown": "5.0.5", "@types/uuid": "10.0.0", - "@typescript-eslint/eslint-plugin": "8.18.1", - "@typescript-eslint/parser": "8.18.1", + "@typescript-eslint/eslint-plugin": "8.19.0", + "@typescript-eslint/parser": "8.19.0", "eslint": "9.17.0", "eslint-config-prettier": "9.1.0", "eslint-plugin-deprecation": "3.0.0", "eslint-plugin-jest": "28.10.0", "eslint-plugin-jest-extended": "2.4.0", "eslint-plugin-prettier": "5.2.1", - "folder-hash": "4.0.4", + "folder-hash": "4.1.0", "husky": "9.1.7", "jest": "29.7.0", "jest-canvas-mock": "2.5.2", @@ -123,16 +123,16 @@ "jest-fail-on-console": "3.3.1", "jest-junit": "16.0.0", "jest-preset-angular": "14.4.2", - "lint-staged": "15.2.11", + "lint-staged": "15.3.0", "ng-mocks": "14.13.1", "ngxtension": "4.2.0", "patch-package": "8.0.0", "prettier": "3.4.2", "rimraf": "6.0.1", - "sass": "1.83.0", + "sass": "1.83.1", "ts-jest": "29.2.5", "typescript": "5.5.4", - "typescript-eslint": "8.18.2", + "typescript-eslint": "8.19.0", "vite-tsconfig-paths": "5.1.4", "vitest": "2.1.8", "weak-napi": "2.0.2" @@ -155,9 +155,9 @@ } }, "node_modules/@analogjs/vite-plugin-angular": { - "version": "1.10.3", - "resolved": "https://registry.npmjs.org/@analogjs/vite-plugin-angular/-/vite-plugin-angular-1.10.3.tgz", - "integrity": "sha512-3EWappJ5K6YopJpq2QRVim8qZgaTQJD0RB4G/DXo+Fg0s27BjDORiaeixqLHIwhUzzZ5FR2d1S7dgIi9zOg4sA==", + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/@analogjs/vite-plugin-angular/-/vite-plugin-angular-1.11.0.tgz", + "integrity": "sha512-18HSwOAVFQjwwRQPq9+duOoubuiut0INo55h0gV3v2TWS+tAP2wXP8SPyAG99P3ySNQB7zMUYE8mVsqLM+8bDA==", "dev": true, "license": "MIT", "dependencies": { @@ -7352,9 +7352,9 @@ } }, "node_modules/@types/node": { - "version": "22.10.2", - "resolved": "https://registry.npmjs.org/@types/node/-/node-22.10.2.tgz", - "integrity": "sha512-Xxr6BBRCAOQixvonOye19wnzyDiUtTeqldOOmj3CkeblonbccA12PFwlufvRdrpjXxqnmUaeiU5EOA+7s5diUQ==", + "version": "22.10.5", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.10.5.tgz", + "integrity": "sha512-F8Q+SeGimwOo86fiovQh8qiXfFEh2/ocYv7tU5pJ3EXMSSxk1Joj5wefpFK2fHTf/N6HKGSxIDBT9f3gCxXPkQ==", "dev": true, "license": "MIT", "dependencies": { @@ -7564,17 +7564,17 @@ "license": "MIT" }, "node_modules/@typescript-eslint/eslint-plugin": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.18.1.tgz", - "integrity": "sha512-Ncvsq5CT3Gvh+uJG0Lwlho6suwDfUXH0HztslDf5I+F2wAFAZMRwYLEorumpKLzmO2suAXZ/td1tBg4NZIi9CQ==", + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.19.0.tgz", + "integrity": "sha512-NggSaEZCdSrFddbctrVjkVZvFC6KGfKfNK0CU7mNK/iKHGKbzT4Wmgm08dKpcZECBu9f5FypndoMyRHkdqfT1Q==", "dev": true, "license": "MIT", "dependencies": { "@eslint-community/regexpp": "^4.10.0", - "@typescript-eslint/scope-manager": "8.18.1", - "@typescript-eslint/type-utils": "8.18.1", - "@typescript-eslint/utils": "8.18.1", - "@typescript-eslint/visitor-keys": "8.18.1", + "@typescript-eslint/scope-manager": "8.19.0", + "@typescript-eslint/type-utils": "8.19.0", + "@typescript-eslint/utils": "8.19.0", + "@typescript-eslint/visitor-keys": "8.19.0", "graphemer": "^1.4.0", "ignore": "^5.3.1", "natural-compare": "^1.4.0", @@ -7593,44 +7593,6 @@ "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/types": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", - "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/utils": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.18.1.tgz", - "integrity": "sha512-8vikiIj2ebrC4WRdcAdDcmnu9Q/MXXwg+STf40BVfT8exDqBCUPdypvzcUPxEqRGKg9ALagZ0UWcYCtn+4W2iQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@eslint-community/eslint-utils": "^4.4.0", - "@typescript-eslint/scope-manager": "8.18.1", - "@typescript-eslint/types": "8.18.1", - "@typescript-eslint/typescript-estree": "8.18.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <5.8.0" - } - }, "node_modules/@typescript-eslint/eslint-plugin/node_modules/ignore": { "version": "5.3.2", "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", @@ -7642,16 +7604,16 @@ } }, "node_modules/@typescript-eslint/parser": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.18.1.tgz", - "integrity": "sha512-rBnTWHCdbYM2lh7hjyXqxk70wvon3p2FyaniZuey5TrcGBpfhVp0OxOa6gxr9Q9YhZFKyfbEnxc24ZnVbbUkCA==", + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.19.0.tgz", + "integrity": "sha512-6M8taKyOETY1TKHp0x8ndycipTVgmp4xtg5QpEZzXxDhNvvHOJi5rLRkLr8SK3jTgD5l4fTlvBiRdfsuWydxBw==", "dev": true, "license": "MIT", "dependencies": { - "@typescript-eslint/scope-manager": "8.18.1", - "@typescript-eslint/types": "8.18.1", - "@typescript-eslint/typescript-estree": "8.18.1", - "@typescript-eslint/visitor-keys": "8.18.1", + "@typescript-eslint/scope-manager": "8.19.0", + "@typescript-eslint/types": "8.19.0", + "@typescript-eslint/typescript-estree": "8.19.0", + "@typescript-eslint/visitor-keys": "8.19.0", "debug": "^4.3.4" }, "engines": { @@ -7666,44 +7628,16 @@ "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/@typescript-eslint/parser/node_modules/@typescript-eslint/types": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", - "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, "node_modules/@typescript-eslint/scope-manager": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.18.1.tgz", - "integrity": "sha512-HxfHo2b090M5s2+/9Z3gkBhI6xBH8OJCFjH9MhQ+nnoZqxU3wNxkLT+VWXWSFWc3UF3Z+CfPAyqdCTdoXtDPCQ==", + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.19.0.tgz", + "integrity": "sha512-hkoJiKQS3GQ13TSMEiuNmSCvhz7ujyqD1x3ShbaETATHrck+9RaDdUbt+osXaUuns9OFwrDTTrjtwsU8gJyyRA==", "dev": true, "license": "MIT", "dependencies": { - "@typescript-eslint/types": "8.18.1", - "@typescript-eslint/visitor-keys": "8.18.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "@typescript-eslint/types": "8.19.0", + "@typescript-eslint/visitor-keys": "8.19.0" }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/scope-manager/node_modules/@typescript-eslint/types": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", - "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", - "dev": true, - "license": "MIT", "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, @@ -7713,14 +7647,14 @@ } }, "node_modules/@typescript-eslint/type-utils": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.18.1.tgz", - "integrity": "sha512-jAhTdK/Qx2NJPNOTxXpMwlOiSymtR2j283TtPqXkKBdH8OAMmhiUfP0kJjc/qSE51Xrq02Gj9NY7MwK+UxVwHQ==", + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.19.0.tgz", + "integrity": "sha512-TZs0I0OSbd5Aza4qAMpp1cdCYVnER94IziudE3JU328YUHgWu9gwiwhag+fuLeJ2LkWLXI+F/182TbG+JaBdTg==", "dev": true, "license": "MIT", "dependencies": { - "@typescript-eslint/typescript-estree": "8.18.1", - "@typescript-eslint/utils": "8.18.1", + "@typescript-eslint/typescript-estree": "8.19.0", + "@typescript-eslint/utils": "8.19.0", "debug": "^4.3.4", "ts-api-utils": "^1.3.0" }, @@ -7736,44 +7670,6 @@ "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/types": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", - "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/utils": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.18.1.tgz", - "integrity": "sha512-8vikiIj2ebrC4WRdcAdDcmnu9Q/MXXwg+STf40BVfT8exDqBCUPdypvzcUPxEqRGKg9ALagZ0UWcYCtn+4W2iQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@eslint-community/eslint-utils": "^4.4.0", - "@typescript-eslint/scope-manager": "8.18.1", - "@typescript-eslint/types": "8.18.1", - "@typescript-eslint/typescript-estree": "8.18.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <5.8.0" - } - }, "node_modules/@typescript-eslint/types": { "version": "8.19.0", "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.19.0.tgz", @@ -7789,14 +7685,14 @@ } }, "node_modules/@typescript-eslint/typescript-estree": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.18.1.tgz", - "integrity": "sha512-z8U21WI5txzl2XYOW7i9hJhxoKKNG1kcU4RzyNvKrdZDmbjkmLBo8bgeiOJmA06kizLI76/CCBAAGlTlEeUfyg==", + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.19.0.tgz", + "integrity": "sha512-WW9PpDaLIFW9LCbucMSdYUuGeFUz1OkWYS/5fwZwTA+l2RwlWFdJvReQqMUMBw4yJWJOfqd7An9uwut2Oj8sLw==", "dev": true, "license": "MIT", "dependencies": { - "@typescript-eslint/types": "8.18.1", - "@typescript-eslint/visitor-keys": "8.18.1", + "@typescript-eslint/types": "8.19.0", + "@typescript-eslint/visitor-keys": "8.19.0", "debug": "^4.3.4", "fast-glob": "^3.3.2", "is-glob": "^4.0.3", @@ -7815,20 +7711,6 @@ "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/@typescript-eslint/types": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", - "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, "node_modules/@typescript-eslint/utils": { "version": "8.19.0", "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.19.0.tgz", @@ -7853,52 +7735,7 @@ "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/scope-manager": { - "version": "8.19.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.19.0.tgz", - "integrity": "sha512-hkoJiKQS3GQ13TSMEiuNmSCvhz7ujyqD1x3ShbaETATHrck+9RaDdUbt+osXaUuns9OFwrDTTrjtwsU8gJyyRA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.19.0", - "@typescript-eslint/visitor-keys": "8.19.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/typescript-estree": { - "version": "8.19.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.19.0.tgz", - "integrity": "sha512-WW9PpDaLIFW9LCbucMSdYUuGeFUz1OkWYS/5fwZwTA+l2RwlWFdJvReQqMUMBw4yJWJOfqd7An9uwut2Oj8sLw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.19.0", - "@typescript-eslint/visitor-keys": "8.19.0", - "debug": "^4.3.4", - "fast-glob": "^3.3.2", - "is-glob": "^4.0.3", - "minimatch": "^9.0.4", - "semver": "^7.6.0", - "ts-api-utils": "^1.3.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "typescript": ">=4.8.4 <5.8.0" - } - }, - "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/visitor-keys": { + "node_modules/@typescript-eslint/visitor-keys": { "version": "8.19.0", "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.19.0.tgz", "integrity": "sha512-mCFtBbFBJDCNCWUl5y6sZSCHXw1DEFEk3c/M3nRK2a4XUB8StGFtmcEMizdjKuBzB6e/smJAAWYug3VrdLMr1w==", @@ -7916,51 +7753,6 @@ "url": "https://opencollective.com/typescript-eslint" } }, - "node_modules/@typescript-eslint/utils/node_modules/eslint-visitor-keys": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", - "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/@typescript-eslint/visitor-keys": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.18.1.tgz", - "integrity": "sha512-Vj0WLm5/ZsD013YeUKn+K0y8p1M0jPpxOkKdbD1wB0ns53a5piVY02zjf072TblEweAbcYiFiPoSMF3kp+VhhQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.18.1", - "eslint-visitor-keys": "^4.2.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/visitor-keys/node_modules/@typescript-eslint/types": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.1.tgz", - "integrity": "sha512-7uoAUsCj66qdNQNpH2G8MyTFlgerum8ubf21s3TSM3XmKXuIn+H2Sifh/ES2nPOPiYSRJWAk0fDkW0APBWcpfw==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, "node_modules/@typescript-eslint/visitor-keys/node_modules/eslint-visitor-keys": { "version": "4.2.0", "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", @@ -11036,9 +10828,9 @@ "license": "MIT" }, "node_modules/emoji-js": { - "version": "3.8.0", - "resolved": "https://registry.npmjs.org/emoji-js/-/emoji-js-3.8.0.tgz", - "integrity": "sha512-A5FNHKlRPRo6RJWrrdGWnoolIBMkVXHy4qkO0V5ahekQPjfVECxvOOWADeAF/SbzRVA9Sxdj24FCoRYGt06skA==", + "version": "3.8.1", + "resolved": "https://registry.npmjs.org/emoji-js/-/emoji-js-3.8.1.tgz", + "integrity": "sha512-yyXMnZLXgqQHAhEm2DKK4Nrca+jbLQfNOP2mLcNTS6XxzzbQLDFHAguPQrtJS4Udot0Pvomwmh1ckQzhrePhKw==", "license": "MIT", "dependencies": { "emoji-datasource": "15.0.1" @@ -12499,14 +12291,14 @@ "license": "ISC" }, "node_modules/folder-hash": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/folder-hash/-/folder-hash-4.0.4.tgz", - "integrity": "sha512-zEyYH+UsHEfJJcCRSf9ai5I4CTZwZ8ObONRuEI5hcEmJY5pS0FUWKruX9mMnYJrgC7MlPFDYnGsK1R+WFYjLlQ==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/folder-hash/-/folder-hash-4.1.0.tgz", + "integrity": "sha512-45DBqW/wE1FdEynQ922zoKsICdAkZEkuyWijChitENaoI+AF/FsTlKOB+EwaEqhNfBZ5dEgLVv694wbpNMIC5g==", "dev": true, "license": "MIT", "dependencies": { - "debug": "^4.3.3", - "minimatch": "~5.1.2" + "debug": "4.4.0", + "minimatch": "10.0.1" }, "bin": { "folder-hash": "bin/folder-hash" @@ -12516,16 +12308,19 @@ } }, "node_modules/folder-hash/node_modules/minimatch": { - "version": "5.1.6", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz", - "integrity": "sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==", + "version": "10.0.1", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.0.1.tgz", + "integrity": "sha512-ethXTt3SGGR+95gudmqJ1eNhRO7eGEGIgYA9vnPatK4/etz2MEVDno5GMCibdMTuBMyElzIlgxMna3K94XDIDQ==", "dev": true, "license": "ISC", "dependencies": { "brace-expansion": "^2.0.1" }, "engines": { - "node": ">=10" + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, "node_modules/follow-redirects": { @@ -15604,13 +15399,13 @@ } }, "node_modules/lint-staged": { - "version": "15.2.11", - "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-15.2.11.tgz", - "integrity": "sha512-Ev6ivCTYRTGs9ychvpVw35m/bcNDuBN+mnTeObCL5h+boS5WzBEC6LHI4I9F/++sZm1m+J2LEiy0gxL/R9TBqQ==", + "version": "15.3.0", + "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-15.3.0.tgz", + "integrity": "sha512-vHFahytLoF2enJklgtOtCtIjZrKD/LoxlaUusd5nh7dWv/dkKQJY74ndFSzxCdv7g0ueGg1ORgTSt4Y9LPZn9A==", "dev": true, "license": "MIT", "dependencies": { - "chalk": "~5.3.0", + "chalk": "~5.4.1", "commander": "~12.1.0", "debug": "~4.4.0", "execa": "~8.0.1", @@ -15658,9 +15453,9 @@ } }, "node_modules/lint-staged/node_modules/chalk": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.3.0.tgz", - "integrity": "sha512-dLitG79d+GV1Nb/VYcCDFivJeK1hiukt9QjRNVOsUtTy1rR1YJsmpGGTZ3qJos+uw7WmWF4wUwBd9jxjocFC2w==", + "version": "5.4.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.4.1.tgz", + "integrity": "sha512-zgVZuo2WcZgfUEmsn6eO3kINexW8RAE4maiQ8QNs8CtpPCSyMiYsULR3HQYkm3w8FIA3SberyMJMSldGsW+U3w==", "dev": true, "license": "MIT", "engines": { @@ -18312,15 +18107,15 @@ "license": "0BSD" }, "node_modules/pdfjs-dist": { - "version": "4.9.155", - "resolved": "https://registry.npmjs.org/pdfjs-dist/-/pdfjs-dist-4.9.155.tgz", - "integrity": "sha512-epRZn6DQQKCOEqbmFsxkiMBm1MHaNrnr6T4VBNP0bsDvdJdmrWcZbS5cgJXW68P0d3uJTlFhF6Wms2tlSgPYig==", + "version": "4.10.38", + "resolved": "https://registry.npmjs.org/pdfjs-dist/-/pdfjs-dist-4.10.38.tgz", + "integrity": "sha512-/Y3fcFrXEAsMjJXeL9J8+ZG9U01LbuWaYypvDW2ycW1jL269L3js3DVBjDJ0Up9Np1uqDXsDrRihHANhZOlwdQ==", "license": "Apache-2.0", "engines": { "node": ">=20" }, "optionalDependencies": { - "@napi-rs/canvas": "^0.1.64" + "@napi-rs/canvas": "^0.1.65" } }, "node_modules/pepjs": { @@ -19668,9 +19463,9 @@ "license": "MIT" }, "node_modules/sass": { - "version": "1.83.0", - "resolved": "https://registry.npmjs.org/sass/-/sass-1.83.0.tgz", - "integrity": "sha512-qsSxlayzoOjdvXMVLkzF84DJFc2HZEL/rFyGIKbbilYtAvlCxyuzUeff9LawTn4btVnLKg75Z8MMr1lxU1lfGw==", + "version": "1.83.1", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.83.1.tgz", + "integrity": "sha512-EVJbDaEs4Rr3F0glJzFSOvtg2/oy2V/YrGFPqPY24UqcLDWcI9ZY5sN+qyO3c/QCZwzgfirvhXvINiJCE/OLcA==", "dev": true, "license": "MIT", "dependencies": { @@ -21636,177 +21431,15 @@ } }, "node_modules/typescript-eslint": { - "version": "8.18.2", - "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.18.2.tgz", - "integrity": "sha512-KuXezG6jHkvC3MvizeXgupZzaG5wjhU3yE8E7e6viOvAvD9xAWYp8/vy0WULTGe9DYDWcQu7aW03YIV3mSitrQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/eslint-plugin": "8.18.2", - "@typescript-eslint/parser": "8.18.2", - "@typescript-eslint/utils": "8.18.2" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <5.8.0" - } - }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/eslint-plugin": { - "version": "8.18.2", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.18.2.tgz", - "integrity": "sha512-adig4SzPLjeQ0Tm+jvsozSGiCliI2ajeURDGHjZ2llnA+A67HihCQ+a3amtPhUakd1GlwHxSRvzOZktbEvhPPg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@eslint-community/regexpp": "^4.10.0", - "@typescript-eslint/scope-manager": "8.18.2", - "@typescript-eslint/type-utils": "8.18.2", - "@typescript-eslint/utils": "8.18.2", - "@typescript-eslint/visitor-keys": "8.18.2", - "graphemer": "^1.4.0", - "ignore": "^5.3.1", - "natural-compare": "^1.4.0", - "ts-api-utils": "^1.3.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "@typescript-eslint/parser": "^8.0.0 || ^8.0.0-alpha.0", - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <5.8.0" - } - }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/parser": { - "version": "8.18.2", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.18.2.tgz", - "integrity": "sha512-y7tcq4StgxQD4mDr9+Jb26dZ+HTZ/SkfqpXSiqeUXZHxOUyjWDKsmwKhJ0/tApR08DgOhrFAoAhyB80/p3ViuA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/scope-manager": "8.18.2", - "@typescript-eslint/types": "8.18.2", - "@typescript-eslint/typescript-estree": "8.18.2", - "@typescript-eslint/visitor-keys": "8.18.2", - "debug": "^4.3.4" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <5.8.0" - } - }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/scope-manager": { - "version": "8.18.2", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.18.2.tgz", - "integrity": "sha512-YJFSfbd0CJjy14r/EvWapYgV4R5CHzptssoag2M7y3Ra7XNta6GPAJPPP5KGB9j14viYXyrzRO5GkX7CRfo8/g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.18.2", - "@typescript-eslint/visitor-keys": "8.18.2" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/type-utils": { - "version": "8.18.2", - "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.18.2.tgz", - "integrity": "sha512-AB/Wr1Lz31bzHfGm/jgbFR0VB0SML/hd2P1yxzKDM48YmP7vbyJNHRExUE/wZsQj2wUCvbWH8poNHFuxLqCTnA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/typescript-estree": "8.18.2", - "@typescript-eslint/utils": "8.18.2", - "debug": "^4.3.4", - "ts-api-utils": "^1.3.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <5.8.0" - } - }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/types": { - "version": "8.18.2", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.18.2.tgz", - "integrity": "sha512-Z/zblEPp8cIvmEn6+tPDIHUbRu/0z5lqZ+NvolL5SvXWT5rQy7+Nch83M0++XzO0XrWRFWECgOAyE8bsJTl1GQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/typescript-estree": { - "version": "8.18.2", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.18.2.tgz", - "integrity": "sha512-WXAVt595HjpmlfH4crSdM/1bcsqh+1weFRWIa9XMTx/XHZ9TCKMcr725tLYqWOgzKdeDrqVHxFotrvWcEsk2Tg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.18.2", - "@typescript-eslint/visitor-keys": "8.18.2", - "debug": "^4.3.4", - "fast-glob": "^3.3.2", - "is-glob": "^4.0.3", - "minimatch": "^9.0.4", - "semver": "^7.6.0", - "ts-api-utils": "^1.3.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "typescript": ">=4.8.4 <5.8.0" - } - }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/utils": { - "version": "8.18.2", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.18.2.tgz", - "integrity": "sha512-Cr4A0H7DtVIPkauj4sTSXVl+VBWewE9/o40KcF3TV9aqDEOWoXF3/+oRXNby3DYzZeCATvbdksYsGZzplwnK/Q==", + "version": "8.19.0", + "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.19.0.tgz", + "integrity": "sha512-Ni8sUkVWYK4KAcTtPjQ/UTiRk6jcsuDhPpxULapUDi8A/l8TSBk+t1GtJA1RsCzIJg0q6+J7bf35AwQigENWRQ==", "dev": true, "license": "MIT", "dependencies": { - "@eslint-community/eslint-utils": "^4.4.0", - "@typescript-eslint/scope-manager": "8.18.2", - "@typescript-eslint/types": "8.18.2", - "@typescript-eslint/typescript-estree": "8.18.2" + "@typescript-eslint/eslint-plugin": "8.19.0", + "@typescript-eslint/parser": "8.19.0", + "@typescript-eslint/utils": "8.19.0" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -21820,47 +21453,6 @@ "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/visitor-keys": { - "version": "8.18.2", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.18.2.tgz", - "integrity": "sha512-zORcwn4C3trOWiCqFQP1x6G3xTRyZ1LYydnj51cRnJ6hxBlr/cKPckk+PKPUw/fXmvfKTcw7bwY3w9izgx5jZw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.18.2", - "eslint-visitor-keys": "^4.2.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/typescript-eslint/node_modules/eslint-visitor-keys": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", - "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/typescript-eslint/node_modules/ignore": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", - "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 4" - } - }, "node_modules/typescript-logic": { "version": "0.0.0", "resolved": "https://registry.npmjs.org/typescript-logic/-/typescript-logic-0.0.0.tgz", diff --git a/package.json b/package.json index 1663425bd516..ceceebed252a 100644 --- a/package.json +++ b/package.json @@ -49,7 +49,7 @@ "dayjs": "1.11.13", "diff-match-patch-typescript": "1.1.0", "dompurify": "3.2.3", - "emoji-js": "3.8.0", + "emoji-js": "3.8.1", "export-to-csv": "1.4.0", "fast-json-patch": "3.1.1", "franc-min": "6.2.0", @@ -68,7 +68,7 @@ "ngx-webstorage": "18.0.0", "papaparse": "5.4.1", "pdf-lib": "1.17.1", - "pdfjs-dist": "4.9.155", + "pdfjs-dist": "4.10.38", "rxjs": "7.8.1", "simple-statistics": "7.8.7", "smoothscroll-polyfill": "0.4.4", @@ -107,7 +107,7 @@ "yargs-parser": "21.1.1" }, "devDependencies": { - "@analogjs/vite-plugin-angular": "1.10.3", + "@analogjs/vite-plugin-angular": "1.11.0", "@angular-builders/jest": "18.0.0", "@angular-devkit/build-angular": "18.2.12", "@angular-eslint/builder": "18.4.1", @@ -127,21 +127,21 @@ "@types/jest": "29.5.14", "@types/lodash-es": "4.17.12", "@types/markdown-it": "14.1.2", - "@types/node": "22.10.2", + "@types/node": "22.10.5", "@types/papaparse": "5.3.15", "@types/smoothscroll-polyfill": "0.3.4", "@types/sockjs-client": "1.5.4", "@types/turndown": "5.0.5", "@types/uuid": "10.0.0", - "@typescript-eslint/eslint-plugin": "8.18.1", - "@typescript-eslint/parser": "8.18.1", + "@typescript-eslint/eslint-plugin": "8.19.0", + "@typescript-eslint/parser": "8.19.0", "eslint": "9.17.0", "eslint-config-prettier": "9.1.0", "eslint-plugin-deprecation": "3.0.0", "eslint-plugin-jest": "28.10.0", "eslint-plugin-jest-extended": "2.4.0", "eslint-plugin-prettier": "5.2.1", - "folder-hash": "4.0.4", + "folder-hash": "4.1.0", "husky": "9.1.7", "jest": "29.7.0", "jest-canvas-mock": "2.5.2", @@ -150,16 +150,16 @@ "jest-fail-on-console": "3.3.1", "jest-junit": "16.0.0", "jest-preset-angular": "14.4.2", - "lint-staged": "15.2.11", + "lint-staged": "15.3.0", "ng-mocks": "14.13.1", "ngxtension": "4.2.0", "patch-package": "8.0.0", "prettier": "3.4.2", "rimraf": "6.0.1", - "sass": "1.83.0", + "sass": "1.83.1", "ts-jest": "29.2.5", "typescript": "5.5.4", - "typescript-eslint": "8.18.2", + "typescript-eslint": "8.19.0", "vite-tsconfig-paths": "5.1.4", "vitest": "2.1.8", "weak-napi": "2.0.2" From 327857d0233168b77625451aa2136577a05be4ca Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Sun, 5 Jan 2025 09:44:54 +0100 Subject: [PATCH 13/34] Development: Update server dependencies --- build.gradle | 6 +++--- gradle.properties | 4 ++-- .../templates/java/test/gradle/projectTemplate/build.gradle | 6 ++---- 3 files changed, 7 insertions(+), 9 deletions(-) diff --git a/build.gradle b/build.gradle index fe1c9e2f9d8d..339abaed3836 100644 --- a/build.gradle +++ b/build.gradle @@ -306,7 +306,7 @@ dependencies { implementation "org.springframework.security:spring-security-oauth2-core:${spring_security_version}" implementation "org.springframework.security:spring-security-oauth2-client:${spring_security_version}" // use newest version of nimbus-jose-jwt to avoid security issues through outdated dependencies - implementation "com.nimbusds:nimbus-jose-jwt:9.48" + implementation "com.nimbusds:nimbus-jose-jwt:10.0.1" implementation "org.springframework.security:spring-security-oauth2-jose:${spring_security_version}" implementation "org.springframework.security:spring-security-crypto:${spring_security_version}" @@ -400,7 +400,7 @@ dependencies { testImplementation "org.springframework.boot:spring-boot-starter-test:${spring_boot_version}" testImplementation "org.springframework.security:spring-security-test:${spring_security_version}" testImplementation "org.springframework.boot:spring-boot-test:${spring_boot_version}" - testImplementation "org.assertj:assertj-core:3.27.0" + testImplementation "org.assertj:assertj-core:3.27.2" testImplementation "org.mockito:mockito-core:${mockito_version}" testImplementation "org.mockito:mockito-junit-jupiter:${mockito_version}" @@ -420,7 +420,7 @@ dependencies { testImplementation "com.h2database:h2:2.2.224" // Lightweight JSON library needed for the internals of the MockRestServiceServer - testImplementation "org.json:json:20240303" + testImplementation "org.json:json:20241224" // NOTE: make sure this corresponds to the version used for JUnit in the testImplementation testRuntimeOnly "org.junit.platform:junit-platform-launcher:${junit_platform_version}" diff --git a/gradle.properties b/gradle.properties index 646b23dcf6fa..37430dbcce1e 100644 --- a/gradle.properties +++ b/gradle.properties @@ -28,7 +28,7 @@ jplag_version=5.1.0 # NOTE: we cannot need to use the latest version 9.x or 10.x here as long as Stanford CoreNLP does not reference it lucene_version=8.11.4 slf4j_version=2.0.16 -sentry_version=7.19.1 +sentry_version=7.20.0 liquibase_version=4.30.0 docker_java_version=3.4.1 logback_version=1.5.15 @@ -41,7 +41,7 @@ mysql_version=9.1.0 # make sure both versions are compatible junit_version=5.11.3 junit_platform_version=1.11.4 -mockito_version=5.14.2 +mockito_version=5.15.2 testcontainer_version=1.20.4 # gradle plugin version diff --git a/src/main/resources/templates/java/test/gradle/projectTemplate/build.gradle b/src/main/resources/templates/java/test/gradle/projectTemplate/build.gradle index b099c0b33b67..611ce8468738 100644 --- a/src/main/resources/templates/java/test/gradle/projectTemplate/build.gradle +++ b/src/main/resources/templates/java/test/gradle/projectTemplate/build.gradle @@ -115,17 +115,15 @@ configurations { behaviorTestRuntime.extendsFrom testRuntime } -task structuralTests(type: Test) { +tasks.register('structuralTests', Test) { testClassesDirs = sourceSets.structuralTest.output.classesDirs classpath += sourceSets.structuralTest.runtimeClasspath - useJUnitPlatform() } -task behaviorTests(type: Test) { +tasks.register('behaviorTests', Test) { testClassesDirs = sourceSets.behaviorTest.output.classesDirs classpath += sourceSets.behaviorTest.runtimeClasspath - useJUnitPlatform() } // %sequential-stop% From 6b6e23e1dd9edef01ddb6cac93e5821b8fdd9e76 Mon Sep 17 00:00:00 2001 From: Simon Entholzer <33342534+SimonEntholzer@users.noreply.github.com> Date: Sun, 5 Jan 2025 20:20:12 +0100 Subject: [PATCH 14/34] Development: Fix syntax error in query in IrisExerciseSettingsRepository (#10104) --- .../artemis/iris/repository/IrisExerciseSettingsRepository.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseSettingsRepository.java b/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseSettingsRepository.java index 8035ec965a2b..74b3e5abf57c 100644 --- a/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseSettingsRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/iris/repository/IrisExerciseSettingsRepository.java @@ -15,7 +15,7 @@ public interface IrisExerciseSettingsRepository extends ArtemisJpaRepository { @Query(""" - SELECT EXISTS(s) + SELECT COUNT(s) > 0 FROM IrisExerciseSettings s WHERE s.exercise.id = :exerciseId AND s.irisTextExerciseChatSettings.enabled = TRUE From 99c5fb97a12d930262317e44368a70ba109f6615 Mon Sep 17 00:00:00 2001 From: Mohamed Bilel Besrour <58034472+BBesrour@users.noreply.github.com> Date: Sun, 5 Jan 2025 20:37:26 +0100 Subject: [PATCH 15/34] Integrated code lifecycle: View build logs in the browser (#9990) --- .../service/BuildLogEntryService.java | 52 +++++++++++++++++++ .../web/localci/BuildLogResource.java | 24 +++++++++ .../entities/programming/build-log.model.ts | 5 ++ .../build-queue/build-queue.component.html | 47 +++++++++++++++-- .../build-queue/build-queue.component.scss | 18 +++++++ .../build-queue/build-queue.component.ts | 38 +++++++++++--- .../build-queue/build-queue.service.ts | 13 +++++ .../app/shared/pipes/artemis-date.pipe.ts | 17 +++--- .../webapp/app/shared/util/global.utils.ts | 5 +- src/main/webapp/i18n/de/buildQueue.json | 6 +++ src/main/webapp/i18n/en/buildQueue.json | 6 +++ .../icl/LocalCIResourceIntegrationTest.java | 25 +++++++++ .../build-queue/build-queue.component.spec.ts | 33 ++++++++++++ .../build-queue/build-queue.service.spec.ts | 47 +++++++++++++++++ 14 files changed, 319 insertions(+), 17 deletions(-) diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildLogEntryService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildLogEntryService.java index 1ee0f9f1470a..d1b016f7ba81 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildLogEntryService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/BuildLogEntryService.java @@ -2,12 +2,15 @@ import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; +import java.io.BufferedReader; import java.io.IOException; +import java.io.InputStreamReader; import java.nio.charset.StandardCharsets; import java.nio.file.DirectoryStream; import java.nio.file.Files; import java.nio.file.Path; import java.time.ZonedDateTime; +import java.time.format.DateTimeParseException; import java.util.ArrayList; import java.util.Collections; import java.util.List; @@ -489,4 +492,53 @@ public boolean buildJobHasLogFile(String buildJobId, ProgrammingExercise program return Files.exists(logPath); } + /** + * Parses the build log entries from a given file and returns them as a list of {@link BuildLogDTO} objects. + * + *

+ * The method reads the file line by line and splits each line into a timestamp and a log message. + * The timestamp is expected to be separated from the log message by a tab character. + * If the timestamp cannot be parsed, the log message is appended to the previous entry. + *

+ * + * @param buildLog The {@link FileSystemResource} representing the build log file. + * @return A list of {@link BuildLogDTO} objects containing the parsed build log entries. + */ + public List parseBuildLogEntries(FileSystemResource buildLog) { + try { + List buildLogEntries = new ArrayList<>(); + try (BufferedReader reader = new BufferedReader(new InputStreamReader(buildLog.getInputStream()))) { + String line; + while ((line = reader.readLine()) != null) { + // split into timestamp and log message + int logMessageParts = 2; + String[] parts = line.split("\t", logMessageParts); + if (parts.length == logMessageParts) { + try { + ZonedDateTime time = ZonedDateTime.parse(parts[0]); + buildLogEntries.add(new BuildLogDTO(time, parts[1])); + } + catch (DateTimeParseException e) { + // If the time cannot be parsed, append the line to the last entry + if (!buildLogEntries.isEmpty()) { + BuildLogDTO lastEntry = buildLogEntries.getLast(); + buildLogEntries.set(buildLogEntries.size() - 1, new BuildLogDTO(lastEntry.time(), lastEntry.log() + "\n\t" + line)); + } + } + } + else { + // If the line does not contain a tab, add it to in a new entry + BuildLogDTO lastEntry = buildLogEntries.getLast(); + buildLogEntries.add(new BuildLogDTO(lastEntry.time(), line)); + } + } + } + return buildLogEntries; + } + catch (IOException e) { + log.error("Error occurred while trying to parse build log entries", e); + return new ArrayList<>(); + } + } + } diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/web/localci/BuildLogResource.java b/src/main/java/de/tum/cit/aet/artemis/programming/web/localci/BuildLogResource.java index c8fd18e69fca..ca33dd04d34e 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/web/localci/BuildLogResource.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/web/localci/BuildLogResource.java @@ -2,6 +2,8 @@ import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_LOCALCI; +import java.util.List; + import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.context.annotation.Profile; @@ -16,6 +18,7 @@ import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; +import de.tum.cit.aet.artemis.buildagent.dto.BuildLogDTO; import de.tum.cit.aet.artemis.core.security.annotations.EnforceAtLeastEditor; import de.tum.cit.aet.artemis.programming.service.BuildLogEntryService; @@ -52,4 +55,25 @@ public ResponseEntity getBuildLogForBuildJob(@PathVariable String buil responseHeaders.setContentDispositionFormData("attachment", "build-" + buildJobId + ".log"); return new ResponseEntity<>(buildLog, responseHeaders, HttpStatus.OK); } + + /** + * GET /build-log/{buildJobId}/entries : get the build log entries for a given result + * + * @param buildJobId the id of the build job for which to retrieve the build log entries + * @return the ResponseEntity with status 200 (OK) and the build log entries in the body, or with status 404 (Not Found) if the build log entries could not be found + */ + @GetMapping("build-log/{buildJobId}/entries") + @EnforceAtLeastEditor + public ResponseEntity> getBuildLogEntriesForBuildJob(@PathVariable String buildJobId) { + FileSystemResource buildLog = buildLogEntryService.retrieveBuildLogsFromFileForBuildJob(buildJobId); + if (buildLog == null) { + return ResponseEntity.notFound().build(); + } + + var buildLogEntries = buildLogEntryService.parseBuildLogEntries(buildLog); + if (buildLogEntries == null || buildLogEntries.isEmpty()) { + return ResponseEntity.notFound().build(); + } + return ResponseEntity.ok(buildLogEntries); + } } diff --git a/src/main/webapp/app/entities/programming/build-log.model.ts b/src/main/webapp/app/entities/programming/build-log.model.ts index fd4135a005cd..18e951c737a6 100644 --- a/src/main/webapp/app/entities/programming/build-log.model.ts +++ b/src/main/webapp/app/entities/programming/build-log.model.ts @@ -8,6 +8,11 @@ export enum BuildLogType { OTHER = 'OTHER', } +export type BuildLogLines = { + time: any; + logLines: string[]; +}; + export type BuildLogEntry = { time: any; log: string; diff --git a/src/main/webapp/app/localci/build-queue/build-queue.component.html b/src/main/webapp/app/localci/build-queue/build-queue.component.html index a8f9a582cbf3..bc447c21135d 100644 --- a/src/main/webapp/app/localci/build-queue/build-queue.component.html +++ b/src/main/webapp/app/localci/build-queue/build-queue.component.html @@ -504,7 +504,7 @@

@@ -617,7 +617,7 @@

- + {{ finishedBuildJob.buildAgentAddress }} @@ -742,7 +742,7 @@

+ + + + + + diff --git a/src/main/webapp/app/localci/build-queue/build-queue.component.scss b/src/main/webapp/app/localci/build-queue/build-queue.component.scss index b67b0520fbb5..1989ed2fb6d9 100644 --- a/src/main/webapp/app/localci/build-queue/build-queue.component.scss +++ b/src/main/webapp/app/localci/build-queue/build-queue.component.scss @@ -34,3 +34,21 @@ .finish-jobs-column-strings { max-width: 180px; } + +.build-output { + height: inherit; + &__entry { + &-date { + width: 200px; + margin-right: 10px; + color: var(--secondary); + font-weight: normal; + float: left; + clear: left; + } + &-text { + margin-bottom: 0; + color: var(--body-color); + } + } +} diff --git a/src/main/webapp/app/localci/build-queue/build-queue.component.ts b/src/main/webapp/app/localci/build-queue/build-queue.component.ts index 79ca4931786c..45b337ba1bd1 100644 --- a/src/main/webapp/app/localci/build-queue/build-queue.component.ts +++ b/src/main/webapp/app/localci/build-queue/build-queue.component.ts @@ -19,6 +19,7 @@ import { NgbModal, NgbTypeahead } from '@ng-bootstrap/ng-bootstrap'; import { LocalStorageService } from 'ngx-webstorage'; import { Observable, OperatorFunction, Subject, Subscription, merge } from 'rxjs'; import { UI_RELOAD_TIME } from 'app/shared/constants/exercise-exam-constants'; +import { BuildLogEntry, BuildLogLines } from 'app/entities/programming/build-log.model'; export class FinishedBuildJobFilter { status?: string = undefined; @@ -159,6 +160,9 @@ export class BuildQueueComponent implements OnInit, OnDestroy { searchSubscription: Subscription; searchTerm?: string = undefined; + rawBuildLogs: BuildLogLines[] = []; + displayedBuildJobId?: string; + constructor( private route: ActivatedRoute, private websocketService: JhiWebsocketService, @@ -386,11 +390,33 @@ export class BuildQueueComponent implements OnInit, OnDestroy { /** * View the build logs of a specific build job - * @param resultId The id of the build job + * @param modal The modal to open + * @param buildJobId The id of the build job + */ + viewBuildLogs(modal: any, buildJobId: string | undefined): void { + if (buildJobId) { + this.openModal(modal, true); + this.displayedBuildJobId = buildJobId; + this.buildQueueService.getBuildJobLogs(buildJobId).subscribe({ + next: (buildLogs: BuildLogEntry[]) => { + this.rawBuildLogs = buildLogs.map((entry) => { + const logLines = entry.log ? entry.log.split('\n') : []; + return { time: entry.time, logLines: logLines }; + }); + }, + error: (res: HttpErrorResponse) => { + onError(this.alertService, res, false); + }, + }); + } + } + + /** + * Download the build logs of a specific build job */ - viewBuildLogs(resultId: string | undefined): void { - if (resultId) { - const url = `/api/build-log/${resultId}`; + downloadBuildLogs(): void { + if (this.displayedBuildJobId) { + const url = `/api/build-log/${this.displayedBuildJobId}`; window.open(url, '_blank'); } } @@ -443,8 +469,8 @@ export class BuildQueueComponent implements OnInit, OnDestroy { /** * Opens the modal. */ - open(content: any) { - this.modalService.open(content); + openModal(modal: any, fullscreen?: boolean, size?: 'sm' | 'lg' | 'xl', scrollable = true, keyboard = true) { + this.modalService.open(modal, { size, keyboard, scrollable, fullscreen }); } /** diff --git a/src/main/webapp/app/localci/build-queue/build-queue.service.ts b/src/main/webapp/app/localci/build-queue/build-queue.service.ts index fdc9bec66e2f..a004acb9eaa8 100644 --- a/src/main/webapp/app/localci/build-queue/build-queue.service.ts +++ b/src/main/webapp/app/localci/build-queue/build-queue.service.ts @@ -7,6 +7,7 @@ import { BuildJob, BuildJobStatistics, FinishedBuildJob, SpanType } from 'app/en import { createNestedRequestOption } from 'app/shared/util/request.util'; import { HttpResponse } from '@angular/common/http'; import { FinishedBuildJobFilter } from 'app/localci/build-queue/build-queue.component'; +import { BuildLogEntry } from 'app/entities/programming/build-log.model'; @Injectable({ providedIn: 'root' }) export class BuildQueueService { @@ -188,4 +189,16 @@ export class BuildQueueService { }), ); } + + /** + * Get all build jobs of a course in the queue + * @param buildJobId + */ + getBuildJobLogs(buildJobId: string): Observable { + return this.http.get(`${this.resourceUrl}/build-log/${buildJobId}/entries`).pipe( + catchError(() => { + return throwError(() => new Error('artemisApp.buildQueue.logs.errorFetchingLogs')); + }), + ); + } } diff --git a/src/main/webapp/app/shared/pipes/artemis-date.pipe.ts b/src/main/webapp/app/shared/pipes/artemis-date.pipe.ts index 36d88d6474f7..cd3749e9ede0 100644 --- a/src/main/webapp/app/shared/pipes/artemis-date.pipe.ts +++ b/src/main/webapp/app/shared/pipes/artemis-date.pipe.ts @@ -36,6 +36,7 @@ export class ArtemisDatePipe implements PipeTransform, OnDestroy { private showTime = true; private showSeconds = false; private showWeekday = false; + private showMilliSeconds = false; private static mobileDeviceSize = 768; constructor(private translateService: TranslateService) {} @@ -47,8 +48,9 @@ export class ArtemisDatePipe implements PipeTransform, OnDestroy { * @param seconds Should seconds be displayed? Defaults to false. * @param timeZone Explicit time zone that should be used instead of the local time zone. * @param weekday Should the weekday be displayed? Defaults to false. + * @param milliSeconds Should milliseconds be displayed? Defaults to false. */ - transform(dateTime: DateType, format: DateFormat = 'long', seconds = false, timeZone: string | undefined = undefined, weekday = false): string { + transform(dateTime: DateType, format: DateFormat = 'long', seconds = false, timeZone: string | undefined = undefined, weekday = false, milliSeconds = false): string { // Return empty string if given dateTime equals null or is not convertible to dayjs. if (!dateTime || !dayjs(dateTime).isValid()) { return ''; @@ -59,6 +61,7 @@ export class ArtemisDatePipe implements PipeTransform, OnDestroy { this.showTime = format !== 'short-date' && format !== 'long-date'; this.showSeconds = seconds; this.showWeekday = weekday; + this.showMilliSeconds = milliSeconds; // Evaluate the format length based on the current window width. this.formatLengthBasedOnWindowWidth(window.innerWidth); @@ -88,12 +91,12 @@ export class ArtemisDatePipe implements PipeTransform, OnDestroy { * @param format Format of the localized date time. Defaults to 'long'. * @param seconds Should seconds be displayed? Defaults to false. */ - static format(locale = 'en', format: DateFormat = 'long', seconds = false): string { + static format(locale = 'en', format: DateFormat = 'long', seconds = false, showMilliSeconds = false): string { const long = format === 'long' || format === 'long-date'; const showDate = format !== 'time'; const showTime = format !== 'short-date' && format !== 'long-date'; const dateFormat = ArtemisDatePipe.dateFormat(long, showDate, locale); - const timeFormat = ArtemisDatePipe.timeFormat(showTime, seconds); + const timeFormat = ArtemisDatePipe.timeFormat(showTime, seconds, showMilliSeconds); return dateFormat + (dateFormat && timeFormat ? ' ' : '') + timeFormat; } @@ -123,7 +126,7 @@ export class ArtemisDatePipe implements PipeTransform, OnDestroy { private format(): string { const dateFormat = ArtemisDatePipe.dateFormat(this.long, this.showDate, this.locale); - const timeFormat = ArtemisDatePipe.timeFormat(this.showTime, this.showSeconds); + const timeFormat = ArtemisDatePipe.timeFormat(this.showTime, this.showSeconds, this.showMilliSeconds); return dateFormat + (dateFormat && timeFormat ? ' ' : '') + timeFormat; } @@ -144,12 +147,14 @@ export class ArtemisDatePipe implements PipeTransform, OnDestroy { return format; } - private static timeFormat(showTime: boolean, showSeconds: boolean): string { + private static timeFormat(showTime: boolean, showSeconds: boolean, showMilliSeconds: boolean): string { if (!showTime) { return ''; } let format = 'HH:mm'; - if (showSeconds) { + if (showMilliSeconds) { + format = 'HH:mm:ss.SSS'; + } else if (showSeconds) { format = 'HH:mm:ss'; } return format; diff --git a/src/main/webapp/app/shared/util/global.utils.ts b/src/main/webapp/app/shared/util/global.utils.ts index 78948d8c3340..3261878a8ef4 100644 --- a/src/main/webapp/app/shared/util/global.utils.ts +++ b/src/main/webapp/app/shared/util/global.utils.ts @@ -77,8 +77,9 @@ export const matchRegexWithLineNumbers = (multiLineText: string, regex: RegExp): * Use alert service to show the error message from the error response * @param alertService the service used to show the exception messages to the user * @param error the error response that's status is used to determine the error message + * @param disableTranslation whether the error message should be translated */ -export const onError = (alertService: AlertService, error: HttpErrorResponse) => { +export const onError = (alertService: AlertService, error: HttpErrorResponse, disableTranslation: boolean = true) => { switch (error.status) { case 400: alertService.error('error.http.400'); @@ -100,7 +101,7 @@ export const onError = (alertService: AlertService, error: HttpErrorResponse) => alertService.addAlert({ type: AlertType.DANGER, message: error.message, - disableTranslation: true, + disableTranslation: disableTranslation, }); break; } diff --git a/src/main/webapp/i18n/de/buildQueue.json b/src/main/webapp/i18n/de/buildQueue.json index faa60fc5c108..7d56dc944003 100644 --- a/src/main/webapp/i18n/de/buildQueue.json +++ b/src/main/webapp/i18n/de/buildQueue.json @@ -74,6 +74,12 @@ "daySpan": "1 Tag", "weekSpan": "7 Tage", "monthSpan": "30 Tage" + }, + "logs": { + "title": "Build Logs für Job", + "download": "Herunterladen", + "noLogs": "Keine Logs verfügbar", + "errorFetchingLogs": "Fehler beim Abrufen von Build-Log-Einträgen für Build-Job" } } } diff --git a/src/main/webapp/i18n/en/buildQueue.json b/src/main/webapp/i18n/en/buildQueue.json index aa065f3617df..63a258d407d1 100644 --- a/src/main/webapp/i18n/en/buildQueue.json +++ b/src/main/webapp/i18n/en/buildQueue.json @@ -74,6 +74,12 @@ "daySpan": "1 day", "weekSpan": "7 days", "monthSpan": "30 days" + }, + "logs": { + "title": "Build Logs for Job", + "download": "Download", + "noLogs": "No logs available", + "errorFetchingLogs": "Failed to get build log entries for build job" } } } diff --git a/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIResourceIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIResourceIntegrationTest.java index 4a5219c6ba7b..eb03d08f5fd8 100644 --- a/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIResourceIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/programming/icl/LocalCIResourceIntegrationTest.java @@ -11,6 +11,7 @@ import java.time.ZonedDateTime; import java.time.temporal.ChronoUnit; import java.util.ArrayList; +import java.util.LinkedHashMap; import java.util.List; import org.junit.jupiter.api.AfterEach; @@ -346,6 +347,30 @@ void testGetBuildLogsForResult() throws Exception { } } + @Test + @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") + void testGetBuildLogsEntriesForResult() throws Exception { + try { + buildJobRepository.save(finishedJobForLogs); + BuildLogDTO buildLogEntry = new BuildLogDTO(ZonedDateTime.now(), "Dummy log"); + buildLogEntryService.saveBuildLogsToFile(List.of(buildLogEntry), "6", programmingExercise); + var response = request.get("/api/build-log/6/entries", HttpStatus.OK, List.class); + + LinkedHashMap responseMap = ((LinkedHashMap) response.getFirst()); + String log = responseMap.get("log").toString(); + ZonedDateTime time = ZonedDateTime.parse(responseMap.get("time").toString()); + assertThat(response).hasSize(1); + assertThat(buildLogEntry.log()).isEqualTo(log); + assertThat(buildLogEntry.time()).isEqualTo(time); + + } + finally { + Path buildLogFile = Path.of("build-logs").resolve(programmingExercise.getCourseViaExerciseGroupOrCourseMember().getShortName()) + .resolve(programmingExercise.getShortName()).resolve("6.log"); + Files.deleteIfExists(buildLogFile); + } + } + @Test @WithMockUser(username = TEST_PREFIX + "admin", roles = "ADMIN") void testGetBuildJobStatistics() throws Exception { diff --git a/src/test/javascript/spec/component/localci/build-queue/build-queue.component.spec.ts b/src/test/javascript/spec/component/localci/build-queue/build-queue.component.spec.ts index b22a67ee3f15..2e51313e967c 100644 --- a/src/test/javascript/spec/component/localci/build-queue/build-queue.component.spec.ts +++ b/src/test/javascript/spec/component/localci/build-queue/build-queue.component.spec.ts @@ -19,6 +19,7 @@ import { LocalStorageService } from 'ngx-webstorage'; import { MockLocalStorageService } from '../../../helpers/mocks/service/mock-local-storage.service'; import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; import { PieChartModule } from '@swimlane/ngx-charts'; +import { BuildLogEntry, BuildLogLines } from '../../../../../../main/webapp/app/entities/programming/build-log.model'; describe('BuildQueueComponent', () => { let component: BuildQueueComponent; @@ -40,6 +41,7 @@ describe('BuildQueueComponent', () => { getFinishedBuildJobs: jest.fn(), getBuildJobStatistics: jest.fn(), getBuildJobStatisticsForCourse: jest.fn(), + getBuildJobLogs: jest.fn(), }; const mockLocalStorageService = new MockLocalStorageService(); @@ -267,6 +269,17 @@ describe('BuildQueueComponent', () => { numberOfAppliedFilters: 0, }; + const buildLogEntries: BuildLogEntry[] = [ + { + time: dayjs('2024-01-01'), + log: 'log1', + }, + { + time: dayjs('2024-01-02'), + log: 'log2', + }, + ]; + beforeEach(waitForAsync(() => { mockActivatedRoute = { params: of({ courseId: testCourseId }) }; @@ -661,4 +674,24 @@ describe('BuildQueueComponent', () => { expect(component.finishedBuildJobFilter.areDatesValid).toBeFalsy(); expect(component.finishedBuildJobFilter.areDurationFiltersValid).toBeFalsy(); }); + + it('should download build logs', () => { + const buildJobId = '1'; + jest.spyOn(window, 'open').mockImplementation(); + + mockBuildQueueService.getBuildJobLogs = jest.fn().mockReturnValue(of(buildLogEntries)); + + const buildLogsMultiLines: BuildLogLines[] = buildLogEntries.map((entry) => { + return { time: entry.time, logLines: entry.log.split('\n') }; + }); + + component.viewBuildLogs(undefined, buildJobId); + + expect(mockBuildQueueService.getBuildJobLogs).toHaveBeenCalledWith(buildJobId); + expect(component.rawBuildLogs).toEqual(buildLogsMultiLines); + + component.downloadBuildLogs(); + + expect(window.open).toHaveBeenCalledWith(`/api/build-log/${component.displayedBuildJobId}`, '_blank'); + }); }); diff --git a/src/test/javascript/spec/component/localci/build-queue/build-queue.service.spec.ts b/src/test/javascript/spec/component/localci/build-queue/build-queue.service.spec.ts index 16a2d81d37ed..8f5a9af490f5 100644 --- a/src/test/javascript/spec/component/localci/build-queue/build-queue.service.spec.ts +++ b/src/test/javascript/spec/component/localci/build-queue/build-queue.service.spec.ts @@ -15,6 +15,7 @@ import { JobTimingInfo } from 'app/entities/job-timing-info.model'; import { BuildConfig } from 'app/entities/programming/build-config.model'; import { FinishedBuildJobFilter } from 'app/localci/build-queue/build-queue.component'; import { provideHttpClient } from '@angular/common/http'; +import { BuildLogEntry } from '../../../../../../main/webapp/app/entities/programming/build-log.model'; describe('BuildQueueService', () => { let service: BuildQueueService; @@ -32,6 +33,17 @@ describe('BuildQueueService', () => { filterOptions.buildStartDateFilterTo = dayjs('2024-01-02'); filterOptions.status = 'SUCCESSFUL'; + const buildLogEntries: BuildLogEntry[] = [ + { + time: dayjs('2024-01-01'), + log: 'log1', + }, + { + time: dayjs('2024-01-02'), + log: 'log2', + }, + ]; + const expectFilterParams = (req: TestRequest, filterOptions: FinishedBuildJobFilter) => { expect(req.request.params.get('buildAgentAddress')).toBe(filterOptions.buildAgentAddress); expect(req.request.params.get('buildDurationLower')).toBe(filterOptions.buildDurationFilterLowerBound?.toString()); @@ -589,6 +601,41 @@ describe('BuildQueueService', () => { expect(errorOccurred).toBeTrue(); })); + it('should return build log entries for a specific build job', () => { + const buildJobId = '1'; + const expectedResponse = buildLogEntries; + + service.getBuildJobLogs(buildJobId).subscribe((data) => { + expect(data).toEqual(expectedResponse); + }); + + const req = httpMock.expectOne(`${service.resourceUrl}/build-log/${buildJobId}/entries`); + expect(req.request.method).toBe('GET'); + req.flush(expectedResponse); + }); + + it('should handle errors when getting build log entries for a specific build job', fakeAsync(() => { + const buildJobId = '1'; + + let errorOccurred = false; + + service.getBuildJobLogs(buildJobId).subscribe({ + error: (err) => { + expect(err.message).toBe('artemisApp.buildQueue.logs.errorFetchingLogs'); + errorOccurred = true; + }, + }); + + const req = httpMock.expectOne(`${service.resourceUrl}/build-log/${buildJobId}/entries`); + expect(req.request.method).toBe('GET'); + + req.flush(null, { status: 500, statusText: 'Internal Server Error' }); + + tick(); + + expect(errorOccurred).toBeTrue(); + })); + afterEach(() => { httpMock.verify(); // Verify that there are no outstanding requests. }); From 294adf7cab365fbfd1678448b4a6d35ef2dca0bd Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Sun, 5 Jan 2025 20:30:36 +0100 Subject: [PATCH 16/34] Development: Update client dependencies --- jest.config.js | 8 +- package-lock.json | 441 +++++++++++++++++++--------------------------- package.json | 16 +- 3 files changed, 193 insertions(+), 272 deletions(-) diff --git a/jest.config.js b/jest.config.js index a5ce2254e1a5..efdaeff41351 100644 --- a/jest.config.js +++ b/jest.config.js @@ -105,10 +105,10 @@ module.exports = { coverageThreshold: { global: { // TODO: in the future, the following values should increase to at least 90% - statements: 87.71, - branches: 73.85, - functions: 82.28, - lines: 87.77, + statements: 87.73, + branches: 73.87, + functions: 82.44, + lines: 87.78, }, }, coverageReporters: ['clover', 'json', 'lcov', 'text-summary'], diff --git a/package-lock.json b/package-lock.json index 8ea357ad52d4..319cdead9f47 100644 --- a/package-lock.json +++ b/package-lock.json @@ -11,13 +11,13 @@ "license": "MIT", "dependencies": { "@angular/animations": "18.2.13", - "@angular/cdk": "18.2.13", + "@angular/cdk": "18.2.14", "@angular/common": "18.2.13", "@angular/compiler": "18.2.13", "@angular/core": "18.2.13", "@angular/forms": "18.2.13", "@angular/localize": "18.2.13", - "@angular/material": "18.2.13", + "@angular/material": "18.2.14", "@angular/platform-browser": "18.2.13", "@angular/platform-browser-dynamic": "18.2.13", "@angular/router": "18.2.13", @@ -74,7 +74,7 @@ "ts-cacheable": "1.0.10", "tslib": "2.8.1", "turndown": "7.2.0", - "uuid": "11.0.3", + "uuid": "11.0.4", "webstomp-client": "1.2.6", "xlsx": "https://cdn.sheetjs.com/xlsx-0.20.3/xlsx-0.20.3.tgz", "zone.js": "0.14.10" @@ -83,11 +83,11 @@ "@analogjs/vite-plugin-angular": "1.11.0", "@angular-builders/jest": "18.0.0", "@angular-devkit/build-angular": "18.2.12", - "@angular-eslint/builder": "18.4.1", - "@angular-eslint/eslint-plugin": "18.4.1", - "@angular-eslint/eslint-plugin-template": "18.4.1", - "@angular-eslint/schematics": "18.4.1", - "@angular-eslint/template-parser": "18.4.1", + "@angular-eslint/builder": "18.4.3", + "@angular-eslint/eslint-plugin": "18.4.3", + "@angular-eslint/eslint-plugin-template": "18.4.3", + "@angular-eslint/schematics": "18.4.3", + "@angular-eslint/template-parser": "18.4.3", "@angular/cli": "18.2.12", "@angular/compiler-cli": "18.2.13", "@angular/language-service": "18.2.13", @@ -429,44 +429,6 @@ } } }, - "node_modules/@angular-devkit/build-angular/node_modules/chokidar": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", - "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", - "dev": true, - "license": "MIT", - "dependencies": { - "anymatch": "~3.1.2", - "braces": "~3.0.2", - "glob-parent": "~5.1.2", - "is-binary-path": "~2.1.0", - "is-glob": "~4.0.1", - "normalize-path": "~3.0.0", - "readdirp": "~3.6.0" - }, - "engines": { - "node": ">= 8.10.0" - }, - "funding": { - "url": "https://paulmillr.com/funding/" - }, - "optionalDependencies": { - "fsevents": "~2.3.2" - } - }, - "node_modules/@angular-devkit/build-angular/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, "node_modules/@angular-devkit/build-angular/node_modules/immutable": { "version": "4.3.7", "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.3.7.tgz", @@ -474,32 +436,6 @@ "dev": true, "license": "MIT" }, - "node_modules/@angular-devkit/build-angular/node_modules/readdirp": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", - "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", - "dev": true, - "license": "MIT", - "dependencies": { - "picomatch": "^2.2.1" - }, - "engines": { - "node": ">=8.10.0" - } - }, - "node_modules/@angular-devkit/build-angular/node_modules/readdirp/node_modules/picomatch": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", - "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8.6" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, "node_modules/@angular-devkit/build-angular/node_modules/sass": { "version": "1.77.6", "resolved": "https://registry.npmjs.org/sass/-/sass-1.77.6.tgz", @@ -593,32 +529,36 @@ } }, "node_modules/@angular-eslint/builder": { - "version": "18.4.1", - "resolved": "https://registry.npmjs.org/@angular-eslint/builder/-/builder-18.4.1.tgz", - "integrity": "sha512-Ofkwd9Rg52K+AgvnV1RXYXVBGJvl5jD7+4dqwoprqXG7YKNTdHy5vqNZ5XDSMb26qjoZF7JC+IKruKFaON/ZaA==", + "version": "18.4.3", + "resolved": "https://registry.npmjs.org/@angular-eslint/builder/-/builder-18.4.3.tgz", + "integrity": "sha512-NzmrXlr7GFE+cjwipY/CxBscZXNqnuK0us1mO6Z2T6MeH6m+rRcdlY/rZyKoRniyNNvuzl6vpEsfMIMmnfebrA==", "dev": true, "license": "MIT", + "dependencies": { + "@angular-devkit/architect": ">= 0.1800.0 < 0.1900.0", + "@angular-devkit/core": ">= 18.0.0 < 19.0.0" + }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": "*" } }, "node_modules/@angular-eslint/bundled-angular-compiler": { - "version": "18.4.1", - "resolved": "https://registry.npmjs.org/@angular-eslint/bundled-angular-compiler/-/bundled-angular-compiler-18.4.1.tgz", - "integrity": "sha512-gCQC0mgBO1bwHDXL9CUgHW+Rf1XGZCLAopoXnggwxGkBCx+oww507t+jrSOxdh+4OTKU4ZfmbtWd7Y8AeXns8w==", + "version": "18.4.3", + "resolved": "https://registry.npmjs.org/@angular-eslint/bundled-angular-compiler/-/bundled-angular-compiler-18.4.3.tgz", + "integrity": "sha512-zdrA8mR98X+U4YgHzUKmivRU+PxzwOL/j8G7eTOvBuq8GPzsP+hvak+tyxlgeGm9HsvpFj9ERHLtJ0xDUPs8fg==", "dev": true, "license": "MIT" }, "node_modules/@angular-eslint/eslint-plugin": { - "version": "18.4.1", - "resolved": "https://registry.npmjs.org/@angular-eslint/eslint-plugin/-/eslint-plugin-18.4.1.tgz", - "integrity": "sha512-FoHwj+AFo8ONKb8wEK5qpo6uefuyklZlDqErJxeC3fpNIJzDe8PWBcJsuZt7Wwm/HeggWgt0Au6h+3IEa0V3BQ==", + "version": "18.4.3", + "resolved": "https://registry.npmjs.org/@angular-eslint/eslint-plugin/-/eslint-plugin-18.4.3.tgz", + "integrity": "sha512-AyJbupiwTBR81P6T59v+aULEnPpZBCBxL2S5QFWfAhNCwWhcof4GihvdK2Z87yhvzDGeAzUFSWl/beJfeFa+PA==", "dev": true, "license": "MIT", "dependencies": { - "@angular-eslint/bundled-angular-compiler": "18.4.1", - "@angular-eslint/utils": "18.4.1" + "@angular-eslint/bundled-angular-compiler": "18.4.3", + "@angular-eslint/utils": "18.4.3" }, "peerDependencies": { "@typescript-eslint/utils": "^7.11.0 || ^8.0.0", @@ -627,14 +567,14 @@ } }, "node_modules/@angular-eslint/eslint-plugin-template": { - "version": "18.4.1", - "resolved": "https://registry.npmjs.org/@angular-eslint/eslint-plugin-template/-/eslint-plugin-template-18.4.1.tgz", - "integrity": "sha512-sofnKpi6wOZ6avVfYYqB7sCgGgWF2HgCZfW+IAp1MtVD2FBa1zTSbbfIZ1I8Akpd22UXa4LKJd0TLwm5XHHkiQ==", + "version": "18.4.3", + "resolved": "https://registry.npmjs.org/@angular-eslint/eslint-plugin-template/-/eslint-plugin-template-18.4.3.tgz", + "integrity": "sha512-ijGlX2N01ayMXTpeQivOA31AszO8OEbu9ZQUCxnu9AyMMhxyi2q50bujRChAvN9YXQfdQtbxuajxV6+aiWb5BQ==", "dev": true, "license": "MIT", "dependencies": { - "@angular-eslint/bundled-angular-compiler": "18.4.1", - "@angular-eslint/utils": "18.4.1", + "@angular-eslint/bundled-angular-compiler": "18.4.3", + "@angular-eslint/utils": "18.4.3", "aria-query": "5.3.2", "axobject-query": "4.1.0" }, @@ -646,31 +586,29 @@ } }, "node_modules/@angular-eslint/schematics": { - "version": "18.4.1", - "resolved": "https://registry.npmjs.org/@angular-eslint/schematics/-/schematics-18.4.1.tgz", - "integrity": "sha512-1+gGodwh+UevtEx9mzZbzP1uY/9NAGEbsn8jisG1TEPDby2wKScQj6U6JwGxoW/Dd/4SIeSdilruZPALkqha7g==", + "version": "18.4.3", + "resolved": "https://registry.npmjs.org/@angular-eslint/schematics/-/schematics-18.4.3.tgz", + "integrity": "sha512-D5maKn5e6n58+8n7jLFLD4g+RGPOPeDSsvPc1sqial5tEKLxAJQJS9WZ28oef3bhkob6C60D+1H0mMmEEVvyVA==", "dev": true, "license": "MIT", "dependencies": { - "@angular-eslint/eslint-plugin": "18.4.1", - "@angular-eslint/eslint-plugin-template": "18.4.1", + "@angular-devkit/core": ">= 18.0.0 < 19.0.0", + "@angular-devkit/schematics": ">= 18.0.0 < 19.0.0", + "@angular-eslint/eslint-plugin": "18.4.3", + "@angular-eslint/eslint-plugin-template": "18.4.3", "ignore": "6.0.2", "semver": "7.6.3", "strip-json-comments": "3.1.1" - }, - "peerDependencies": { - "@angular-devkit/core": ">= 18.0.0 < 19.0.0", - "@angular-devkit/schematics": ">= 18.0.0 < 19.0.0" } }, "node_modules/@angular-eslint/template-parser": { - "version": "18.4.1", - "resolved": "https://registry.npmjs.org/@angular-eslint/template-parser/-/template-parser-18.4.1.tgz", - "integrity": "sha512-LsStXVyso/89gQU5eiJebB/b1j+wrRtTLjk+ODVUTa7NGCCT7B7xI6ToTchkBEpSTHLT9pEQXHsHer3FymsQRQ==", + "version": "18.4.3", + "resolved": "https://registry.npmjs.org/@angular-eslint/template-parser/-/template-parser-18.4.3.tgz", + "integrity": "sha512-JZMPtEB8yNip3kg4WDEWQyObSo2Hwf+opq2ElYuwe85GQkGhfJSJ2CQYo4FSwd+c5MUQAqESNRg9QqGYauDsiw==", "dev": true, "license": "MIT", "dependencies": { - "@angular-eslint/bundled-angular-compiler": "18.4.1", + "@angular-eslint/bundled-angular-compiler": "18.4.3", "eslint-scope": "^8.0.2" }, "peerDependencies": { @@ -679,13 +617,13 @@ } }, "node_modules/@angular-eslint/utils": { - "version": "18.4.1", - "resolved": "https://registry.npmjs.org/@angular-eslint/utils/-/utils-18.4.1.tgz", - "integrity": "sha512-F5UGE1J/CRmTbl8vjexQRwRglNqnJwdXCUejaG+qlGssSHoWcRB+ubbR/na3PdnzeJdBE6DkLYElXnOQZ6YKfg==", + "version": "18.4.3", + "resolved": "https://registry.npmjs.org/@angular-eslint/utils/-/utils-18.4.3.tgz", + "integrity": "sha512-w0bJ9+ELAEiPBSTPPm9bvDngfu1d8JbzUhvs2vU+z7sIz/HMwUZT5S4naypj2kNN0gZYGYrW0lt+HIbW87zTAQ==", "dev": true, "license": "MIT", "dependencies": { - "@angular-eslint/bundled-angular-compiler": "18.4.1" + "@angular-eslint/bundled-angular-compiler": "18.4.3" }, "peerDependencies": { "@typescript-eslint/utils": "^7.11.0 || ^8.0.0", @@ -777,44 +715,6 @@ } } }, - "node_modules/@angular/build/node_modules/chokidar": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", - "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", - "dev": true, - "license": "MIT", - "dependencies": { - "anymatch": "~3.1.2", - "braces": "~3.0.2", - "glob-parent": "~5.1.2", - "is-binary-path": "~2.1.0", - "is-glob": "~4.0.1", - "normalize-path": "~3.0.0", - "readdirp": "~3.6.0" - }, - "engines": { - "node": ">= 8.10.0" - }, - "funding": { - "url": "https://paulmillr.com/funding/" - }, - "optionalDependencies": { - "fsevents": "~2.3.2" - } - }, - "node_modules/@angular/build/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, "node_modules/@angular/build/node_modules/immutable": { "version": "4.3.7", "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.3.7.tgz", @@ -822,32 +722,6 @@ "dev": true, "license": "MIT" }, - "node_modules/@angular/build/node_modules/readdirp": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", - "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", - "dev": true, - "license": "MIT", - "dependencies": { - "picomatch": "^2.2.1" - }, - "engines": { - "node": ">=8.10.0" - } - }, - "node_modules/@angular/build/node_modules/readdirp/node_modules/picomatch": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", - "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8.6" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, "node_modules/@angular/build/node_modules/sass": { "version": "1.77.6", "resolved": "https://registry.npmjs.org/sass/-/sass-1.77.6.tgz", @@ -867,9 +741,9 @@ } }, "node_modules/@angular/cdk": { - "version": "18.2.13", - "resolved": "https://registry.npmjs.org/@angular/cdk/-/cdk-18.2.13.tgz", - "integrity": "sha512-yBKoqcOwmwXnc5phFMEEMO130/Bz9beQLJrKzIS87f6TXaGCeBs4xrPHq2i7Xx/2TqvMiOD9ucjmlVbtGvNG3w==", + "version": "18.2.14", + "resolved": "https://registry.npmjs.org/@angular/cdk/-/cdk-18.2.14.tgz", + "integrity": "sha512-vDyOh1lwjfVk9OqoroZAP8pf3xxKUvyl+TVR8nJxL4c5fOfUFkD7l94HaanqKSRwJcI2xiztuu92IVoHn8T33Q==", "license": "MIT", "dependencies": { "tslib": "^2.3.0" @@ -981,6 +855,34 @@ "typescript": ">=5.4 <5.6" } }, + "node_modules/@angular/compiler-cli/node_modules/chokidar": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-4.0.3.tgz", + "integrity": "sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA==", + "license": "MIT", + "dependencies": { + "readdirp": "^4.0.1" + }, + "engines": { + "node": ">= 14.16.0" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/@angular/compiler-cli/node_modules/readdirp": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-4.0.2.tgz", + "integrity": "sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==", + "license": "MIT", + "engines": { + "node": ">= 14.16.0" + }, + "funding": { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + }, "node_modules/@angular/core": { "version": "18.2.13", "resolved": "https://registry.npmjs.org/@angular/core/-/core-18.2.13.tgz", @@ -1050,16 +952,16 @@ } }, "node_modules/@angular/material": { - "version": "18.2.13", - "resolved": "https://registry.npmjs.org/@angular/material/-/material-18.2.13.tgz", - "integrity": "sha512-Gxyyo6G+IXJwgf6zDTjPfFJ2PnjC2YXWKGkKKG2oR0jfiYiovDvNR4oXxhsztTwkaxLwck/gscoVTSQXMkU5fg==", + "version": "18.2.14", + "resolved": "https://registry.npmjs.org/@angular/material/-/material-18.2.14.tgz", + "integrity": "sha512-28pxzJP49Mymt664WnCtPkKeg7kXUsQKTKGf/Kl95rNTEdTJLbnlcc8wV0rT0yQNR7kXgpfBnG7h0ETLv/iu5Q==", "license": "MIT", "dependencies": { "tslib": "^2.3.0" }, "peerDependencies": { "@angular/animations": "^18.0.0 || ^19.0.0", - "@angular/cdk": "18.2.13", + "@angular/cdk": "18.2.14", "@angular/common": "^18.0.0 || ^19.0.0", "@angular/core": "^18.0.0 || ^19.0.0", "@angular/forms": "^18.0.0 || ^19.0.0", @@ -7381,6 +7283,12 @@ "@types/node": "*" } }, + "node_modules/@types/prop-types": { + "version": "15.7.14", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.14.tgz", + "integrity": "sha512-gNMvNH49DJ7OJYv+KAKn0Xp45p8PLl6zo2YnvDIbTd4J6MER2BmWN49TG7n9LvkyihINxeKW8+3bfS2yDC9dzQ==", + "license": "MIT" + }, "node_modules/@types/qs": { "version": "6.9.17", "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.17.tgz", @@ -7396,11 +7304,12 @@ "license": "MIT" }, "node_modules/@types/react": { - "version": "19.0.2", - "resolved": "https://registry.npmjs.org/@types/react/-/react-19.0.2.tgz", - "integrity": "sha512-USU8ZI/xyKJwFTpjSVIrSeHBVAGagkHQKPNbxeWwql/vDmnTIBgx+TJnhFnj1NXgz8XfprU0egV2dROLGpsBEg==", + "version": "18.3.18", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.18.tgz", + "integrity": "sha512-t4yC+vtgnkYjNSKlFx1jkAhH8LgTo2N/7Qvi83kdEaUtMDiwpbLAktKDaAMlRcJ5eSxZkH74eEGt1ky31d7kfQ==", "license": "MIT", "dependencies": { + "@types/prop-types": "*", "csstype": "^3.0.2" } }, @@ -9348,18 +9257,41 @@ } }, "node_modules/chokidar": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-4.0.3.tgz", - "integrity": "sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA==", + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", + "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", + "dev": true, "license": "MIT", "dependencies": { - "readdirp": "^4.0.1" + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" }, "engines": { - "node": ">= 14.16.0" + "node": ">= 8.10.0" }, "funding": { "url": "https://paulmillr.com/funding/" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/chokidar/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" } }, "node_modules/chownr": { @@ -12075,10 +12007,20 @@ "license": "MIT" }, "node_modules/fast-uri": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.0.3.tgz", - "integrity": "sha512-aLrHthzCjH5He4Z2H9YZ+v6Ujb9ocRuW6ZzkJQOrTxleEijANq4v1TsaPaVG1PZcuurEzrLcWRyYBYXD5cEiaw==", + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.0.4.tgz", + "integrity": "sha512-G3iTQw1DizJQ5eEqj1CbFCWhq+pzum7qepkxU7rS1FGZDqjYKcrguo9XDRbV7EgPnn8CgaPigTq+NEjyioeYZQ==", "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/fastify" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/fastify" + } + ], "license": "BSD-3-Clause" }, "node_modules/fastq": { @@ -18905,16 +18847,29 @@ "license": "MIT" }, "node_modules/readdirp": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-4.0.2.tgz", - "integrity": "sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==", + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, "license": "MIT", + "dependencies": { + "picomatch": "^2.2.1" + }, "engines": { - "node": ">= 14.16.0" + "node": ">=8.10.0" + } + }, + "node_modules/readdirp/node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" }, "funding": { - "type": "individual", - "url": "https://paulmillr.com/funding/" + "url": "https://github.com/sponsors/jonschlinkert" } }, "node_modules/redux": { @@ -19524,6 +19479,36 @@ } } }, + "node_modules/sass/node_modules/chokidar": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-4.0.3.tgz", + "integrity": "sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA==", + "dev": true, + "license": "MIT", + "dependencies": { + "readdirp": "^4.0.1" + }, + "engines": { + "node": ">= 14.16.0" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/sass/node_modules/readdirp": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-4.0.2.tgz", + "integrity": "sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 14.16.0" + }, + "funding": { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + }, "node_modules/sax": { "version": "1.4.1", "resolved": "https://registry.npmjs.org/sax/-/sax-1.4.1.tgz", @@ -21674,9 +21659,9 @@ } }, "node_modules/uuid": { - "version": "11.0.3", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.0.3.tgz", - "integrity": "sha512-d0z310fCWv5dJwnX1Y/MncBAqGMKEzlBb1AOf7z9K8ALnd0utBX/msg/fA0+sbyN1ihbMsLhrBlnl1ak7Wa0rg==", + "version": "11.0.4", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.0.4.tgz", + "integrity": "sha512-IzL6VtTTYcAhA/oghbFJ1Dkmqev+FpQWnCBaKq/gUluLxliWvO8DPFWfIviRmYbtaavtSQe4WBL++rFjdcGWEg==", "funding": [ "https://github.com/sponsors/broofa", "https://github.com/sponsors/ctavan" @@ -22668,44 +22653,6 @@ } } }, - "node_modules/webpack-dev-server/node_modules/chokidar": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", - "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", - "dev": true, - "license": "MIT", - "dependencies": { - "anymatch": "~3.1.2", - "braces": "~3.0.2", - "glob-parent": "~5.1.2", - "is-binary-path": "~2.1.0", - "is-glob": "~4.0.1", - "normalize-path": "~3.0.0", - "readdirp": "~3.6.0" - }, - "engines": { - "node": ">= 8.10.0" - }, - "funding": { - "url": "https://paulmillr.com/funding/" - }, - "optionalDependencies": { - "fsevents": "~2.3.2" - } - }, - "node_modules/webpack-dev-server/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, "node_modules/webpack-dev-server/node_modules/http-proxy-middleware": { "version": "2.0.7", "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-2.0.7.tgz", @@ -22731,32 +22678,6 @@ } } }, - "node_modules/webpack-dev-server/node_modules/picomatch": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", - "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8.6" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/webpack-dev-server/node_modules/readdirp": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", - "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", - "dev": true, - "license": "MIT", - "dependencies": { - "picomatch": "^2.2.1" - }, - "engines": { - "node": ">=8.10.0" - } - }, "node_modules/webpack-merge": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-6.0.1.tgz", diff --git a/package.json b/package.json index ceceebed252a..37a9cdd0a97a 100644 --- a/package.json +++ b/package.json @@ -14,13 +14,13 @@ ], "dependencies": { "@angular/animations": "18.2.13", - "@angular/cdk": "18.2.13", + "@angular/cdk": "18.2.14", "@angular/common": "18.2.13", "@angular/compiler": "18.2.13", "@angular/core": "18.2.13", "@angular/forms": "18.2.13", "@angular/localize": "18.2.13", - "@angular/material": "18.2.13", + "@angular/material": "18.2.14", "@angular/platform-browser": "18.2.13", "@angular/platform-browser-dynamic": "18.2.13", "@angular/router": "18.2.13", @@ -77,7 +77,7 @@ "ts-cacheable": "1.0.10", "tslib": "2.8.1", "turndown": "7.2.0", - "uuid": "11.0.3", + "uuid": "11.0.4", "webstomp-client": "1.2.6", "xlsx": "https://cdn.sheetjs.com/xlsx-0.20.3/xlsx-0.20.3.tgz", "zone.js": "0.14.10" @@ -110,11 +110,11 @@ "@analogjs/vite-plugin-angular": "1.11.0", "@angular-builders/jest": "18.0.0", "@angular-devkit/build-angular": "18.2.12", - "@angular-eslint/builder": "18.4.1", - "@angular-eslint/eslint-plugin": "18.4.1", - "@angular-eslint/eslint-plugin-template": "18.4.1", - "@angular-eslint/schematics": "18.4.1", - "@angular-eslint/template-parser": "18.4.1", + "@angular-eslint/builder": "18.4.3", + "@angular-eslint/eslint-plugin": "18.4.3", + "@angular-eslint/eslint-plugin-template": "18.4.3", + "@angular-eslint/schematics": "18.4.3", + "@angular-eslint/template-parser": "18.4.3", "@angular/cli": "18.2.12", "@angular/compiler-cli": "18.2.13", "@angular/language-service": "18.2.13", From 1c58a6da6dbb1bc4834a9fdd68777a26cb64e806 Mon Sep 17 00:00:00 2001 From: Paul Rangger <48455539+PaRangger@users.noreply.github.com> Date: Sun, 5 Jan 2025 20:44:55 +0100 Subject: [PATCH 17/34] Programming exercises: Improve task bar design (#10095) --- ...ise-instruction-step-wizard.component.html | 7 ++- ...rcise-instruction-step-wizard.component.ts | 4 +- ...ming-exercise-instruction-step-wizard.scss | 62 ++++++++++++++----- ...ise-instruction-task-status.component.html | 2 +- ...rcise-instruction-task-status.component.ts | 5 +- .../spec/helpers/sample/problemStatement.json | 4 +- 6 files changed, 56 insertions(+), 28 deletions(-) diff --git a/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.component.html b/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.component.html index 3ac4387efbfc..4ab5f509adf9 100644 --- a/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.component.html +++ b/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.component.html @@ -1,7 +1,8 @@
-
-
+
+
+
@for (step of steps; let i = $index; track i) {
@@ -21,7 +22,7 @@ } @if (step.done === TestCaseState.NOT_EXECUTED || step.done === TestCaseState.NO_RESULT) { - + }
diff --git a/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.component.ts b/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.component.ts index ae6ddf5ec31e..4bd4d7cea397 100644 --- a/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.component.ts +++ b/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.component.ts @@ -5,7 +5,7 @@ import { TaskArray } from 'app/exercises/programming/shared/instructions-render/ import { FeedbackComponent } from 'app/exercises/shared/feedback/feedback.component'; import { Exercise, ExerciseType } from 'app/entities/exercise.model'; import { Result } from 'app/entities/result.model'; -import { faCheck, faQuestion, faTimes } from '@fortawesome/free-solid-svg-icons'; +import { faCheck, faCircle, faTimes } from '@fortawesome/free-solid-svg-icons'; @Component({ selector: 'jhi-programming-exercise-instructions-step-wizard', @@ -24,7 +24,7 @@ export class ProgrammingExerciseInstructionStepWizardComponent implements OnChan // Icons faTimes = faTimes; faCheck = faCheck; - faQuestion = faQuestion; + faCircle = faCircle; constructor( private modalService: NgbModal, diff --git a/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.scss b/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.scss index 0166e9ca0dbb..0bafde8c3845 100644 --- a/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.scss +++ b/src/main/webapp/app/exercises/programming/shared/instructions-render/step-wizard/programming-exercise-instruction-step-wizard.scss @@ -1,7 +1,12 @@ +@import 'node_modules/bootstrap/scss/functions'; +@import 'node_modules/bootstrap/scss/variables'; +@import 'node_modules/bootstrap/scss/mixins'; + +$step-font-size: 0.6rem; +$step-button-size: 21px; // Make sure to keep it uneven so that centering works + .card-second-header { - padding: 1rem; - background-color: var(--programming-exercise-instruction-step-wizard-card-header-background); - border-bottom: 1px solid var(--programming-exercise-instruction-step-wizard-card-header-border); + padding: 1rem 1rem 0 1rem; } .stepwizard { @@ -18,7 +23,7 @@ .stepwizard-row { display: flex; flex-wrap: wrap; - justify-content: space-evenly; + justify-content: space-between; } .stepwizard-step { @@ -40,29 +45,43 @@ cursor: default; } + .btn { + cursor: default; + background-color: var(--bs-card-bg); + width: $step-button-size; + height: $step-button-size; + text-align: center; + padding: 0; + font-size: $step-font-size; + line-height: 1; + border-radius: 15px; + border-width: 2px; + border-color: $gray-600; + display: inline-flex; + align-items: center; + justify-content: center; + } + &--no-result.btn { cursor: default; } - &--failed.btn { + & &--failed.btn { cursor: pointer; + color: $danger; + border-color: $danger; } - .btn { - cursor: default; - background-color: var(--programming-exercise-instruction-step-wizard-card-header-background); - width: 30px; - height: 30px; - text-align: center; - padding: 6px 0; - font-size: 12px; - line-height: 1.428571429; - border-radius: 15px; - border-color: var(--programming-exercise-instruction-step-wizard-btn-border-color); + & &--success.btn { + cursor: pointer; + color: $success; + border-color: $success; } + & &--no-result.btn, .btn.not-done { - color: var(--programming-exercise-instruction-step-wizard-circle-not-done-color); + color: $gray-600; + font-size: 0.4rem; } } @@ -73,3 +92,12 @@ .test-passing:hover { background-color: var(--programming-exercise-instruction-step-wizard-test-passing-hover-color); } + +.stepwizard-row-line { + position: absolute; + top: 50%; + left: 0; + right: 0; + margin: 0; + border-top: 3px solid $gray-500; +} diff --git a/src/main/webapp/app/exercises/programming/shared/instructions-render/task/programming-exercise-instruction-task-status.component.html b/src/main/webapp/app/exercises/programming/shared/instructions-render/task/programming-exercise-instruction-task-status.component.html index 0277334c00d2..24a3764c7443 100644 --- a/src/main/webapp/app/exercises/programming/shared/instructions-render/task/programming-exercise-instruction-task-status.component.html +++ b/src/main/webapp/app/exercises/programming/shared/instructions-render/task/programming-exercise-instruction-task-status.component.html @@ -6,7 +6,7 @@ } @if (testCaseState === TestCaseState.NO_RESULT || testCaseState === TestCaseState.NOT_EXECUTED) { - + } @if (taskName) { diff --git a/src/main/webapp/app/exercises/programming/shared/instructions-render/task/programming-exercise-instruction-task-status.component.ts b/src/main/webapp/app/exercises/programming/shared/instructions-render/task/programming-exercise-instruction-task-status.component.ts index f30522e47281..e85002c954af 100644 --- a/src/main/webapp/app/exercises/programming/shared/instructions-render/task/programming-exercise-instruction-task-status.component.ts +++ b/src/main/webapp/app/exercises/programming/shared/instructions-render/task/programming-exercise-instruction-task-status.component.ts @@ -1,6 +1,5 @@ import { Component, Input } from '@angular/core'; -import { faCheckCircle, faTimesCircle } from '@fortawesome/free-regular-svg-icons'; -import { faQuestionCircle } from '@fortawesome/free-solid-svg-icons'; +import { faCheckCircle, faCircleDot, faTimesCircle } from '@fortawesome/free-regular-svg-icons'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; import { Exercise, ExerciseType } from 'app/entities/exercise.model'; import { Result } from 'app/entities/result.model'; @@ -41,7 +40,7 @@ export class ProgrammingExerciseInstructionTaskStatusComponent { hasMessage: boolean; // Icons - faQuestionCircle = faQuestionCircle; + faCircleDot = faCircleDot; farCheckCircle = faCheckCircle; farTimesCircle = faTimesCircle; diff --git a/src/test/javascript/spec/helpers/sample/problemStatement.json b/src/test/javascript/spec/helpers/sample/problemStatement.json index 6a9297f35013..19a5eb5ac1b3 100644 --- a/src/test/javascript/spec/helpers/sample/problemStatement.json +++ b/src/test/javascript/spec/helpers/sample/problemStatement.json @@ -7,8 +7,8 @@ "problemStatementBothFailedRendered": "
    \n
  1. Implement Bubble Sort: artemisApp.editor.testStatusLabels.noResult
    \nImplement the method performSort(List<Date>) in the class BubbleSort. Make sure to follow the Bubble Sort algorithm exactly.
  2. \n
  3. Implement Merge Sort: artemisApp.editor.testStatusLabels.noResult
    \nImplement the method performSort(List<Date>) in the class MergeSort. Make sure to follow the Merge Sort algorithm exactly.
  4. \n
\n", "problemStatementBothFailedHtml": "
    \n
  1. Implement Bubble Sort: artemisApp.editor.testStatusLabels.testFailing
    \nImplement the method performSort(List<Date>) in the class BubbleSort. Make sure to follow the Bubble Sort algorithm exactly.
  2. \n
  3. Implement Merge Sort: artemisApp.editor.testStatusLabels.testPassing
    \nImplement the method performSort(List<Date>) in the class MergeSort. Make sure to follow the Merge Sort algorithm exactly.
  4. \n
\n", "problemStatementBubbleSortFailsRendered": "
    \n
  1. Implement Bubble Sort: artemisApp.editor.testStatusLabels.noResult
    \nImplement the method performSort(List<Date>) in the class BubbleSort. Make sure to follow the Bubble Sort algorithm exactly.
  2. \n
  3. Implement Merge Sort: artemisApp.editor.testStatusLabels.noResult
    \nImplement the method performSort(List<Date>) in the class MergeSort. Make sure to follow the Merge Sort algorithm exactly.
  4. \n
\n", - "problemStatementBubbleSortNotExecutedHtml": "
    \n
  1. Implement Bubble SortartemisApp.editor.testStatusLabels.totalTestsPassing: [{\"totalTests\":1,\"passedTests\":0}]
    \nImplement the method performSort(List<Date>) in the class BubbleSort. Make sure to follow the Bubble Sort algorithm exactly.
  2. \n
  3. Implement Merge SortartemisApp.editor.testStatusLabels.totalTestsPassing: [{\"totalTests\":1,\"passedTests\":1}]
    \nImplement the method performSort(List<Date>) in the class MergeSort. Make sure to follow the Merge Sort algorithm exactly.
  4. \n
", + "problemStatementBubbleSortNotExecutedHtml": "
    \n
  1. Implement Bubble SortartemisApp.editor.testStatusLabels.totalTestsPassing: [{\"totalTests\":1,\"passedTests\":0}]
    \nImplement the method performSort(List<Date>) in the class BubbleSort. Make sure to follow the Bubble Sort algorithm exactly.
  2. \n
  3. Implement Merge SortartemisApp.editor.testStatusLabels.totalTestsPassing: [{\"totalTests\":1,\"passedTests\":1}]
    \nImplement the method performSort(List<Date>) in the class MergeSort. Make sure to follow the Merge Sort algorithm exactly.
  4. \n
", "problemStatementEmptySecondTask": "1. [task][Bubble Sort](1) \n Implement the method. \n 2. [task][Merge Sort]() \n Implement the method.", - "problemStatementEmptySecondTaskNotExecutedHtml": "
    \n
  1. Bubble SortartemisApp.editor.testStatusLabels.totalTestsPassing: [{\"totalTests\":1,\"passedTests\":1}]
    \nImplement the method.
  2. \n
  3. Merge SortartemisApp.editor.testStatusLabels.noTests
    \nImplement the method.
  4. \n
", + "problemStatementEmptySecondTaskNotExecutedHtml": "
    \n
  1. Bubble SortartemisApp.editor.testStatusLabels.totalTestsPassing: [{\"totalTests\":1,\"passedTests\":1}]
    \nImplement the method.
  2. \n
  3. Merge SortartemisApp.editor.testStatusLabels.noTests
    \nImplement the method.
  4. \n
", "problemStatementPlantUMLWithTest": "@startuml\nclass Policy {\n1)>+configure()\n2)>+testWithParenthesis()}\n@enduml" } From 24bab3c4bce83b9f64a7a512e8fd74b86bf6ce6f Mon Sep 17 00:00:00 2001 From: Simon Entholzer <33342534+SimonEntholzer@users.noreply.github.com> Date: Sun, 5 Jan 2025 20:46:52 +0100 Subject: [PATCH 18/34] Development: Add different execution modes for local playwright setup (#10064) --- supporting_scripts/playwright/README.md | 25 ++++++++++++++++--- .../playwright/runArtemisInDocker_linux.sh | 2 +- .../playwright/runArtemisInDocker_macOS.sh | 2 +- supporting_scripts/playwright/setupUsers.sh | 2 +- .../playwright/startPlaywright.sh | 6 ++--- .../playwright/startPlaywrightUI.sh | 16 ++++++++++++ 6 files changed, 43 insertions(+), 10 deletions(-) create mode 100755 supporting_scripts/playwright/startPlaywrightUI.sh diff --git a/supporting_scripts/playwright/README.md b/supporting_scripts/playwright/README.md index 3358943720f6..9a1160755d08 100644 --- a/supporting_scripts/playwright/README.md +++ b/supporting_scripts/playwright/README.md @@ -16,10 +16,27 @@ In case you stop the client, you can simply re-run it at the root of the Artemis ## 2. Setup users Playwright needs users for it's tests. If you do not have users set up, you can simply do so by running: -`setupUsers.sh` + +```bash +setupUsers.sh +``` + This will create 20 test users. -## 3. Setup Playwright and run Playwright in UI-mode +## 3. Setup Playwright and run Playwright tests + +You can run Playwright tests in two different ways: running all tests or running in UI mode. + +### Running All Tests +The `startPlaywright.sh` script runs the full suite of Playwright tests in a headless mode, outputting the results to the command line. +- Executes all test cases defined in the Playwright test suite. +- Runs in a headless environment for faster execution. +- Outputs test results, including logs, in the terminal. + +### Running Tests in UI mode +The `startPlaywrightUI.sh` script starts Playwright in a graphical mode for debugging and interactive test execution. +- Launches a browser window to display available test cases. +- Allows manual selection and execution of individual or multiple tests. +- Provides real-time debugging features, such as visual test steps and screenshots. +- Useful for debugging and inspecting browser behavior during test execution. -Simply run: `startPlaywright.sh`. This will install the necessary dependencies for playwright and start it in UI mode. -If you already have playwright installed, you can also start playwright directly from the `src/test/playwright` directory with `npm run playwright:open`. diff --git a/supporting_scripts/playwright/runArtemisInDocker_linux.sh b/supporting_scripts/playwright/runArtemisInDocker_linux.sh index c979ea2d32b1..1bac4655fa2c 100755 --- a/supporting_scripts/playwright/runArtemisInDocker_linux.sh +++ b/supporting_scripts/playwright/runArtemisInDocker_linux.sh @@ -2,7 +2,7 @@ set -e -artemis_path="$(readlink -f "$(dirname $0)/../..")" +artemis_path="$(readlink -f "$(dirname "$0")/../..")" cd "$artemis_path/docker" diff --git a/supporting_scripts/playwright/runArtemisInDocker_macOS.sh b/supporting_scripts/playwright/runArtemisInDocker_macOS.sh index ab8dc424e740..9a084684d74c 100755 --- a/supporting_scripts/playwright/runArtemisInDocker_macOS.sh +++ b/supporting_scripts/playwright/runArtemisInDocker_macOS.sh @@ -2,7 +2,7 @@ set -e -artemis_path="$(readlink -f "$(dirname $0)/../..")" +artemis_path="$(readlink -f "$(dirname "$0")/../..")" cd "$artemis_path/docker" open -a Docker diff --git a/supporting_scripts/playwright/setupUsers.sh b/supporting_scripts/playwright/setupUsers.sh index cab70289a9d7..ea235692db48 100755 --- a/supporting_scripts/playwright/setupUsers.sh +++ b/supporting_scripts/playwright/setupUsers.sh @@ -3,7 +3,7 @@ # We use the supporting scripts to create users set -e -artemis_path="$(readlink -f "$(dirname $0)/../..")" +artemis_path="$(readlink -f "$(dirname "$0")/../..")" cd "$artemis_path/supporting_scripts" diff --git a/supporting_scripts/playwright/startPlaywright.sh b/supporting_scripts/playwright/startPlaywright.sh index 412b88919978..77d2832e585e 100755 --- a/supporting_scripts/playwright/startPlaywright.sh +++ b/supporting_scripts/playwright/startPlaywright.sh @@ -2,7 +2,7 @@ set -e -artemis_path="$(readlink -f "$(dirname $0)/../..")" +artemis_path="$(readlink -f "$(dirname "$0")/../..")" echo "Installing Playwright and dependencies" @@ -12,5 +12,5 @@ npm install npm run playwright:setup-local || true -echo "Starting Playwright in UI mode" -npm run playwright:open +echo "Run all playwright tests" +npm run playwright:test diff --git a/supporting_scripts/playwright/startPlaywrightUI.sh b/supporting_scripts/playwright/startPlaywrightUI.sh new file mode 100755 index 000000000000..0abe6e8bd2c6 --- /dev/null +++ b/supporting_scripts/playwright/startPlaywrightUI.sh @@ -0,0 +1,16 @@ +#!/bin/sh + +set -e + +artemis_path="$(readlink -f "$(dirname "$0")/../..")" + +echo "Installing Playwright and dependencies" + +cd "$artemis_path/src/test/playwright" + +npm install + +npm run playwright:setup-local || true + +echo "Start Playwright in UI mode" +npm run playwright:open From e12d70ea9972b86d1e4986f46d631a558592314b Mon Sep 17 00:00:00 2001 From: Michal Kawka <73854755+coolchock@users.noreply.github.com> Date: Sun, 5 Jan 2025 20:52:50 +0100 Subject: [PATCH 19/34] Development: Migrate exam scores module to signals, inject and standalone (#9921) --- ...scores-average-scores-graph.component.html | 2 +- ...m-scores-average-scores-graph.component.ts | 38 ++++++++-------- .../exam-scores/exam-scores.component.html | 4 +- .../exam/exam-scores/exam-scores.component.ts | 36 +++++++++------- .../exam/exam-scores/exam-scores.module.ts | 28 ------------ .../app/exam/exam-scores/exam-scores.route.ts | 22 +++++----- .../app/exam/manage/exam-management.module.ts | 5 +-- ...res-average-scores-graph.component.spec.ts | 33 +++++++------- .../exam-scores/exam-scores.component.spec.ts | 43 ++++++++----------- 9 files changed, 94 insertions(+), 117 deletions(-) delete mode 100644 src/main/webapp/app/exam/exam-scores/exam-scores.module.ts diff --git a/src/main/webapp/app/exam/exam-scores/exam-scores-average-scores-graph.component.html b/src/main/webapp/app/exam/exam-scores/exam-scores-average-scores-graph.component.html index 35ab48c1854a..958122437f2b 100644 --- a/src/main/webapp/app/exam/exam-scores/exam-scores-average-scores-graph.component.html +++ b/src/main/webapp/app/exam/exam-scores/exam-scores-average-scores-graph.component.html @@ -1,6 +1,6 @@
-
+
(); + course = input.required(); courseId: number; examId: number; @@ -39,14 +49,6 @@ export class ExamScoresAverageScoresGraphComponent implements OnInit { xScaleMax = 100; lookup: NameToValueMap = {}; - constructor( - private navigationUtilService: ArtemisNavigationUtilService, - private activatedRoute: ActivatedRoute, - private service: StatisticsService, - private translateService: TranslateService, - private localeConversionService: LocaleConversionService, - ) {} - ngOnInit(): void { this.activatedRoute.params.subscribe((params) => { this.courseId = +params['courseId']; @@ -56,12 +58,12 @@ export class ExamScoresAverageScoresGraphComponent implements OnInit { } private initializeChart(): void { - this.lookup[this.averageScores.title] = { absoluteValue: this.averageScores.averagePoints! }; - const exerciseGroupAverage = this.averageScores.averagePercentage ? this.averageScores.averagePercentage : 0; - this.ngxData.push({ name: this.averageScores.title, value: exerciseGroupAverage }); + this.lookup[this.averageScores().title] = { absoluteValue: this.averageScores().averagePoints! }; + const exerciseGroupAverage = this.averageScores().averagePercentage ?? 0; + this.ngxData.push({ name: this.averageScores().title, value: exerciseGroupAverage }); this.ngxColor.domain.push(this.determineColor(true, exerciseGroupAverage)); this.xScaleMax = this.xScaleMax > exerciseGroupAverage ? this.xScaleMax : exerciseGroupAverage; - this.averageScores.exerciseResults.forEach((exercise) => { + this.averageScores().exerciseResults.forEach((exercise) => { const exerciseAverage = exercise.averagePercentage ?? 0; this.xScaleMax = this.xScaleMax > exerciseAverage ? this.xScaleMax : exerciseAverage; this.ngxData.push({ name: exercise.exerciseId + ' ' + exercise.title, value: exerciseAverage }); @@ -77,14 +79,14 @@ export class ExamScoresAverageScoresGraphComponent implements OnInit { } roundAndPerformLocalConversion(points: number | undefined) { - return this.localeConversionService.toLocaleString(roundValueSpecifiedByCourseSettings(points, this.course), this.course!.accuracyOfScores!); + return this.localeConversionService.toLocaleString(roundValueSpecifiedByCourseSettings(points, this.course()), this.course()!.accuracyOfScores!); } /** * We navigate to the exercise scores page when the user clicks on a data point */ navigateToExercise(exerciseId: number, exerciseType: ExerciseType) { - navigateToExamExercise(this.navigationUtilService, this.courseId, this.examId, this.averageScores.exerciseGroupId, exerciseType, exerciseId, 'scores'); + navigateToExamExercise(this.navigationUtilService, this.courseId, this.examId, this.averageScores().exerciseGroupId, exerciseType, exerciseId, 'scores'); } /** diff --git a/src/main/webapp/app/exam/exam-scores/exam-scores.component.html b/src/main/webapp/app/exam/exam-scores/exam-scores.component.html index 2fe2e07c3e3a..40b23cc91347 100644 --- a/src/main/webapp/app/exam/exam-scores/exam-scores.component.html +++ b/src/main/webapp/app/exam/exam-scores/exam-scores.component.html @@ -35,11 +35,11 @@

@if (examScoreDTO.maxPoints) {

{{ 'artemisApp.examScores.maxPoints' | artemisTranslate }}: {{ localize(examScoreDTO.maxPoints) }},
} - +
{{ exerciseGroups.length }} {{ 'artemisApp.examScores.noExerciseGroups' | artemisTranslate }}
,
- +
{{ aggregatedExamResults.noOfRegisteredUsers }} {{ 'artemisApp.examScores.registered' | artemisTranslate }}
diff --git a/src/main/webapp/app/exam/exam-scores/exam-scores.component.ts b/src/main/webapp/app/exam/exam-scores/exam-scores.component.ts index 94a6f375b3b7..97bb7fb19d39 100644 --- a/src/main/webapp/app/exam/exam-scores/exam-scores.component.ts +++ b/src/main/webapp/app/exam/exam-scores/exam-scores.component.ts @@ -1,8 +1,8 @@ -import { ChangeDetectionStrategy, ChangeDetectorRef, Component, OnDestroy, OnInit } from '@angular/core'; +import { ChangeDetectionStrategy, ChangeDetectorRef, Component, OnDestroy, OnInit, inject } from '@angular/core'; import { Subscription, forkJoin, of } from 'rxjs'; import { catchError } from 'rxjs/operators'; import { ExamManagementService } from 'app/exam/manage/exam-management.service'; -import { ActivatedRoute } from '@angular/router'; +import { ActivatedRoute, RouterLink } from '@angular/router'; import { SortService } from 'app/shared/service/sort.service'; import { download, generateCsv, mkConfig } from 'export-to-csv'; import { @@ -58,6 +58,11 @@ import { USERNAME_KEY, } from 'app/shared/export/export-constants'; import { BonusStrategy } from 'app/entities/bonus.model'; +import { ExamScoresAverageScoresGraphComponent } from 'app/exam/exam-scores/exam-scores-average-scores-graph.component'; +import { ArtemisParticipantScoresModule } from 'app/shared/participant-scores/participant-scores.module'; +import { ExportModule } from 'app/shared/export/export.module'; +import { ArtemisSharedCommonModule } from 'app/shared/shared-common.module'; +import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; export enum MedianType { PASSED, @@ -70,8 +75,21 @@ export enum MedianType { templateUrl: './exam-scores.component.html', changeDetection: ChangeDetectionStrategy.OnPush, styleUrls: ['./exam-scores.component.scss', '../../shared/chart/vertical-bar-chart.scss'], + standalone: true, + imports: [RouterLink, ArtemisSharedComponentModule, ArtemisSharedCommonModule, ExamScoresAverageScoresGraphComponent, ArtemisParticipantScoresModule, ExportModule], }) export class ExamScoresComponent implements OnInit, OnDestroy { + private route = inject(ActivatedRoute); + private examService = inject(ExamManagementService); + private sortService = inject(SortService); + private alertService = inject(AlertService); + private changeDetector = inject(ChangeDetectorRef); + private languageHelper = inject(JhiLanguageHelper); + private localeConversionService = inject(LocaleConversionService); + private participantScoresService = inject(ParticipantScoresService); + private gradingSystemService = inject(GradingSystemService); + private courseManagementService = inject(CourseManagementService); + public examScoreDTO: ExamScoreDTO; public exerciseGroups: ExerciseGroup[]; public studentResults: StudentResult[]; @@ -130,18 +148,6 @@ export class ExamScoresComponent implements OnInit, OnDestroy { faExclamationTriangle = faExclamationTriangle; private languageChangeSubscription?: Subscription; - constructor( - private route: ActivatedRoute, - private examService: ExamManagementService, - private sortService: SortService, - private alertService: AlertService, - private changeDetector: ChangeDetectorRef, - private languageHelper: JhiLanguageHelper, - private localeConversionService: LocaleConversionService, - private participantScoresService: ParticipantScoresService, - private gradingSystemService: GradingSystemService, - private courseManagementService: CourseManagementService, - ) {} ngOnInit() { this.route.params.subscribe((params) => { @@ -676,7 +682,7 @@ export class ExamScoresComponent implements OnInit, OnDestroy { * Localizes a number, e.g. switching the decimal separator */ localize(numberToLocalize: number): string { - return this.localeConversionService.toLocaleString(numberToLocalize, this.course!.accuracyOfScores!); + return this.localeConversionService.toLocaleString(numberToLocalize, this.course?.accuracyOfScores); } /** diff --git a/src/main/webapp/app/exam/exam-scores/exam-scores.module.ts b/src/main/webapp/app/exam/exam-scores/exam-scores.module.ts deleted file mode 100644 index e439030cb776..000000000000 --- a/src/main/webapp/app/exam/exam-scores/exam-scores.module.ts +++ /dev/null @@ -1,28 +0,0 @@ -import { NgModule } from '@angular/core'; -import { ExamScoresComponent } from './exam-scores.component'; -import { ArtemisSharedModule } from 'app/shared/shared.module'; -import { ArtemisExamScoresRoutingModule } from 'app/exam/exam-scores/exam-scores.route'; -import { ArtemisDataTableModule } from 'app/shared/data-table/data-table.module'; -import { NgxDatatableModule } from '@siemens/ngx-datatable'; -import { ArtemisResultModule } from 'app/exercises/shared/result/result.module'; -import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; -import { ExamScoresAverageScoresGraphComponent } from 'app/exam/exam-scores/exam-scores-average-scores-graph.component'; -import { BarChartModule } from '@swimlane/ngx-charts'; -import { ArtemisParticipantScoresModule } from 'app/shared/participant-scores/participant-scores.module'; -import { ExportModule } from 'app/shared/export/export.module'; - -@NgModule({ - declarations: [ExamScoresComponent, ExamScoresAverageScoresGraphComponent], - imports: [ - ArtemisSharedModule, - ArtemisExamScoresRoutingModule, - ArtemisDataTableModule, - NgxDatatableModule, - ArtemisResultModule, - ArtemisSharedComponentModule, - BarChartModule, - ArtemisParticipantScoresModule, - ExportModule, - ], -}) -export class ArtemisExamScoresModule {} diff --git a/src/main/webapp/app/exam/exam-scores/exam-scores.route.ts b/src/main/webapp/app/exam/exam-scores/exam-scores.route.ts index 2635eae62e5f..c0b815477e42 100644 --- a/src/main/webapp/app/exam/exam-scores/exam-scores.route.ts +++ b/src/main/webapp/app/exam/exam-scores/exam-scores.route.ts @@ -1,13 +1,21 @@ -import { NgModule } from '@angular/core'; -import { RouterModule, Routes } from '@angular/router'; -import { UserRouteAccessService } from 'app/core/auth/user-route-access-service'; +import { Route, Routes } from '@angular/router'; import { ExamScoresComponent } from 'app/exam/exam-scores/exam-scores.component'; import { Authority } from 'app/shared/constants/authority.constants'; +import { UserRouteAccessService } from 'app/core/auth/user-route-access-service'; -const routes: Routes = [ +export const examScoresRoute: Route[] = [ { path: ':examId/scores', component: ExamScoresComponent, + }, +]; + +const EXAM_SCORES_ROUTES = [...examScoresRoute]; + +export const examScoresState: Routes = [ + { + path: '', + children: EXAM_SCORES_ROUTES, data: { authorities: [Authority.ADMIN, Authority.INSTRUCTOR], pageTitle: 'artemisApp.examScores.title', @@ -15,9 +23,3 @@ const routes: Routes = [ canActivate: [UserRouteAccessService], }, ]; - -@NgModule({ - imports: [RouterModule.forChild(routes)], - exports: [RouterModule], -}) -export class ArtemisExamScoresRoutingModule {} diff --git a/src/main/webapp/app/exam/manage/exam-management.module.ts b/src/main/webapp/app/exam/manage/exam-management.module.ts index 5590adf04e27..e95bb9d2d5ee 100644 --- a/src/main/webapp/app/exam/manage/exam-management.module.ts +++ b/src/main/webapp/app/exam/manage/exam-management.module.ts @@ -1,6 +1,5 @@ import { NgModule } from '@angular/core'; import { RouterModule } from '@angular/router'; -import { ArtemisExamScoresModule } from 'app/exam/exam-scores/exam-scores.module'; import { ExamManagementComponent } from 'app/exam/manage/exam-management.component'; import { examManagementState } from 'app/exam/manage/exam-management.route'; import { ExamUpdateComponent } from 'app/exam/manage/exams/exam-update.component'; @@ -67,9 +66,10 @@ import { ArtemisProgrammingExerciseModule } from 'app/exercises/programming/shar import { DetailModule } from 'app/detail-overview-list/detail.module'; import { ArtemisDurationFromSecondsPipe } from 'app/shared/pipes/artemis-duration-from-seconds.pipe'; import { NoDataComponent } from 'app/shared/no-data-component'; +import { examScoresState } from 'app/exam/exam-scores/exam-scores.route'; import { GitDiffLineStatComponent } from 'app/exercises/programming/git-diff-report/git-diff-line-stat.component'; -const ENTITY_STATES = [...examManagementState]; +const ENTITY_STATES = [...examManagementState, ...examScoresState]; @NgModule({ // TODO: For better modularization we could define an exercise module with the corresponding exam routes @@ -77,7 +77,6 @@ const ENTITY_STATES = [...examManagementState]; imports: [ RouterModule.forChild(ENTITY_STATES), ArtemisTextExerciseModule, - ArtemisExamScoresModule, ArtemisSharedModule, FormDateTimePickerModule, ArtemisSharedComponentModule, diff --git a/src/test/javascript/spec/component/exam/exam-scores/exam-scores-average-scores-graph.component.spec.ts b/src/test/javascript/spec/component/exam/exam-scores/exam-scores-average-scores-graph.component.spec.ts index f962626e8e37..49da1a162d28 100644 --- a/src/test/javascript/spec/component/exam/exam-scores/exam-scores-average-scores-graph.component.spec.ts +++ b/src/test/javascript/spec/component/exam/exam-scores/exam-scores-average-scores-graph.component.spec.ts @@ -1,6 +1,6 @@ import { ComponentFixture, TestBed } from '@angular/core/testing'; import { TranslateService } from '@ngx-translate/core'; -import { MockDirective, MockModule, MockPipe, MockProvider } from 'ng-mocks'; +import { MockDirective, MockModule, MockProvider } from 'ng-mocks'; import { of } from 'rxjs'; import { HttpResponse } from '@angular/common/http'; import { ExamScoresAverageScoresGraphComponent } from 'app/exam/exam-scores/exam-scores-average-scores-graph.component'; @@ -9,7 +9,6 @@ import { MockTranslateService } from '../../../helpers/mocks/service/mock-transl import { AggregatedExerciseGroupResult, AggregatedExerciseResult } from 'app/exam/exam-scores/exam-score-dtos.model'; import { CourseManagementService } from 'app/course/manage/course-management.service'; import { BarChartModule } from '@swimlane/ngx-charts'; -import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; import { GraphColors } from 'app/entities/statistics.model'; import { NgxChartsSingleSeriesDataEntry } from 'app/shared/chart/ngx-charts-datatypes'; import { ExerciseType } from 'app/entities/exercise.model'; @@ -56,7 +55,7 @@ describe('ExamScoresAverageScoresGraphComponent', () => { beforeEach(() => { TestBed.configureTestingModule({ imports: [ArtemisTestModule, MockModule(BarChartModule), RouterModule.forRoot([])], - declarations: [ExamScoresAverageScoresGraphComponent, MockPipe(ArtemisTranslatePipe), MockDirective(TranslateDirective)], + declarations: [ExamScoresAverageScoresGraphComponent, MockDirective(TranslateDirective)], providers: [ MockProvider(CourseManagementService, { find: () => { @@ -70,16 +69,14 @@ describe('ExamScoresAverageScoresGraphComponent', () => { }), { provide: TranslateService, useClass: MockTranslateService }, ], - }) - .compileComponents() - .then(() => { - fixture = TestBed.createComponent(ExamScoresAverageScoresGraphComponent); - component = fixture.componentInstance; - navigateToExerciseMock = jest.spyOn(component, 'navigateToExercise').mockImplementation(); - - component.averageScores = returnValue; - fixture.detectChanges(); - }); + }).compileComponents(); + + fixture = TestBed.createComponent(ExamScoresAverageScoresGraphComponent); + component = fixture.componentInstance; + navigateToExerciseMock = jest.spyOn(component, 'navigateToExercise').mockImplementation(); + + fixture.componentRef.setInput('averageScores', returnValue); + fixture.detectChanges(); }); it('should set ngx data objects and bar colors correctly', () => { @@ -99,8 +96,9 @@ describe('ExamScoresAverageScoresGraphComponent', () => { }); const adaptExpectedData = (averagePoints: number, newColor: string, expectedColorDomain: string[], expectedData: NgxChartsSingleSeriesDataEntry[]) => { - component.averageScores.averagePoints = averagePoints; - component.averageScores.averagePercentage = averagePoints * 10; + component.averageScores().averagePoints = averagePoints; + component.averageScores().averagePercentage = averagePoints * 10; + expectedColorDomain[0] = newColor; expectedData[0].value = averagePoints * 10; component.ngxColor.domain = []; @@ -146,7 +144,10 @@ describe('ExamScoresAverageScoresGraphComponent', () => { it('should look up absolute value', () => { const roundAndPerformLocalConversionSpy = jest.spyOn(component, 'roundAndPerformLocalConversion'); - component.course = { accuracyOfScores: 2 }; + const updatedCourse = { + accuracyOfScores: 2, + }; + fixture.componentRef.setInput('course', updatedCourse); component.lookup['test'] = { absoluteValue: 40 }; const result = component.lookupAbsoluteValue('test'); diff --git a/src/test/javascript/spec/component/exam/exam-scores/exam-scores.component.spec.ts b/src/test/javascript/spec/component/exam/exam-scores/exam-scores.component.spec.ts index df9ddaa86480..3d82fa3ba673 100644 --- a/src/test/javascript/spec/component/exam/exam-scores/exam-scores.component.spec.ts +++ b/src/test/javascript/spec/component/exam/exam-scores/exam-scores.component.spec.ts @@ -1,5 +1,6 @@ -import { HttpResponse } from '@angular/common/http'; +import { HttpResponse, provideHttpClient } from '@angular/common/http'; import { ComponentFixture, TestBed } from '@angular/core/testing'; +import { provideHttpClientTesting } from '@angular/common/http/testing'; import { ActivatedRoute, Router } from '@angular/router'; import { FaIconComponent } from '@fortawesome/angular-fontawesome'; import { TranslateService } from '@ngx-translate/core'; @@ -14,15 +15,14 @@ import { ExerciseResult, StudentResult, } from 'app/exam/exam-scores/exam-score-dtos.model'; +import { MockComponent, MockDirective, MockModule, MockPipe, MockProvider } from 'ng-mocks'; +import { BrowserAnimationsModule } from '@angular/platform-browser/animations'; import { ExamScoresComponent, MedianType } from 'app/exam/exam-scores/exam-scores.component'; import { ExamManagementService } from 'app/exam/manage/exam-management.service'; -import { HelpIconComponent } from 'app/shared/components/help-icon.component'; -import { DeleteButtonDirective } from 'app/shared/delete-dialog/delete-button.directive'; import { ParticipantScoresService, ScoresDTO } from 'app/shared/participant-scores/participant-scores.service'; import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; import { SortService } from 'app/shared/service/sort.service'; import { cloneDeep } from 'lodash-es'; -import { MockComponent, MockDirective, MockPipe, MockProvider } from 'ng-mocks'; import { EMPTY, of } from 'rxjs'; import { GradingSystemService } from 'app/grading-system/grading-system.service'; import { GradingScale } from 'app/entities/grading-scale.model'; @@ -35,7 +35,6 @@ import { CourseManagementService } from 'app/course/manage/course-management.ser import { MockRouter } from '../../../helpers/mocks/mock-router'; import { AccountService } from 'app/core/auth/account.service'; import { MockRouterLinkDirective } from '../../../helpers/mocks/directive/mock-router-link.directive'; -import { ParticipantScoresDistributionComponent } from 'app/shared/participant-scores/participant-scores-distribution/participant-scores-distribution.component'; import { LocaleConversionService } from 'app/shared/service/locale-conversion.service'; import { ArtemisNavigationUtilService } from 'app/utils/navigation.utils'; import { CsvDecimalSeparator, CsvExportOptions, CsvFieldSeparator, CsvQuoteStrings } from 'app/shared/export/export-modal.component'; @@ -55,9 +54,9 @@ import { REGISTRATION_NUMBER_KEY, USERNAME_KEY, } from 'app/shared/export/export-constants'; -import { ExportButtonComponent } from 'app/shared/export/export-button.component'; import { PlagiarismVerdict } from 'app/exercises/shared/plagiarism/types/PlagiarismVerdict'; import { BonusStrategy } from 'app/entities/bonus.model'; +import { MockTranslateService } from '../../../helpers/mocks/service/mock-translate.service'; describe('ExamScoresComponent', () => { let fixture: ComponentFixture; @@ -276,30 +275,28 @@ describe('ExamScoresComponent', () => { beforeEach(() => { TestBed.configureTestingModule({ + imports: [MockModule(BrowserAnimationsModule)], declarations: [ ExamScoresComponent, MockPipe(ArtemisTranslatePipe), MockComponent(FaIconComponent), - MockComponent(HelpIconComponent), - MockComponent(ExportButtonComponent), MockDirective(TranslateDirective), MockDirective(SortByDirective), MockDirective(SortDirective), - MockDirective(DeleteButtonDirective), MockComponent(ExamScoresAverageScoresGraphComponent), MockRouterLinkDirective, - MockComponent(ParticipantScoresDistributionComponent), ], providers: [ { provide: ActivatedRoute, useValue: { params: of({ courseId: 1, examId: 1 }) } }, { provide: Router, useClass: MockRouter }, + { provide: TranslateService, useClass: MockTranslateService }, + provideHttpClient(), + provideHttpClientTesting(), MockProvider(AccountService), MockProvider(ArtemisNavigationUtilService), - MockProvider(TranslateService), MockProvider(ExamManagementService), MockProvider(SortService), MockProvider(AlertService), - MockProvider(ParticipantScoresService), MockProvider(GradingSystemService, { findGradingScaleForExam: () => { return of( @@ -328,18 +325,16 @@ describe('ExamScoresComponent', () => { }, }), ], - }) - .compileComponents() - .then(() => { - fixture = TestBed.createComponent(ExamScoresComponent); - comp = fixture.componentInstance; - examService = fixture.debugElement.injector.get(ExamManagementService); - gradingSystemService = fixture.debugElement.injector.get(GradingSystemService); - const participationScoreService = fixture.debugElement.injector.get(ParticipantScoresService); - findExamScoresSpy = jest - .spyOn(participationScoreService, 'findExamScores') - .mockReturnValue(of(new HttpResponse({ body: [examScoreStudent1, examScoreStudent2, examScoreStudent3] }))); - }); + }).compileComponents(); + + fixture = TestBed.createComponent(ExamScoresComponent); + comp = fixture.componentInstance; + examService = fixture.debugElement.injector.get(ExamManagementService); + gradingSystemService = fixture.debugElement.injector.get(GradingSystemService); + const participationScoreService = fixture.debugElement.injector.get(ParticipantScoresService); + findExamScoresSpy = jest + .spyOn(participationScoreService, 'findExamScores') + .mockReturnValue(of(new HttpResponse({ body: [examScoreStudent1, examScoreStudent2, examScoreStudent3] }))); }); afterEach(() => { From ec1f0dfe956bac1fb8d8b25688d48396f2b2120e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Christoph=20Kn=C3=B6dlseder?= <53149143+chrisknedl@users.noreply.github.com> Date: Sun, 5 Jan 2025 20:53:57 +0100 Subject: [PATCH 20/34] Assessment: Keep long feedback after saving an assessment (#10090) --- .../LongFeedbackTextRepository.java | 4 ++ .../assessment/service/ResultService.java | 21 ++++---- .../service/ProgrammingAssessmentService.java | 2 - ...ProgrammingIntegrationIndependentTest.java | 4 ++ .../ProgrammingAssessmentIntegrationTest.java | 51 ++++++++++++++++++- 5 files changed, 70 insertions(+), 12 deletions(-) diff --git a/src/main/java/de/tum/cit/aet/artemis/assessment/repository/LongFeedbackTextRepository.java b/src/main/java/de/tum/cit/aet/artemis/assessment/repository/LongFeedbackTextRepository.java index 4e0c502a75db..f3a8909a6978 100644 --- a/src/main/java/de/tum/cit/aet/artemis/assessment/repository/LongFeedbackTextRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/assessment/repository/LongFeedbackTextRepository.java @@ -52,6 +52,10 @@ public interface LongFeedbackTextRepository extends ArtemisJpaRepository feedbackIds); + @Modifying + @Transactional + void deleteByFeedbackId(final Long feedbackId); + default LongFeedbackText findByFeedbackIdWithFeedbackAndResultAndParticipationElseThrow(final Long feedbackId) { return getValueElseThrow(findWithFeedbackAndResultAndParticipationByFeedbackId(feedbackId), feedbackId); } diff --git a/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java b/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java index 92b1f6e19030..7c91008842c4 100644 --- a/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java +++ b/src/main/java/de/tum/cit/aet/artemis/assessment/service/ResultService.java @@ -522,23 +522,26 @@ private void handleFeedbackPersistence(Feedback feedback, Result result, Map feedbacks = new ArrayList<>(); var manualLongFeedback = new Feedback().credits(1.00).type(FeedbackType.MANUAL_UNREFERENCED); manualLongFeedback.setDetailText("abc".repeat(5000)); @@ -553,6 +554,54 @@ void updateManualProgrammingExerciseResult_addFeedbackAfterManualLongFeedback() assertThat(savedAutomaticLongFeedback.getDetailText()).isEqualTo(manualLongFeedback.getDetailText()); } + @ParameterizedTest + @ValueSource(booleans = { true, false }) + @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") + void shouldKeepExistingLongFeedbackWhenSavingAnAssessment(boolean submit) throws Exception { + var manualLongFeedback = new Feedback().credits(0.0); + var longText = "abc".repeat(5000); + manualLongFeedback.setDetailText(longText); + var result = new Result().feedbacks(List.of(manualLongFeedback)).score(0.0); + result = resultRepository.save(result); + + LinkedMultiValueMap params = new LinkedMultiValueMap<>(); + params.add("submit", String.valueOf(submit)); + result = request.putWithResponseBodyAndParams("/api/participations/" + programmingExerciseStudentParticipation.getId() + "/manual-results", result, Result.class, + HttpStatus.OK, params); + + var longFeedbackText = longFeedbackTextRepository.findByFeedbackId(result.getFeedbacks().getFirst().getId()); + assertThat(longFeedbackText).isPresent(); + assertThat(longFeedbackText.get().getText()).isEqualTo(longText); + } + + @ParameterizedTest + @ValueSource(booleans = { true, false }) + @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") + void shouldUpdateUnreferencedLongFeedbackWhenSavingAnAssessment(boolean submit) throws Exception { + var manualLongFeedback = new Feedback().credits(0.0).type(FeedbackType.MANUAL_UNREFERENCED); + var longText = "abc".repeat(5000); + manualLongFeedback.setDetailText(longText); + var result = new Result().feedbacks(List.of(manualLongFeedback)).score(0.0); + result = resultRepository.save(result); + + var newLongText = "def".repeat(5000); + manualLongFeedback = result.getFeedbacks().getFirst(); + + // The actual complete longtext is still stored in the detailText field when the result is sent from the client + var detailText = Feedback.class.getDeclaredField("detailText"); + detailText.setAccessible(true); + detailText.set(manualLongFeedback, newLongText); + + LinkedMultiValueMap params = new LinkedMultiValueMap<>(); + params.add("submit", String.valueOf(submit)); + result = request.putWithResponseBodyAndParams("/api/participations/" + programmingExerciseStudentParticipation.getId() + "/manual-results", result, Result.class, + HttpStatus.OK, params); + + var longFeedbackText = longFeedbackTextRepository.findByFeedbackId(result.getFeedbacks().getFirst().getId()); + assertThat(longFeedbackText).isPresent(); + assertThat(longFeedbackText.get().getText()).isEqualTo(newLongText); + } + @Test @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") void updateManualProgrammingExerciseResult_addFeedbackAfterAutomaticLongFeedback() throws Exception { From 60a21d7d4889aa6a1a582c0bde278dac9558e3df Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Christoph=20Kn=C3=B6dlseder?= <53149143+chrisknedl@users.noreply.github.com> Date: Sun, 5 Jan 2025 20:55:57 +0100 Subject: [PATCH 21/34] Programming exercises: Remember scrolling position when switching between files in the online code editor (#10053) --- .../monaco/code-editor-monaco.component.ts | 29 ++++++++-- .../monaco-editor/monaco-editor.component.ts | 8 +++ .../code-editor-monaco.component.spec.ts | 58 ++++++++++++------- 3 files changed, 69 insertions(+), 26 deletions(-) diff --git a/src/main/webapp/app/exercises/programming/shared/code-editor/monaco/code-editor-monaco.component.ts b/src/main/webapp/app/exercises/programming/shared/code-editor/monaco/code-editor-monaco.component.ts index b306438919d2..60ea6ca7fa0e 100644 --- a/src/main/webapp/app/exercises/programming/shared/code-editor/monaco/code-editor-monaco.component.ts +++ b/src/main/webapp/app/exercises/programming/shared/code-editor/monaco/code-editor-monaco.component.ts @@ -42,7 +42,7 @@ import { ArtemisProgrammingManualAssessmentModule } from 'app/exercises/programm import { CodeEditorHeaderComponent } from 'app/exercises/programming/shared/code-editor/header/code-editor-header.component'; import { ArtemisSharedModule } from 'app/shared/shared.module'; -type FileSession = { [fileName: string]: { code: string; cursor: EditorPosition; loadingError: boolean } }; +type FileSession = { [fileName: string]: { code: string; cursor: EditorPosition; scrollTop: number; loadingError: boolean } }; type FeedbackWithLineAndReference = Feedback & { line: number; reference: string }; export type Annotation = { fileName: string; row: number; column: number; text: string; type: string; timestamp: number; hash?: string }; @Component({ @@ -157,6 +157,11 @@ export class CodeEditorMonacoComponent implements OnChanges { this.editor().reset(); } if ((changes.selectedFile && this.selectedFile()) || editorWasRefreshed) { + const previousFileName: string | undefined = changes.selectedFile?.previousValue; + // we save the old scrollTop before switching to another file + if (previousFileName && this.fileSession()[previousFileName]) { + this.fileSession()[previousFileName].scrollTop = this.editor().getScrollTop(); + } await this.selectFileInEditor(this.selectedFile()); this.setBuildAnnotations(this.annotationsArray); this.newFeedbackLines.set([]); @@ -196,7 +201,10 @@ export class CodeEditorMonacoComponent implements OnChanges { this.onError.emit('loadingFailed'); } } - this.fileSession.set({ ...this.fileSession(), [fileName]: { code: fileContent, loadingError, cursor: { column: 0, lineNumber: 0 } } }); + this.fileSession.set({ + ...this.fileSession(), + [fileName]: { code: fileContent, loadingError: loadingError, scrollTop: 0, cursor: { column: 0, lineNumber: 0 } }, + }); } const code = this.fileSession()[fileName].code; @@ -204,17 +212,26 @@ export class CodeEditorMonacoComponent implements OnChanges { // Since fetching the file may take some time, we need to check if the file is still selected. if (!this.binaryFileSelected() && this.selectedFile() === fileName) { - this.editor().changeModel(fileName, code); - this.editor().setPosition(this.fileSession()[fileName].cursor); + this.switchToSelectedFile(fileName, code); } this.loadingCount.set(this.loadingCount() - 1); } + switchToSelectedFile(selectedFileName: string, code: string): void { + this.editor().changeModel(selectedFileName, code); + this.editor().setPosition(this.fileSession()[selectedFileName].cursor); + this.editor().setScrollTop(this.fileSession()[this.selectedFile()!].scrollTop ?? 0); + } + onFileTextChanged(text: string): void { if (this.selectedFile() && this.fileSession()[this.selectedFile()!]) { const previousText = this.fileSession()[this.selectedFile()!].code; + const previousScrollTop = this.fileSession()[this.selectedFile()!].scrollTop; if (previousText !== text) { - this.fileSession.set({ ...this.fileSession(), [this.selectedFile()!]: { code: text, loadingError: false, cursor: this.editor().getPosition() } }); + this.fileSession.set({ + ...this.fileSession(), + [this.selectedFile()!]: { code: text, loadingError: false, scrollTop: previousScrollTop, cursor: this.editor().getPosition() }, + }); this.onFileContentChange.emit({ file: this.selectedFile()!, fileContent: text }); } } @@ -405,7 +422,7 @@ export class CodeEditorMonacoComponent implements OnChanges { this.fileSession.set(this.fileService.updateFileReferences(this.fileSession(), fileChange)); this.storeAnnotations([fileChange.fileName]); } else if (fileChange instanceof CreateFileChange && fileChange.fileType === FileType.FILE) { - this.fileSession.set({ ...this.fileSession(), [fileChange.fileName]: { code: '', cursor: { lineNumber: 0, column: 0 }, loadingError: false } }); + this.fileSession.set({ ...this.fileSession(), [fileChange.fileName]: { code: '', cursor: { lineNumber: 0, column: 0 }, scrollTop: 0, loadingError: false } }); } this.setBuildAnnotations(this.annotationsArray); } diff --git a/src/main/webapp/app/shared/monaco-editor/monaco-editor.component.ts b/src/main/webapp/app/shared/monaco-editor/monaco-editor.component.ts index 288f53cc9c07..ad49c129270f 100644 --- a/src/main/webapp/app/shared/monaco-editor/monaco-editor.component.ts +++ b/src/main/webapp/app/shared/monaco-editor/monaco-editor.component.ts @@ -189,6 +189,14 @@ export class MonacoEditorComponent implements OnInit, OnDestroy { this._editor.setPosition(position); } + getScrollTop(): number { + return this._editor.getScrollTop(); + } + + setScrollTop(scrollTop: number) { + this._editor.setScrollTop(scrollTop); + } + setSelection(range: EditorRange): void { this._editor.setSelection(range); } diff --git a/src/test/javascript/spec/component/code-editor/code-editor-monaco.component.spec.ts b/src/test/javascript/spec/component/code-editor/code-editor-monaco.component.spec.ts index 9b658fbf5ece..31a18a01eb44 100644 --- a/src/test/javascript/spec/component/code-editor/code-editor-monaco.component.spec.ts +++ b/src/test/javascript/spec/component/code-editor/code-editor-monaco.component.spec.ts @@ -117,11 +117,11 @@ describe('CodeEditorMonacoComponent', () => { [() => fixture.componentRef.setInput('disableActions', true), true], [() => fixture.componentRef.setInput('commitState', CommitState.CONFLICT), true], [() => fixture.componentRef.setInput('selectedFile', undefined), true], - [() => comp.fileSession.set({ ['file']: { code: '', cursor: { lineNumber: 0, column: 0 }, loadingError: true } }), true], // TODO: convert to signal + [() => comp.fileSession.set({ ['file']: { code: '', cursor: { lineNumber: 0, column: 0 }, loadingError: true, scrollTop: 0 } }), true], // TODO: convert to signal ])('should correctly lock the editor on changes', (setup: () => void, shouldLock: boolean) => { fixture.componentRef.setInput('selectedFile', 'file'); comp.fileSession.set({ - [comp.selectedFile()!]: { code: 'some code', cursor: { lineNumber: 0, column: 0 }, loadingError: false }, + [comp.selectedFile()!]: { code: 'some code', cursor: { lineNumber: 0, column: 0 }, loadingError: false, scrollTop: 0 }, }); fixture.detectChanges(); setup(); @@ -131,7 +131,7 @@ describe('CodeEditorMonacoComponent', () => { it('should update the file session and notify when the file content changes', () => { const selectedFile = 'file'; const fileSession = { - [selectedFile]: { code: 'some unchanged code', cursor: { lineNumber: 1, column: 1 }, loadingError: false }, + [selectedFile]: { code: 'some unchanged code', cursor: { lineNumber: 1, column: 1 }, loadingError: false, scrollTop: 0 }, }; const newCode = 'some new code'; const valueCallbackStub = jest.fn(); @@ -142,7 +142,7 @@ describe('CodeEditorMonacoComponent', () => { comp.onFileTextChanged(newCode); expect(valueCallbackStub).toHaveBeenCalledExactlyOnceWith({ file: selectedFile, fileContent: newCode }); expect(comp.fileSession()).toEqual({ - [selectedFile]: { ...fileSession[selectedFile], code: newCode }, + [selectedFile]: { ...fileSession[selectedFile], code: newCode, scrollTop: 0 }, }); }); @@ -154,7 +154,7 @@ describe('CodeEditorMonacoComponent', () => { const changeModelStub = jest.spyOn(comp.editor(), 'changeModel').mockImplementation(); const presentFileName = 'present-file'; const presentFileSession = { - [presentFileName]: { code: 'code\ncode', cursor: { lineNumber: 1, column: 2 }, loadingError: false }, + [presentFileName]: { code: 'code\ncode', cursor: { lineNumber: 1, column: 2 }, scrollTop: 0, loadingError: false }, }; fixture.detectChanges(); comp.fileSession.set(presentFileSession); @@ -165,7 +165,7 @@ describe('CodeEditorMonacoComponent', () => { expect(loadFileFromRepositoryStub).toHaveBeenCalledExactlyOnceWith(fileToLoad.fileName); expect(comp.fileSession()).toEqual({ ...presentFileSession, - [fileToLoad.fileName]: { code: fileToLoad.fileContent, cursor: { column: 0, lineNumber: 0 }, loadingError: false }, + [fileToLoad.fileName]: { code: fileToLoad.fileContent, cursor: { column: 0, lineNumber: 0 }, scrollTop: 0, loadingError: false }, }); expect(setPositionStub).toHaveBeenCalledTimes(2); expect(changeModelStub).toHaveBeenCalledTimes(2); @@ -174,7 +174,7 @@ describe('CodeEditorMonacoComponent', () => { it('should load a selected file after a loading error', async () => { const fileToLoad = { fileName: 'file-to-load', fileContent: 'some code' }; // File session after loading fails - const fileSession = { [fileToLoad.fileName]: { code: '', loadingError: true, cursor: { lineNumber: 0, column: 0 } } }; + const fileSession = { [fileToLoad.fileName]: { code: '', loadingError: true, cursor: { lineNumber: 0, column: 0 }, scrollTop: 0 } }; const loadedFileSubject = new BehaviorSubject(fileToLoad); loadFileFromRepositoryStub.mockReturnValue(loadedFileSubject); comp.fileSession.set(fileSession); @@ -182,14 +182,16 @@ describe('CodeEditorMonacoComponent', () => { fixture.detectChanges(); await new Promise(process.nextTick); expect(loadFileFromRepositoryStub).toHaveBeenCalledOnce(); - expect(comp.fileSession()).toEqual({ [fileToLoad.fileName]: { code: fileToLoad.fileContent, loadingError: false, cursor: { lineNumber: 0, column: 0 } } }); + expect(comp.fileSession()).toEqual({ + [fileToLoad.fileName]: { code: fileToLoad.fileContent, loadingError: false, cursor: { lineNumber: 0, column: 0 }, scrollTop: 0 }, + }); }); it('should not load binaries into the editor', async () => { const changeModelSpy = jest.spyOn(comp.editor(), 'changeModel'); const fileName = 'file-to-load'; comp.fileSession.set({ - [fileName]: { code: '\0\0\0\0 (binary content)', loadingError: false, cursor: { lineNumber: 0, column: 0 } }, + [fileName]: { code: '\0\0\0\0 (binary content)', loadingError: false, cursor: { lineNumber: 0, column: 0 }, scrollTop: 0 }, }); fixture.detectChanges(); fixture.componentRef.setInput('selectedFile', fileName); @@ -214,7 +216,9 @@ describe('CodeEditorMonacoComponent', () => { await new Promise(process.nextTick); expect(loadFileFromRepositoryStub).toHaveBeenCalledOnce(); expect(errorCallbackStub).toHaveBeenCalledExactlyOnceWith(errorCode); - expect(comp.fileSession()).toEqual({ [fileToLoad.fileName]: { code: '', loadingError: true, cursor: { lineNumber: 0, column: 0 } } }); + expect(comp.fileSession()).toEqual({ + [fileToLoad.fileName]: { code: '', loadingError: true, cursor: { lineNumber: 0, column: 0 }, scrollTop: 0 }, + }); }); it('should discard local changes when the editor is refreshed', async () => { @@ -224,13 +228,13 @@ describe('CodeEditorMonacoComponent', () => { loadFileFromRepositoryStub.mockReturnValue(reloadedFileSubject); fixture.componentRef.setInput('selectedFile', fileToReload.fileName); comp.fileSession.set({ - [fileToReload.fileName]: { code: 'some local undiscarded changes', cursor: { lineNumber: 0, column: 0 }, loadingError: false }, + [fileToReload.fileName]: { code: 'some local undiscarded changes', cursor: { lineNumber: 0, column: 0 }, scrollTop: 0, loadingError: false }, }); fixture.componentRef.setInput('editorState', EditorState.CLEAN); // Simulate a refresh of the editor. await comp.ngOnChanges({ editorState: new SimpleChange(EditorState.REFRESHING, EditorState.CLEAN, false) }); expect(comp.fileSession()).toEqual({ - [fileToReload.fileName]: { code: fileToReload.fileContent, cursor: { lineNumber: 0, column: 0 }, loadingError: false }, + [fileToReload.fileName]: { code: fileToReload.fileContent, cursor: { lineNumber: 0, column: 0 }, loadingError: false, scrollTop: 0 }, }); expect(editorResetStub).toHaveBeenCalledOnce(); }); @@ -238,7 +242,7 @@ describe('CodeEditorMonacoComponent', () => { it('should only load the currently selected file', async () => { const changeModelSpy = jest.spyOn(comp.editor(), 'changeModel'); // Occurs when the first file load takes a while, but the user has already selected another file. - comp.fileSession.set({ ['file2']: { code: 'code2', cursor: { lineNumber: 0, column: 0 }, loadingError: false } }); + comp.fileSession.set({ ['file2']: { code: 'code2', cursor: { lineNumber: 0, column: 0 }, loadingError: false, scrollTop: 0 } }); fixture.detectChanges(); fixture.componentRef.setInput('selectedFile', 'file1'); const longLoadingFileSubject = new Subject(); @@ -258,7 +262,7 @@ describe('CodeEditorMonacoComponent', () => { fixture.detectChanges(); const selectedFile = 'file1'; const fileSession = { - [selectedFile]: { code: 'code\ncode', cursor: { lineNumber: 1, column: 2 }, loadingError: false }, + [selectedFile]: { code: 'code\ncode', cursor: { lineNumber: 1, column: 2 }, loadingError: false, scrollTop: 0 }, }; comp.fileSession.set(fileSession); fixture.componentRef.setInput('selectedFile', selectedFile); @@ -426,8 +430,8 @@ describe('CodeEditorMonacoComponent', () => { const newFileName = 'new-file-name'; const otherFileName = 'other-file'; const fileSession = { - [oldFileName]: { code: 'renamed', cursor: { lineNumber: 0, column: 0 }, loadingError: false }, - [otherFileName]: { code: 'unrelated', cursor: { lineNumber: 0, column: 0 }, loadingError: false }, + [oldFileName]: { code: 'renamed', cursor: { lineNumber: 0, column: 0 }, loadingError: false, scrollTop: 0 }, + [otherFileName]: { code: 'unrelated', cursor: { lineNumber: 0, column: 0 }, loadingError: false, scrollTop: 0 }, }; fixture.detectChanges(); comp.fileSession.set({ ...fileSession }); @@ -443,8 +447,8 @@ describe('CodeEditorMonacoComponent', () => { const fileToDeleteName = 'file-to-delete'; const otherFileName = 'other-file'; const fileSession = { - [fileToDeleteName]: { code: 'will be deleted', cursor: { lineNumber: 0, column: 0 }, loadingError: false }, - [otherFileName]: { code: 'unrelated', cursor: { lineNumber: 0, column: 0 }, loadingError: false }, + [fileToDeleteName]: { code: 'will be deleted', cursor: { lineNumber: 0, column: 0 }, loadingError: false, scrollTop: 0 }, + [otherFileName]: { code: 'unrelated', cursor: { lineNumber: 0, column: 0 }, loadingError: false, scrollTop: 0 }, }; fixture.detectChanges(); comp.fileSession.set({ ...fileSession }); @@ -459,7 +463,7 @@ describe('CodeEditorMonacoComponent', () => { const fileToCreateName = 'file-to-create'; const otherFileName = 'other-file'; const fileSession = { - [otherFileName]: { code: 'unrelated', cursor: { lineNumber: 0, column: 0 }, loadingError: false }, + [otherFileName]: { code: 'unrelated', cursor: { lineNumber: 0, column: 0 }, loadingError: false, scrollTop: 0 }, }; fixture.detectChanges(); comp.fileSession.set({ ...fileSession }); @@ -467,7 +471,7 @@ describe('CodeEditorMonacoComponent', () => { await comp.onFileChange(createFileChange); expect(comp.fileSession()).toEqual({ [otherFileName]: fileSession[otherFileName], - [fileToCreateName]: { code: '', cursor: { lineNumber: 0, column: 0 }, loadingError: false }, + [fileToCreateName]: { code: '', cursor: { lineNumber: 0, column: 0 }, loadingError: false, scrollTop: 0 }, }); }); @@ -477,4 +481,18 @@ describe('CodeEditorMonacoComponent', () => { comp.highlightLines(1, 2); expect(highlightStub).toHaveBeenCalledExactlyOnceWith(1, 2, CodeEditorMonacoComponent.CLASS_DIFF_LINE_HIGHLIGHT); }); + + it('should remember scroll position', async () => { + const setScrollTopStub = jest.spyOn(comp.editor(), 'setScrollTop'); + const scrolledFile = 'file'; + const scrollTop = 42; + const fileSession = { + [scrolledFile]: { code: 'unrelated', cursor: { lineNumber: 0, column: 0 }, loadingError: false, scrollTop: scrollTop }, + }; + fixture.detectChanges(); + comp.fileSession.set(fileSession); + fixture.componentRef.setInput('selectedFile', scrolledFile); + await comp.selectFileInEditor(scrolledFile); + expect(setScrollTopStub).toHaveBeenCalledExactlyOnceWith(scrollTop); + }); }); From 4ed301f2199f02fa35eb316d6020c0d1cc771d97 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maximilian=20S=C3=B6lch?= Date: Sun, 5 Jan 2025 20:57:07 +0100 Subject: [PATCH 22/34] Development: Migrate client code for the assessment dashboard (#10086) --- ...essment-dashboard-information.component.ts | 14 +++-- .../assessment-dashboard.component.ts | 53 ++++++++++++++----- .../assessment-dashboard.module.ts | 4 +- .../exam-assessment-buttons.component.ts | 26 ++++----- ...nt-dashboard-information.component.spec.ts | 2 +- .../assessment-dashboard.component.spec.ts | 1 - .../exam-assessment-buttons.component.spec.ts | 2 +- 7 files changed, 68 insertions(+), 34 deletions(-) diff --git a/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard-information.component.ts b/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard-information.component.ts index b2d0ef1ee8d3..36819ac62b0f 100644 --- a/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard-information.component.ts +++ b/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard-information.component.ts @@ -1,10 +1,14 @@ -import { Component, Input, OnChanges, OnDestroy, OnInit } from '@angular/core'; +import { Component, Input, OnChanges, OnDestroy, OnInit, inject } from '@angular/core'; import { DueDateStat } from 'app/course/dashboards/due-date-stat.model'; -import { LegendPosition } from '@swimlane/ngx-charts'; +import { LegendPosition, PieChartModule } from '@swimlane/ngx-charts'; import { TranslateService } from '@ngx-translate/core'; import { GraphColors } from 'app/entities/statistics.model'; import { Subscription } from 'rxjs'; import { Course } from 'app/entities/course.model'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { ArtemisSidePanelModule } from 'app/shared/side-panel/side-panel.module'; +import { RouterLink } from '@angular/router'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; export class AssessmentDashboardInformationEntry { constructor( @@ -33,8 +37,12 @@ export class AssessmentDashboardInformationEntry { selector: 'jhi-assessment-dashboard-information', templateUrl: './assessment-dashboard-information.component.html', styleUrls: ['./assessment-dashboard-information.component.scss'], + standalone: true, + imports: [TranslateDirective, PieChartModule, ArtemisSidePanelModule, RouterLink, ArtemisTranslatePipe], }) export class AssessmentDashboardInformationComponent implements OnInit, OnChanges, OnDestroy { + private translateService = inject(TranslateService); + @Input() isExamMode: boolean; @Input() course: Course; @Input() examId?: number; @@ -71,8 +79,6 @@ export class AssessmentDashboardInformationComponent implements OnInit, OnChange themeSubscription: Subscription; - constructor(private translateService: TranslateService) {} - ngOnInit(): void { this.setup(); this.translateService.onLangChange.subscribe(() => { diff --git a/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard.component.ts b/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard.component.ts index a39b6af80e77..8ada9ad9321b 100644 --- a/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard.component.ts +++ b/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard.component.ts @@ -1,5 +1,5 @@ -import { Component, OnInit } from '@angular/core'; -import { ActivatedRoute } from '@angular/router'; +import { Component, OnInit, inject } from '@angular/core'; +import { ActivatedRoute, RouterLink } from '@angular/router'; import { CourseManagementService } from '../../manage/course-management.service'; import { AlertService } from 'app/core/util/alert.service'; import { User } from 'app/core/user/user.model'; @@ -17,19 +17,55 @@ import { Exam } from 'app/entities/exam/exam.model'; import { ExamManagementService } from 'app/exam/manage/exam-management.service'; import { ExerciseService } from 'app/exercises/shared/exercise/exercise.service'; import { QuizExercise } from 'app/entities/quiz/quiz-exercise.model'; -import { AssessmentDashboardInformationEntry } from './assessment-dashboard-information.component'; +import { AssessmentDashboardInformationComponent, AssessmentDashboardInformationEntry } from './assessment-dashboard-information.component'; import { TutorIssue, TutorIssueComplaintsChecker, TutorIssueRatingChecker, TutorIssueScoreChecker } from 'app/course/dashboards/assessment-dashboard/tutor-issue'; import { TutorLeaderboardElement } from 'app/shared/dashboards/tutor-leaderboard/tutor-leaderboard.model'; import { faClipboard, faHeartBroken, faSort, faTable } from '@fortawesome/free-solid-svg-icons'; import { DocumentationType } from 'app/shared/components/documentation-button/documentation-button.component'; +import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; +import { FaIconComponent } from '@fortawesome/angular-fontawesome'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { ExamAssessmentButtonsComponent } from 'app/course/dashboards/assessment-dashboard/exam-assessment-buttons/exam-assessment-buttons.component'; +import { FormsModule } from '@angular/forms'; +import { ArtemisSharedCommonModule } from 'app/shared/shared-common.module'; +import { NgbTooltip } from '@ng-bootstrap/ng-bootstrap'; +import { ArtemisTutorParticipationGraphModule } from 'app/shared/dashboards/tutor-participation-graph/tutor-participation-graph.module'; +import { ArtemisExerciseAssessmentDashboardModule } from 'app/exercises/shared/dashboards/tutor/exercise-assessment-dashboard.module'; +import { ArtemisTutorLeaderboardModule } from 'app/shared/dashboards/tutor-leaderboard/tutor-leaderboard.module'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; @Component({ selector: 'jhi-assessment-dashboard', templateUrl: './assessment-dashboard.component.html', styleUrls: ['./exam-assessment-buttons/exam-assessment-buttons.component.scss'], providers: [CourseManagementService], + standalone: true, + imports: [ + ArtemisSharedComponentModule, + RouterLink, + FaIconComponent, + TranslateDirective, + ExamAssessmentButtonsComponent, + AssessmentDashboardInformationComponent, + FormsModule, + ArtemisSharedCommonModule, + NgbTooltip, + ArtemisTutorParticipationGraphModule, + ArtemisExerciseAssessmentDashboardModule, + ArtemisTutorLeaderboardModule, + ArtemisTranslatePipe, + ], }) export class AssessmentDashboardComponent implements OnInit { + private courseService = inject(CourseManagementService); + private exerciseService = inject(ExerciseService); + private examManagementService = inject(ExamManagementService); + private alertService = inject(AlertService); + private accountService = inject(AccountService); + private route = inject(ActivatedRoute); + private guidedTourService = inject(GuidedTourService); + private sortService = inject(SortService); + readonly TeamFilterProp = TeamFilterProp; readonly documentationType: DocumentationType = 'Assessment'; @@ -79,17 +115,6 @@ export class AssessmentDashboardComponent implements OnInit { faClipboard = faClipboard; faHeartBroken = faHeartBroken; - constructor( - private courseService: CourseManagementService, - private exerciseService: ExerciseService, - private examManagementService: ExamManagementService, - private alertService: AlertService, - private accountService: AccountService, - private route: ActivatedRoute, - private guidedTourService: GuidedTourService, - private sortService: SortService, - ) {} - /** * On init set the courseID, load all exercises and statistics for tutors and set the identity for the AccountService. */ diff --git a/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard.module.ts b/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard.module.ts index 1095436e3162..d4edadae0921 100644 --- a/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard.module.ts +++ b/src/main/webapp/app/course/dashboards/assessment-dashboard/assessment-dashboard.module.ts @@ -26,7 +26,9 @@ const ENTITY_STATES = [...assessmentDashboardRoute]; ArtemisExerciseAssessmentDashboardModule, ArtemisSharedComponentModule, PieChartModule, + AssessmentDashboardComponent, + AssessmentDashboardInformationComponent, + ExamAssessmentButtonsComponent, ], - declarations: [AssessmentDashboardComponent, AssessmentDashboardInformationComponent, ExamAssessmentButtonsComponent], }) export class ArtemisAssessmentDashboardModule {} diff --git a/src/main/webapp/app/course/dashboards/assessment-dashboard/exam-assessment-buttons/exam-assessment-buttons.component.ts b/src/main/webapp/app/course/dashboards/assessment-dashboard/exam-assessment-buttons/exam-assessment-buttons.component.ts index 7bc20e00e55c..de7010811fae 100644 --- a/src/main/webapp/app/course/dashboards/assessment-dashboard/exam-assessment-buttons/exam-assessment-buttons.component.ts +++ b/src/main/webapp/app/course/dashboards/assessment-dashboard/exam-assessment-buttons/exam-assessment-buttons.component.ts @@ -1,5 +1,5 @@ -import { Component, OnInit } from '@angular/core'; -import { ActivatedRoute } from '@angular/router'; +import { Component, OnInit, inject } from '@angular/core'; +import { ActivatedRoute, RouterLink } from '@angular/router'; import { StudentExamService } from 'app/exam/manage/student-exams/student-exam.service'; import { Subscription, forkJoin } from 'rxjs'; import { tap } from 'rxjs/operators'; @@ -15,12 +15,24 @@ import { AccountService } from 'app/core/auth/account.service'; import { onError } from 'app/shared/util/global.utils'; import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; import { faClipboard } from '@fortawesome/free-solid-svg-icons'; +import { FaIconComponent } from '@fortawesome/angular-fontawesome'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; @Component({ selector: 'jhi-exam-assessment-buttons', templateUrl: './exam-assessment-buttons.component.html', + standalone: true, + imports: [RouterLink, FaIconComponent, TranslateDirective], }) export class ExamAssessmentButtonsComponent implements OnInit { + private route = inject(ActivatedRoute); + private examManagementService = inject(ExamManagementService); + private studentExamService = inject(StudentExamService); + private courseService = inject(CourseManagementService); + private alertService = inject(AlertService); + private accountService = inject(AccountService); + private artemisTranslatePipe = inject(ArtemisTranslatePipe); + courseId: number; examId: number; studentExams: StudentExam[]; @@ -38,16 +50,6 @@ export class ExamAssessmentButtonsComponent implements OnInit { // icons faClipboard = faClipboard; - constructor( - private route: ActivatedRoute, - private examManagementService: ExamManagementService, - private studentExamService: StudentExamService, - private courseService: CourseManagementService, - private alertService: AlertService, - private accountService: AccountService, - private artemisTranslatePipe: ArtemisTranslatePipe, - ) {} - /** * Initialize the courseId and examId */ diff --git a/src/test/javascript/spec/component/assessment-dashboard/assessment-dashboard-information.component.spec.ts b/src/test/javascript/spec/component/assessment-dashboard/assessment-dashboard-information.component.spec.ts index 92d0ec9f0f89..ee0f7b8f686f 100644 --- a/src/test/javascript/spec/component/assessment-dashboard/assessment-dashboard-information.component.spec.ts +++ b/src/test/javascript/spec/component/assessment-dashboard/assessment-dashboard-information.component.spec.ts @@ -20,7 +20,7 @@ describe('AssessmentDashboardInformationComponent', () => { beforeEach(() => { TestBed.configureTestingModule({ imports: [ArtemisTestModule, MockModule(PieChartModule)], - declarations: [AssessmentDashboardInformationComponent, MockPipe(ArtemisTranslatePipe), MockComponent(SidePanelComponent)], + declarations: [MockPipe(ArtemisTranslatePipe), MockComponent(SidePanelComponent)], providers: [{ provide: TranslateService, useClass: MockTranslateService }], }).compileComponents(); diff --git a/src/test/javascript/spec/component/assessment-dashboard/assessment-dashboard.component.spec.ts b/src/test/javascript/spec/component/assessment-dashboard/assessment-dashboard.component.spec.ts index bc3cfc327be3..55bb55097cc4 100644 --- a/src/test/javascript/spec/component/assessment-dashboard/assessment-dashboard.component.spec.ts +++ b/src/test/javascript/spec/component/assessment-dashboard/assessment-dashboard.component.spec.ts @@ -142,7 +142,6 @@ describe('AssessmentDashboardInformationComponent', () => { return TestBed.configureTestingModule({ imports: [ArtemisTestModule, MockModule(RouterModule)], declarations: [ - AssessmentDashboardComponent, MockComponent(TutorLeaderboardComponent), MockComponent(TutorParticipationGraphComponent), MockComponent(AssessmentDashboardInformationComponent), diff --git a/src/test/javascript/spec/component/assessment-dashboard/exam-assessment-buttons.component.spec.ts b/src/test/javascript/spec/component/assessment-dashboard/exam-assessment-buttons.component.spec.ts index e3e18abc527a..eb9e8da45f07 100644 --- a/src/test/javascript/spec/component/assessment-dashboard/exam-assessment-buttons.component.spec.ts +++ b/src/test/javascript/spec/component/assessment-dashboard/exam-assessment-buttons.component.spec.ts @@ -116,7 +116,7 @@ describe('ExamAssessmentButtons', () => { TestBed.configureTestingModule({ imports: [ArtemisTestModule], - declarations: [ExamAssessmentButtonsComponent, MockDirective(MockHasAnyAuthorityDirective), MockPipe(ArtemisTranslatePipe), MockRouterLinkDirective], + declarations: [MockDirective(MockHasAnyAuthorityDirective), MockPipe(ArtemisTranslatePipe), MockRouterLinkDirective], providers, }) .compileComponents() From a5579eb89ce54397da6e3e9a326bda02e2d00504 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maximilian=20S=C3=B6lch?= Date: Sun, 5 Jan 2025 21:06:27 +0100 Subject: [PATCH 23/34] Development: Migrate client code for complaints (#10085) --- .../assessment/assessment-shared.module.ts | 4 +-- .../complaints/complaint-response.service.ts | 10 +++--- .../app/complaints/complaint.service.ts | 10 +++--- .../complaints-student-view.component.ts | 22 +++++++----- .../complaints-for-tutor.component.ts | 24 +++++++------ .../complaints-for-tutor.module.ts | 14 -------- .../app/complaints/complaints.module.ts | 3 +- .../form/complaints-form.component.ts | 16 +++++---- .../list-of-complaints.component.ts | 34 +++++++++++-------- .../list-of-complaints.module.ts | 4 +-- .../request/complaint-request.component.ts | 7 ++++ .../response/complaint-response.component.ts | 6 ++++ .../course/manage/course-management.module.ts | 4 +-- .../assess/programming-assessment.module.ts | 4 +-- .../programming-repository.module.ts | 6 ++-- .../text-submission-assessment.module.ts | 4 +-- .../assessment-layout.component.spec.ts | 5 +-- .../complaints-for-tutor.component.spec.ts | 19 +++++++---- .../complaints-form.component.spec.ts | 8 +---- .../list-of-complaints.component.spec.ts | 9 ++--- 20 files changed, 110 insertions(+), 103 deletions(-) delete mode 100644 src/main/webapp/app/complaints/complaints-for-tutor/complaints-for-tutor.module.ts diff --git a/src/main/webapp/app/assessment/assessment-shared.module.ts b/src/main/webapp/app/assessment/assessment-shared.module.ts index a2df1e03cc23..fe386487df8c 100644 --- a/src/main/webapp/app/assessment/assessment-shared.module.ts +++ b/src/main/webapp/app/assessment/assessment-shared.module.ts @@ -6,7 +6,6 @@ import { AssessmentComplaintAlertComponent } from './assessment-complaint-alert/ import { ScoreDisplayComponent } from '../shared/score-display/score-display.component'; import { UnreferencedFeedbackDetailComponent } from 'app/assessment/unreferenced-feedback-detail/unreferenced-feedback-detail.component'; import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; -import { ArtemisComplaintsForTutorModule } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.module'; import { AssessmentLocksComponent } from 'app/assessment/assessment-locks/assessment-locks.component'; import { RouterModule } from '@angular/router'; import { assessmentLocksRoute } from 'app/assessment/assessment-locks/assessment-locks.route'; @@ -18,13 +17,14 @@ import { ArtemisFeedbackModule } from 'app/exercises/shared/feedback/feedback.mo import { AssessmentNoteComponent } from 'app/assessment/assessment-note/assessment-note.component'; import { FeedbackContentPipe } from 'app/shared/pipes/feedback-content.pipe'; import { QuotePipe } from 'app/shared/pipes/quote.pipe'; +import { ComplaintsForTutorComponent } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.component'; const ENTITY_STATES = [...assessmentLocksRoute]; @NgModule({ imports: [ ArtemisSharedModule, - ArtemisComplaintsForTutorModule, + ComplaintsForTutorComponent, ArtemisSharedComponentModule, RouterModule.forChild(ENTITY_STATES), ArtemisMarkdownModule, diff --git a/src/main/webapp/app/complaints/complaint-response.service.ts b/src/main/webapp/app/complaints/complaint-response.service.ts index 54e801ba687f..4ab4cb080f2e 100644 --- a/src/main/webapp/app/complaints/complaint-response.service.ts +++ b/src/main/webapp/app/complaints/complaint-response.service.ts @@ -1,4 +1,4 @@ -import { Injectable } from '@angular/core'; +import { Injectable, inject } from '@angular/core'; import { HttpClient, HttpResponse } from '@angular/common/http'; import { Observable } from 'rxjs'; import { map } from 'rxjs/operators'; @@ -12,12 +12,10 @@ type EntityResponseType = HttpResponse; @Injectable({ providedIn: 'root' }) export class ComplaintResponseService { - private resourceUrl = 'api/complaints'; + private http = inject(HttpClient); + private accountService = inject(AccountService); - constructor( - private http: HttpClient, - private accountService: AccountService, - ) {} + private resourceUrl = 'api/complaints'; /** * Checks if a complaint response is locked for the currently logged-in user diff --git a/src/main/webapp/app/complaints/complaint.service.ts b/src/main/webapp/app/complaints/complaint.service.ts index 496cdaeacc6d..5ea2662dca9f 100644 --- a/src/main/webapp/app/complaints/complaint.service.ts +++ b/src/main/webapp/app/complaints/complaint.service.ts @@ -1,4 +1,4 @@ -import { Injectable } from '@angular/core'; +import { Injectable, inject } from '@angular/core'; import { HttpClient, HttpResponse } from '@angular/common/http'; import { Observable } from 'rxjs'; import dayjs from 'dayjs/esm'; @@ -31,12 +31,10 @@ export interface IComplaintService { @Injectable({ providedIn: 'root' }) export class ComplaintService implements IComplaintService { - private resourceUrl = 'api/complaints'; + private http = inject(HttpClient); + private complaintResponseService = inject(ComplaintResponseService); - constructor( - private http: HttpClient, - private complaintResponseService: ComplaintResponseService, - ) {} + private resourceUrl = 'api/complaints'; /** * Checks if a complaint is locked for the currently logged-in user diff --git a/src/main/webapp/app/complaints/complaints-for-students/complaints-student-view.component.ts b/src/main/webapp/app/complaints/complaints-for-students/complaints-student-view.component.ts index b469dcab432f..ed0bef126614 100644 --- a/src/main/webapp/app/complaints/complaints-for-students/complaints-student-view.component.ts +++ b/src/main/webapp/app/complaints/complaints-for-students/complaints-student-view.component.ts @@ -1,4 +1,4 @@ -import { Component, Input, OnInit } from '@angular/core'; +import { Component, Input, OnInit, inject } from '@angular/core'; import { Exercise, getCourseFromExercise } from 'app/entities/exercise.model'; import { Complaint, ComplaintType } from 'app/entities/complaint.model'; import { ComplaintService } from 'app/complaints/complaint.service'; @@ -14,13 +14,26 @@ import dayjs from 'dayjs/esm'; import { HttpResponse } from '@angular/common/http'; import { faInfoCircle } from '@fortawesome/free-solid-svg-icons'; import { CourseManagementService } from 'app/course/manage/course-management.service'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { FaIconComponent } from '@fortawesome/angular-fontawesome'; +import { ComplaintsFormComponent } from 'app/complaints/form/complaints-form.component'; +import { ComplaintRequestComponent } from 'app/complaints/request/complaint-request.component'; +import { ComplaintResponseComponent } from 'app/complaints/response/complaint-response.component'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; @Component({ selector: 'jhi-complaint-student-view', templateUrl: './complaints-student-view.component.html', styleUrls: ['../complaints.scss'], + standalone: true, + imports: [TranslateDirective, FaIconComponent, ComplaintsFormComponent, ComplaintRequestComponent, ComplaintResponseComponent, ArtemisTranslatePipe], }) export class ComplaintsStudentViewComponent implements OnInit { + private complaintService = inject(ComplaintService); + private serverDateService = inject(ArtemisServerDateService); + private accountService = inject(AccountService); + private courseService = inject(CourseManagementService); + @Input() exercise: Exercise; @Input() participation: StudentParticipation; @Input() result?: Result; @@ -46,13 +59,6 @@ export class ComplaintsStudentViewComponent implements OnInit { // Icons faInfoCircle = faInfoCircle; - constructor( - private complaintService: ComplaintService, - private serverDateService: ArtemisServerDateService, - private accountService: AccountService, - private courseService: CourseManagementService, - ) {} - /** * Loads the number of allowed complaints and feedback requests */ diff --git a/src/main/webapp/app/complaints/complaints-for-tutor/complaints-for-tutor.component.ts b/src/main/webapp/app/complaints/complaints-for-tutor/complaints-for-tutor.component.ts index 71c185e7d488..a98e34d60b17 100644 --- a/src/main/webapp/app/complaints/complaints-for-tutor/complaints-for-tutor.component.ts +++ b/src/main/webapp/app/complaints/complaints-for-tutor/complaints-for-tutor.component.ts @@ -1,4 +1,4 @@ -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; +import { Component, EventEmitter, Input, OnInit, Output, inject } from '@angular/core'; import { AlertService } from 'app/core/util/alert.service'; import { HttpErrorResponse } from '@angular/common/http'; import { ComplaintResponseService } from 'app/complaints/complaint-response.service'; @@ -13,15 +13,27 @@ import { Submission } from 'app/entities/submission.model'; import { isAllowedToRespondToComplaintAction } from 'app/assessment/assessment.service'; import { Course } from 'app/entities/course.model'; import { ComplaintAction, ComplaintResponseUpdateDTO } from 'app/entities/complaint-response-dto.model'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { FormsModule } from '@angular/forms'; +import { TextareaModule } from 'app/shared/textarea/textarea.module'; +import { ArtemisSharedCommonModule } from 'app/shared/shared-common.module'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; export type AssessmentAfterComplaint = { complaintResponse: ComplaintResponse; onSuccess: () => void; onError: () => void }; @Component({ selector: 'jhi-complaints-for-tutor-form', templateUrl: './complaints-for-tutor.component.html', - providers: [], + standalone: true, + imports: [TranslateDirective, FormsModule, TextareaModule, ArtemisSharedCommonModule, ArtemisTranslatePipe], }) export class ComplaintsForTutorComponent implements OnInit { + private alertService = inject(AlertService); + private complaintResponseService = inject(ComplaintResponseService); + private activatedRoute = inject(ActivatedRoute); + private router = inject(Router); + private location = inject(Location); + @Input() complaint: Complaint; @Input() isTestRun = false; @Input() isAssessor = false; @@ -43,14 +55,6 @@ export class ComplaintsForTutorComponent implements OnInit { course?: Course; maxComplaintResponseTextLimit: number; - constructor( - private alertService: AlertService, - private complaintResponseService: ComplaintResponseService, - private activatedRoute: ActivatedRoute, - private router: Router, - private location: Location, - ) {} - ngOnInit(): void { this.course = getCourseFromExercise(this.exercise!); diff --git a/src/main/webapp/app/complaints/complaints-for-tutor/complaints-for-tutor.module.ts b/src/main/webapp/app/complaints/complaints-for-tutor/complaints-for-tutor.module.ts deleted file mode 100644 index 6b3e0eef93f4..000000000000 --- a/src/main/webapp/app/complaints/complaints-for-tutor/complaints-for-tutor.module.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { NgModule } from '@angular/core'; -import { ArtemisSharedModule } from 'app/shared/shared.module'; -import { ComplaintsForTutorComponent } from './complaints-for-tutor.component'; -import { ComplaintResponseService } from 'app/complaints/complaint-response.service'; -import { ComplaintService } from 'app/complaints/complaint.service'; -import { TextareaModule } from 'app/shared/textarea/textarea.module'; - -@NgModule({ - imports: [ArtemisSharedModule, TextareaModule], - declarations: [ComplaintsForTutorComponent], - exports: [ComplaintsForTutorComponent], - providers: [ComplaintService, ComplaintResponseService], -}) -export class ArtemisComplaintsForTutorModule {} diff --git a/src/main/webapp/app/complaints/complaints.module.ts b/src/main/webapp/app/complaints/complaints.module.ts index e851a892a538..709ea3322cc7 100644 --- a/src/main/webapp/app/complaints/complaints.module.ts +++ b/src/main/webapp/app/complaints/complaints.module.ts @@ -9,8 +9,7 @@ import { ComplaintResponseComponent } from 'app/complaints/response/complaint-re import { TextareaModule } from 'app/shared/textarea/textarea.module'; @NgModule({ - imports: [ArtemisSharedModule, TextareaModule], - declarations: [ComplaintsFormComponent, ComplaintsStudentViewComponent, ComplaintRequestComponent, ComplaintResponseComponent], + imports: [ArtemisSharedModule, TextareaModule, ComplaintsFormComponent, ComplaintsStudentViewComponent, ComplaintRequestComponent, ComplaintResponseComponent], exports: [ComplaintsStudentViewComponent], providers: [ComplaintService], }) diff --git a/src/main/webapp/app/complaints/form/complaints-form.component.ts b/src/main/webapp/app/complaints/form/complaints-form.component.ts index 015b05f95a95..5d5265a527e1 100644 --- a/src/main/webapp/app/complaints/form/complaints-form.component.ts +++ b/src/main/webapp/app/complaints/form/complaints-form.component.ts @@ -1,5 +1,5 @@ import { HttpErrorResponse } from '@angular/common/http'; -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; +import { Component, EventEmitter, Input, OnInit, Output, inject } from '@angular/core'; import { ComplaintService } from 'app/complaints/complaint.service'; import { AlertService } from 'app/core/util/alert.service'; import { ComplaintType } from 'app/entities/complaint.model'; @@ -7,13 +7,22 @@ import { Course } from 'app/entities/course.model'; import { Exercise, getCourseFromExercise } from 'app/entities/exercise.model'; import { onError } from 'app/shared/util/global.utils'; import { ComplaintRequestDTO } from 'app/entities/complaint-request-dto.model'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { FormsModule } from '@angular/forms'; +import { TextareaModule } from 'app/shared/textarea/textarea.module'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; @Component({ selector: 'jhi-complaint-form', templateUrl: './complaints-form.component.html', styleUrls: ['../complaints.scss'], + standalone: true, + imports: [TranslateDirective, FormsModule, TextareaModule, ArtemisTranslatePipe], }) export class ComplaintsFormComponent implements OnInit { + private complaintService = inject(ComplaintService); + private alertService = inject(AlertService); + @Input() exercise: Exercise; @Input() resultId: number; @Input() examId?: number; @@ -27,11 +36,6 @@ export class ComplaintsFormComponent implements OnInit { readonly ComplaintType = ComplaintType; - constructor( - private complaintService: ComplaintService, - private alertService: AlertService, - ) {} - ngOnInit(): void { this.course = getCourseFromExercise(this.exercise); this.maxComplaintTextLimit = this.course?.maxComplaintTextLimit ?? 0; diff --git a/src/main/webapp/app/complaints/list-of-complaints/list-of-complaints.component.ts b/src/main/webapp/app/complaints/list-of-complaints/list-of-complaints.component.ts index 131d552ffc4c..a2a3c2197a0b 100644 --- a/src/main/webapp/app/complaints/list-of-complaints/list-of-complaints.component.ts +++ b/src/main/webapp/app/complaints/list-of-complaints/list-of-complaints.component.ts @@ -1,4 +1,4 @@ -import { Component, OnInit } from '@angular/core'; +import { Component, OnInit, inject } from '@angular/core'; import { AlertService } from 'app/core/util/alert.service'; import { ComplaintService } from 'app/complaints/complaint.service'; import { CourseManagementService } from 'app/course/manage/course-management.service'; @@ -8,20 +8,36 @@ import { ActivatedRoute, Router } from '@angular/router'; import { Course } from 'app/entities/course.model'; import { Observable, combineLatestWith } from 'rxjs'; import { StudentParticipation } from 'app/entities/participation/student-participation.model'; -import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; +import { NgbModal, NgbTooltip } from '@ng-bootstrap/ng-bootstrap'; import { SortService } from 'app/shared/service/sort.service'; import { ArtemisDatePipe } from 'app/shared/pipes/artemis-date.pipe'; import { TranslateService } from '@ngx-translate/core'; import { onError } from 'app/shared/util/global.utils'; import { getLinkToSubmissionAssessment } from 'app/utils/navigation.utils'; import { faExclamationTriangle, faFolderOpen, faSort } from '@fortawesome/free-solid-svg-icons'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { FormsModule } from '@angular/forms'; +import { ArtemisSharedCommonModule } from 'app/shared/shared-common.module'; +import { FaIconComponent } from '@fortawesome/angular-fontawesome'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; @Component({ selector: 'jhi-complaint-list', templateUrl: './list-of-complaints.component.html', - providers: [], + standalone: true, + imports: [TranslateDirective, FormsModule, ArtemisSharedCommonModule, FaIconComponent, NgbTooltip, ArtemisTranslatePipe], }) export class ListOfComplaintsComponent implements OnInit { + complaintService = inject(ComplaintService); + private alertService = inject(AlertService); + private route = inject(ActivatedRoute); + private router = inject(Router); + private modalService = inject(NgbModal); + private sortService = inject(SortService); + private translateService = inject(TranslateService); + private artemisDatePipe = inject(ArtemisDatePipe); + private courseManagementService = inject(CourseManagementService); + readonly ComplaintType = ComplaintType; public complaints: Complaint[] = []; @@ -49,18 +65,6 @@ export class ListOfComplaintsComponent implements OnInit { readonly FILTER_OPTION_ADDRESSED_COMPLAINTS = 4; // the number passed by the chart through the route indicating that only addressed complaints should be shown - constructor( - public complaintService: ComplaintService, - private alertService: AlertService, - private route: ActivatedRoute, - private router: Router, - private modalService: NgbModal, - private sortService: SortService, - private translateService: TranslateService, - private artemisDatePipe: ArtemisDatePipe, - private courseManagementService: CourseManagementService, - ) {} - ngOnInit(): void { this.route.params.pipe(combineLatestWith(this.route.queryParams, this.route.data)).subscribe((result) => { const params = result[0]; diff --git a/src/main/webapp/app/complaints/list-of-complaints/list-of-complaints.module.ts b/src/main/webapp/app/complaints/list-of-complaints/list-of-complaints.module.ts index df108f196b51..94122fae0136 100644 --- a/src/main/webapp/app/complaints/list-of-complaints/list-of-complaints.module.ts +++ b/src/main/webapp/app/complaints/list-of-complaints/list-of-complaints.module.ts @@ -3,13 +3,11 @@ import { ListOfComplaintsComponent } from './list-of-complaints.component'; import { ComplaintService } from 'app/complaints/complaint.service'; import { RouterModule } from '@angular/router'; import { listOfComplaintsRoute } from 'app/complaints/list-of-complaints/list-of-complaints.route'; -import { ArtemisSharedModule } from 'app/shared/shared.module'; const ENTITY_STATES = [...listOfComplaintsRoute]; @NgModule({ - imports: [ArtemisSharedModule, RouterModule.forChild(ENTITY_STATES)], - declarations: [ListOfComplaintsComponent], + imports: [RouterModule.forChild(ENTITY_STATES), ListOfComplaintsComponent], exports: [ListOfComplaintsComponent], providers: [ComplaintService], }) diff --git a/src/main/webapp/app/complaints/request/complaint-request.component.ts b/src/main/webapp/app/complaints/request/complaint-request.component.ts index ab9817fd3794..61065c172d4e 100644 --- a/src/main/webapp/app/complaints/request/complaint-request.component.ts +++ b/src/main/webapp/app/complaints/request/complaint-request.component.ts @@ -1,9 +1,16 @@ import { Component, Input } from '@angular/core'; import { Complaint, ComplaintType } from 'app/entities/complaint.model'; +import { NgbTooltip } from '@ng-bootstrap/ng-bootstrap'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { FormsModule } from '@angular/forms'; +import { ArtemisSharedCommonModule } from 'app/shared/shared-common.module'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; @Component({ selector: 'jhi-complaint-request', templateUrl: './complaint-request.component.html', + standalone: true, + imports: [NgbTooltip, TranslateDirective, FormsModule, ArtemisSharedCommonModule, ArtemisTranslatePipe], }) export class ComplaintRequestComponent { @Input() complaint: Complaint; diff --git a/src/main/webapp/app/complaints/response/complaint-response.component.ts b/src/main/webapp/app/complaints/response/complaint-response.component.ts index aabdeb3e3073..c2443cf7ea60 100644 --- a/src/main/webapp/app/complaints/response/complaint-response.component.ts +++ b/src/main/webapp/app/complaints/response/complaint-response.component.ts @@ -1,9 +1,15 @@ import { Component, Input } from '@angular/core'; import { Complaint, ComplaintType } from 'app/entities/complaint.model'; +import { NgbTooltip } from '@ng-bootstrap/ng-bootstrap'; +import { FormsModule } from '@angular/forms'; +import { ArtemisSharedCommonModule } from 'app/shared/shared-common.module'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; @Component({ selector: 'jhi-complaint-response', templateUrl: './complaint-response.component.html', + standalone: true, + imports: [NgbTooltip, FormsModule, ArtemisSharedCommonModule, ArtemisTranslatePipe], }) export class ComplaintResponseComponent { @Input() complaint: Complaint; diff --git a/src/main/webapp/app/course/manage/course-management.module.ts b/src/main/webapp/app/course/manage/course-management.module.ts index 68b0ce410db3..3aeca48c59d3 100644 --- a/src/main/webapp/app/course/manage/course-management.module.ts +++ b/src/main/webapp/app/course/manage/course-management.module.ts @@ -23,7 +23,6 @@ import { ArtemisParticipationModule } from 'app/exercises/shared/participation/p import { ArtemisModelingExerciseManagementModule } from 'app/exercises/modeling/manage/modeling-exercise-management.module'; import { ArtemisCourseScoresModule } from 'app/course/course-scores/course-scores.module'; import { ArtemisExerciseScoresModule } from 'app/exercises/shared/exercise-scores/exercise-scores.module'; -import { ArtemisComplaintsForTutorModule } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.module'; import { ArtemisFileUploadAssessmentModule } from 'app/exercises/file-upload/assess/file-upload-assessment.module'; import { ArtemisModelingAssessmentEditorModule } from 'app/exercises/modeling/assess/modeling-assessment-editor/modeling-assessment-editor.module'; import { NgxDatatableModule } from '@siemens/ngx-datatable'; @@ -68,6 +67,7 @@ import { SubmissionResultStatusModule } from 'app/overview/submission-result-sta import { ImageCropperModalComponent } from 'app/course/manage/image-cropper-modal.component'; import { HeaderCourseComponent } from 'app/overview/header-course.component'; import { ArtemisMarkdownEditorModule } from 'app/shared/markdown-editor/markdown-editor.module'; +import { ComplaintsForTutorComponent } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.component'; @NgModule({ imports: [ @@ -95,7 +95,7 @@ import { ArtemisMarkdownEditorModule } from 'app/shared/markdown-editor/markdown ArtemisColorSelectorModule, ArtemisDashboardsModule, ArtemisParticipationModule, - ArtemisComplaintsForTutorModule, + ComplaintsForTutorComponent, ArtemisListOfComplaintsModule, ArtemisFileUploadAssessmentModule, ArtemisModelingAssessmentEditorModule, diff --git a/src/main/webapp/app/exercises/programming/assess/programming-assessment.module.ts b/src/main/webapp/app/exercises/programming/assess/programming-assessment.module.ts index 944d2a220e60..7ae991b6ac22 100644 --- a/src/main/webapp/app/exercises/programming/assess/programming-assessment.module.ts +++ b/src/main/webapp/app/exercises/programming/assess/programming-assessment.module.ts @@ -5,7 +5,6 @@ import { FormDateTimePickerModule } from 'app/shared/date-time-picker/date-time- import { FormsModule } from '@angular/forms'; import { FeatureToggleModule } from 'app/shared/feature-toggle/feature-toggle.module'; import { ProgrammingAssessmentRepoExportButtonComponent } from 'app/exercises/programming/assess/repo-export/programming-assessment-repo-export-button.component'; -import { ArtemisComplaintsForTutorModule } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.module'; import { ProgrammingAssessmentRepoExportDialogComponent } from 'app/exercises/programming/assess/repo-export/programming-assessment-repo-export-dialog.component'; import { ArtemisProgrammingAssessmentRoutingModule } from 'app/exercises/programming/assess/programming-assessment.route'; import { ArtemisAssessmentSharedModule } from 'app/assessment/assessment-shared.module'; @@ -18,6 +17,7 @@ import { AssessmentInstructionsModule } from 'app/assessment/assessment-instruct import { ArtemisHeaderExercisePageWithDetailsModule } from 'app/exercises/shared/exercise-headers/exercise-headers.module'; import { OrionTutorAssessmentComponent } from 'app/orion/assessment/orion-tutor-assessment.component'; import { SubmissionResultStatusModule } from 'app/overview/submission-result-status.module'; +import { ComplaintsForTutorComponent } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.component'; @NgModule({ imports: [ @@ -26,7 +26,7 @@ import { SubmissionResultStatusModule } from 'app/overview/submission-result-sta FormDateTimePickerModule, FormsModule, FeatureToggleModule, - ArtemisComplaintsForTutorModule, + ComplaintsForTutorComponent, ArtemisProgrammingAssessmentRoutingModule, ArtemisAssessmentSharedModule, ArtemisCodeEditorModule, diff --git a/src/main/webapp/app/exercises/programming/participate/programming-repository.module.ts b/src/main/webapp/app/exercises/programming/participate/programming-repository.module.ts index 49617d7f20b1..a765439ae8c9 100644 --- a/src/main/webapp/app/exercises/programming/participate/programming-repository.module.ts +++ b/src/main/webapp/app/exercises/programming/participate/programming-repository.module.ts @@ -10,7 +10,6 @@ import { RepositoryViewComponent } from 'app/localvc/repository-view/repository- import { ArtemisProgrammingRepositoryRoutingModule } from 'app/exercises/programming/participate/programming-repository-routing.module'; import { FormDateTimePickerModule } from 'app/shared/date-time-picker/date-time-picker.module'; import { FeatureToggleModule } from 'app/shared/feature-toggle/feature-toggle.module'; -import { ArtemisComplaintsForTutorModule } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.module'; import { ArtemisAssessmentSharedModule } from 'app/assessment/assessment-shared.module'; import { ArtemisProgrammingManualAssessmentModule } from 'app/exercises/programming/assess/programming-manual-assessment.module'; import { AssessmentInstructionsModule } from 'app/assessment/assessment-instructions/assessment-instructions.module'; @@ -19,7 +18,8 @@ import { ArtemisProgrammingExerciseModule } from 'app/exercises/programming/shar import { CommitHistoryComponent } from 'app/localvc/commit-history/commit-history.component'; import { CommitDetailsViewComponent } from 'app/localvc/commit-details-view/commit-details-view.component'; import { ArtemisProgrammingExerciseActionsModule } from 'app/exercises/programming/shared/actions/programming-exercise-actions.module'; -import { GitDiffReportComponent } from 'app/exercises/programming/git-diff-report/git-diff-report.component'; +import { GitDiffReportComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report.component'; +import { ComplaintsForTutorComponent } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.component'; @NgModule({ imports: [ @@ -28,7 +28,7 @@ import { GitDiffReportComponent } from 'app/exercises/programming/git-diff-repor FormDateTimePickerModule, FormsModule, FeatureToggleModule, - ArtemisComplaintsForTutorModule, + ComplaintsForTutorComponent, ArtemisProgrammingRepositoryRoutingModule, ArtemisAssessmentSharedModule, ArtemisCodeEditorModule, diff --git a/src/main/webapp/app/exercises/text/assess/text-submission-assessment.module.ts b/src/main/webapp/app/exercises/text/assess/text-submission-assessment.module.ts index 0c119a592dbe..a4492e2e7ad7 100644 --- a/src/main/webapp/app/exercises/text/assess/text-submission-assessment.module.ts +++ b/src/main/webapp/app/exercises/text/assess/text-submission-assessment.module.ts @@ -14,13 +14,13 @@ import { ManualTextblockSelectionComponent } from 'app/exercises/text/assess/man import { ArtemisConfirmIconModule } from 'app/shared/confirm-icon/confirm-icon.module'; import { TextSharedModule } from 'app/exercises/text/shared/text-shared.module'; import { ArtemisResultModule } from 'app/exercises/shared/result/result.module'; -import { ArtemisComplaintsForTutorModule } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.module'; import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; import { ArtemisGradingInstructionLinkIconModule } from 'app/shared/grading-instruction-link-icon/grading-instruction-link-icon.module'; import { SubmissionResultStatusModule } from 'app/overview/submission-result-status.module'; import { TextblockFeedbackDropdownComponent } from 'app/exercises/text/assess/textblock-feedback-editor/dropdown/textblock-feedback-dropdown.component'; import { ArtemisAssessmentProgressLabelModule } from 'app/exercises/shared/assessment-progress-label/assessment-progress-label.module'; import { ArtemisFeedbackModule } from 'app/exercises/shared/feedback/feedback.module'; +import { ComplaintsForTutorComponent } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.component'; const ENTITY_STATES = [...textSubmissionAssessmentRoutes]; @@ -30,7 +30,7 @@ const ENTITY_STATES = [...textSubmissionAssessmentRoutes]; RouterModule.forChild(ENTITY_STATES), ArtemisSharedModule, ArtemisResultModule, - ArtemisComplaintsForTutorModule, + ComplaintsForTutorComponent, ArtemisSharedComponentModule, ArtemisAssessmentSharedModule, AssessmentInstructionsModule, diff --git a/src/test/javascript/spec/component/assessment-shared/assessment-layout.component.spec.ts b/src/test/javascript/spec/component/assessment-shared/assessment-layout.component.spec.ts index 8d05c1958d04..31021c4ce9b2 100644 --- a/src/test/javascript/spec/component/assessment-shared/assessment-layout.component.spec.ts +++ b/src/test/javascript/spec/component/assessment-shared/assessment-layout.component.spec.ts @@ -9,7 +9,6 @@ import { Complaint } from 'app/entities/complaint.model'; import { MockComponent, MockDirective, MockModule, MockProvider } from 'ng-mocks'; import { AssessmentWarningComponent } from 'app/assessment/assessment-warning/assessment-warning.component'; import { TranslateDirective } from 'app/shared/language/translate.directive'; -import { TranslatePipeMock } from '../../helpers/mocks/service/mock-translate.service'; import { MockQueryParamsDirective, MockRouterLinkDirective } from '../../helpers/mocks/directive/mock-router-link.directive'; import { TextAssessmentAnalytics } from 'app/exercises/text/assess/analytics/text-assesment-analytics.service'; import { ActivatedRoute } from '@angular/router'; @@ -23,16 +22,14 @@ describe('AssessmentLayoutComponent', () => { beforeEach(() => { TestBed.configureTestingModule({ - imports: [ArtemisTestModule, MockModule(NgbTooltipModule)], + imports: [ArtemisTestModule, MockModule(NgbTooltipModule), MockComponent(ComplaintsForTutorComponent)], declarations: [ AssessmentLayoutComponent, AssessmentHeaderComponent, AssessmentNoteComponent, - MockComponent(ComplaintsForTutorComponent), MockComponent(AssessmentComplaintAlertComponent), MockComponent(AssessmentWarningComponent), MockDirective(TranslateDirective), - TranslatePipeMock, MockRouterLinkDirective, MockQueryParamsDirective, ], diff --git a/src/test/javascript/spec/component/complaints/complaints-for-tutor.component.spec.ts b/src/test/javascript/spec/component/complaints/complaints-for-tutor.component.spec.ts index 867a683b12d1..6e46424e9514 100644 --- a/src/test/javascript/spec/component/complaints/complaints-for-tutor.component.spec.ts +++ b/src/test/javascript/spec/component/complaints/complaints-for-tutor.component.spec.ts @@ -18,6 +18,8 @@ import { Course } from 'app/entities/course.model'; import { Exercise } from 'app/entities/exercise.model'; import { TranslateDirective } from 'app/shared/language/translate.directive'; import { provideRouter } from '@angular/router'; +import { TranslateService } from '@ngx-translate/core'; +import { MockTranslateService } from '../../helpers/mocks/service/mock-translate.service'; describe('ComplaintsForTutorComponent', () => { let complaintsForTutorComponent: ComplaintsForTutorComponent; @@ -30,14 +32,17 @@ describe('ComplaintsForTutorComponent', () => { beforeEach(() => { TestBed.configureTestingModule({ imports: [FormsModule], - declarations: [ - ComplaintsForTutorComponent, - MockPipe(ArtemisTranslatePipe), - MockPipe(ArtemisDatePipe), - MockComponent(TextareaCounterComponent), - MockDirective(TranslateDirective), + declarations: [MockPipe(ArtemisTranslatePipe), MockPipe(ArtemisDatePipe), MockComponent(TextareaCounterComponent), MockDirective(TranslateDirective)], + providers: [ + provideRouter([]), + MockProvider(ComplaintResponseService), + MockProvider(ComplaintService), + MockProvider(AlertService), + { + provide: TranslateService, + useClass: MockTranslateService, + }, ], - providers: [provideRouter([]), MockProvider(ComplaintResponseService), MockProvider(ComplaintService), MockProvider(AlertService)], }) .compileComponents() .then(() => { diff --git a/src/test/javascript/spec/component/complaints/complaints-form.component.spec.ts b/src/test/javascript/spec/component/complaints/complaints-form.component.spec.ts index 488c740be038..2139f0157523 100644 --- a/src/test/javascript/spec/component/complaints/complaints-form.component.spec.ts +++ b/src/test/javascript/spec/component/complaints/complaints-form.component.spec.ts @@ -32,13 +32,7 @@ describe('ComplaintsFormComponent', () => { beforeEach(() => { TestBed.configureTestingModule({ imports: [ArtemisTestModule], - declarations: [ - ComplaintsFormComponent, - MockPipe(ArtemisTranslatePipe), - MockDirective(NgModel), - MockDirective(TranslateDirective), - MockComponent(TextareaCounterComponent), - ], + declarations: [MockPipe(ArtemisTranslatePipe), MockDirective(NgModel), MockDirective(TranslateDirective), MockComponent(TextareaCounterComponent)], providers: [ MockProvider(AlertService), { diff --git a/src/test/javascript/spec/component/complaints/list-of-complaints.component.spec.ts b/src/test/javascript/spec/component/complaints/list-of-complaints.component.spec.ts index e3abe463b09d..c2ca0a831fec 100644 --- a/src/test/javascript/spec/component/complaints/list-of-complaints.component.spec.ts +++ b/src/test/javascript/spec/component/complaints/list-of-complaints.component.spec.ts @@ -1,7 +1,6 @@ import { ComponentFixture, TestBed, fakeAsync, tick } from '@angular/core/testing'; import { ActivatedRoute, Router } from '@angular/router'; import { FaIconComponent } from '@fortawesome/angular-fontawesome'; -import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; import { TranslateService } from '@ngx-translate/core'; import { ComplaintService, EntityResponseTypeArray, IComplaintService } from 'app/complaints/complaint.service'; import { ListOfComplaintsComponent } from 'app/complaints/list-of-complaints/list-of-complaints.component'; @@ -24,8 +23,8 @@ import { MockActivatedRoute } from '../../helpers/mocks/activated-route/mock-act import { MockRouter } from '../../helpers/mocks/mock-router'; import { MockComplaintService } from '../../helpers/mocks/service/mock-complaint.service'; import { MockCourseManagementService } from '../../helpers/mocks/service/mock-course-management.service'; -import { MockNgbModalService } from '../../helpers/mocks/service/mock-ngb-modal.service'; import { MockTranslateService, TranslatePipeMock } from '../../helpers/mocks/service/mock-translate.service'; +import { ArtemisSharedCommonModule } from 'app/shared/shared-common.module'; describe('ListOfComplaintsComponent', () => { let fixture: ComponentFixture; @@ -65,19 +64,21 @@ describe('ListOfComplaintsComponent', () => { beforeEach(() => { TestBed.configureTestingModule({ - declarations: [ListOfComplaintsComponent, TranslatePipeMock, MockComponent(FaIconComponent)], + declarations: [TranslatePipeMock, MockComponent(FaIconComponent)], providers: [ MockProvider(AlertService), MockProvider(SortService), { provide: ComplaintService, useClass: MockComplaintService }, { provide: Router, useClass: MockRouter }, - { provide: NgbModal, useClass: MockNgbModalService }, { provide: ActivatedRoute, useValue: new MockActivatedRoute() }, { provide: TranslateService, useClass: MockTranslateService }, { provide: ArtemisDatePipe, useClass: TranslatePipeMock }, { provide: CourseManagementService, useClass: MockCourseManagementService }, ], }) + .overrideComponent(ListOfComplaintsComponent, { + remove: { imports: [ArtemisSharedCommonModule] }, + }) .compileComponents() .then(() => { fixture = TestBed.createComponent(ListOfComplaintsComponent); From 2648a0325284e83385b02d1ae1526a9214eca6a4 Mon Sep 17 00:00:00 2001 From: Paul Rangger <48455539+PaRangger@users.noreply.github.com> Date: Sun, 5 Jan 2025 21:07:44 +0100 Subject: [PATCH 24/34] Development: Remove unused reference in posting models (#10075) --- .../artemis/communication/domain/AnswerPost.java | 13 ------------- .../cit/aet/artemis/communication/domain/Post.java | 13 ------------- .../repository/ConversationMessageRepository.java | 1 - .../repository/SavedPostRepository.java | 10 ++++++++++ .../communication/service/AnswerMessageService.java | 5 +++++ .../service/ConversationMessagingService.java | 5 +++++ .../service/SavedPostScheduleService.java | 2 +- 7 files changed, 21 insertions(+), 28 deletions(-) diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/domain/AnswerPost.java b/src/main/java/de/tum/cit/aet/artemis/communication/domain/AnswerPost.java index 9469d9e6d818..bcb3218eb621 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/domain/AnswerPost.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/domain/AnswerPost.java @@ -14,7 +14,6 @@ import org.hibernate.annotations.Cache; import org.hibernate.annotations.CacheConcurrencyStrategy; -import org.hibernate.annotations.SQLRestriction; import com.fasterxml.jackson.annotation.JsonIgnore; import com.fasterxml.jackson.annotation.JsonInclude; @@ -39,13 +38,6 @@ public class AnswerPost extends Posting { @OneToMany(mappedBy = "answerPost", cascade = CascadeType.REMOVE, orphanRemoval = true, fetch = FetchType.EAGER) private Set reactions = new HashSet<>(); - /*** - * The value 1 represents an answer post, given by the enum {{@link PostingType}} - */ - @OneToMany(mappedBy = "postId", cascade = CascadeType.REMOVE, orphanRemoval = true, fetch = FetchType.LAZY) - @SQLRestriction("post_type = 1") - private Set savedPosts = new HashSet<>(); - @ManyToOne @JsonIncludeProperties({ "id", "exercise", "lecture", "course", "courseWideContext", "conversation", "author" }) private Post post; @@ -90,11 +82,6 @@ public void setPost(Post post) { this.post = post; } - @JsonIgnore - public Set getSavedPosts() { - return savedPosts; - } - @JsonProperty("isSaved") public boolean getIsSaved() { return isSaved; diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/domain/Post.java b/src/main/java/de/tum/cit/aet/artemis/communication/domain/Post.java index 3bb92cb6a540..f6511017f1c6 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/domain/Post.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/domain/Post.java @@ -21,7 +21,6 @@ import org.hibernate.annotations.Cache; import org.hibernate.annotations.CacheConcurrencyStrategy; -import org.hibernate.annotations.SQLRestriction; import com.fasterxml.jackson.annotation.JsonIgnore; import com.fasterxml.jackson.annotation.JsonInclude; @@ -57,13 +56,6 @@ public class Post extends Posting { @OneToMany(mappedBy = "post", cascade = CascadeType.REMOVE, orphanRemoval = true, fetch = FetchType.EAGER) private Set answers = new HashSet<>(); - /*** - * The value 0 represents a post, given by the enum {{@link PostingType}} - */ - @OneToMany(mappedBy = "postId", cascade = CascadeType.REMOVE, orphanRemoval = true, fetch = FetchType.LAZY) - @SQLRestriction("post_type = 0") - private Set savedPosts = new HashSet<>(); - @ElementCollection(fetch = FetchType.EAGER) @CollectionTable(name = "post_tag", joinColumns = @JoinColumn(name = "post_id")) @Column(name = "text") @@ -235,11 +227,6 @@ public void setVoteCount(Integer voteCount) { this.voteCount = voteCount != null ? voteCount : 0; } - @JsonIgnore - public Set getSavedPosts() { - return savedPosts; - } - @JsonProperty("isSaved") public boolean getIsSaved() { return isSaved; diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/repository/ConversationMessageRepository.java b/src/main/java/de/tum/cit/aet/artemis/communication/repository/ConversationMessageRepository.java index 16c5be3aedc8..2952c5213432 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/repository/ConversationMessageRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/repository/ConversationMessageRepository.java @@ -114,7 +114,6 @@ private PageImpl findPostsWithSpecification(Pageable pageable, Specificati LEFT JOIN FETCH p.conversation LEFT JOIN FETCH p.reactions LEFT JOIN FETCH p.tags - LEFT JOIN FETCH p.savedPosts LEFT JOIN FETCH p.answers a LEFT JOIN FETCH a.reactions LEFT JOIN FETCH a.post diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/repository/SavedPostRepository.java b/src/main/java/de/tum/cit/aet/artemis/communication/repository/SavedPostRepository.java index e0a00a5896aa..f490650f4fc4 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/repository/SavedPostRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/repository/SavedPostRepository.java @@ -46,6 +46,16 @@ public interface SavedPostRepository extends ArtemisJpaRepository findSavedPostByPostIdAndPostType(Long postId, PostingType postType); + /*** * Query all post ids that a user has saved by a certain posting type. Cached by user id and post type. * diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/service/AnswerMessageService.java b/src/main/java/de/tum/cit/aet/artemis/communication/service/AnswerMessageService.java index f7645c202f63..d195ef870d7a 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/service/AnswerMessageService.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/service/AnswerMessageService.java @@ -12,6 +12,7 @@ import de.tum.cit.aet.artemis.communication.domain.AnswerPost; import de.tum.cit.aet.artemis.communication.domain.Post; +import de.tum.cit.aet.artemis.communication.domain.PostingType; import de.tum.cit.aet.artemis.communication.domain.conversation.Channel; import de.tum.cit.aet.artemis.communication.domain.conversation.Conversation; import de.tum.cit.aet.artemis.communication.domain.notification.SingleUserNotification; @@ -209,6 +210,10 @@ public void deleteAnswerMessageById(Long courseId, Long answerMessageId) { answerPostRepository.deleteById(answerMessageId); preparePostForBroadcast(updatedMessage); + // Delete all connected saved posts + var savedPosts = savedPostRepository.findSavedPostByPostIdAndPostType(answerMessageId, PostingType.ANSWER); + savedPostRepository.deleteAll(savedPosts); + broadcastForPost(new PostDTO(updatedMessage, MetisCrudAction.UPDATE), course.getId(), null, null); } diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/service/ConversationMessagingService.java b/src/main/java/de/tum/cit/aet/artemis/communication/service/ConversationMessagingService.java index 06f9409bddc0..9f66ebb10a21 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/service/ConversationMessagingService.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/service/ConversationMessagingService.java @@ -27,6 +27,7 @@ import de.tum.cit.aet.artemis.communication.domain.DisplayPriority; import de.tum.cit.aet.artemis.communication.domain.NotificationType; import de.tum.cit.aet.artemis.communication.domain.Post; +import de.tum.cit.aet.artemis.communication.domain.PostingType; import de.tum.cit.aet.artemis.communication.domain.conversation.Channel; import de.tum.cit.aet.artemis.communication.domain.conversation.Conversation; import de.tum.cit.aet.artemis.communication.domain.conversation.GroupChat; @@ -371,6 +372,10 @@ public void deleteMessageById(Long courseId, Long postId) { conversationParticipantRepository.decrementUnreadMessagesCountOfParticipants(conversation.getId(), user.getId()); conversation = conversationService.getConversationById(conversation.getId()); + // Delete all connected saved posts + var savedPosts = savedPostRepository.findSavedPostByPostIdAndPostType(postId, PostingType.POST); + savedPostRepository.deleteAll(savedPosts); + conversationService.notifyAllConversationMembersAboutUpdate(conversation); preparePostForBroadcast(post); broadcastForPost(new PostDTO(post, MetisCrudAction.DELETE), course.getId(), null, null); diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/service/SavedPostScheduleService.java b/src/main/java/de/tum/cit/aet/artemis/communication/service/SavedPostScheduleService.java index 25e5922031f5..e2487b77e887 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/service/SavedPostScheduleService.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/service/SavedPostScheduleService.java @@ -45,7 +45,7 @@ public void cleanupArchivedSavedPosts() { /** * Cleans up all saved posts where the post entity does not exist anymore */ - @Scheduled(cron = "0 0 0 * * *") + @Scheduled(cron = "0 5 0 * * *") public void cleanupOrphanedSavedPosts() { List orphanedPosts = savedPostRepository.findOrphanedPostReferences(); if (!orphanedPosts.isEmpty()) { From 3267090f81931b411a734bffe0015503b6c4fd36 Mon Sep 17 00:00:00 2001 From: Julian Gassner Date: Sun, 5 Jan 2025 21:11:26 +0100 Subject: [PATCH 25/34] Development: Change posting deletion logic to not abort when deconstructing a post (#10073) --- .../webapp/app/shared/metis/posting.directive.ts | 15 ++++++++++----- 1 file changed, 10 insertions(+), 5 deletions(-) diff --git a/src/main/webapp/app/shared/metis/posting.directive.ts b/src/main/webapp/app/shared/metis/posting.directive.ts index 71c40bf0b12e..a9896c5a1ffc 100644 --- a/src/main/webapp/app/shared/metis/posting.directive.ts +++ b/src/main/webapp/app/shared/metis/posting.directive.ts @@ -49,6 +49,7 @@ export abstract class PostingDirective implements OnInit, OnD ngOnDestroy(): void { if (this.deleteTimer !== undefined) { clearTimeout(this.deleteTimer); + this.deletePostingWithoutTimeout(); } if (this.deleteInterval !== undefined) { @@ -72,11 +73,7 @@ export abstract class PostingDirective implements OnInit, OnD this.deleteTimer = setTimeout( () => { - if (this.isAnswerPost) { - this.metisService.deleteAnswerPost(this.posting); - } else { - this.metisService.deletePost(this.posting); - } + this.deletePostingWithoutTimeout(); }, // We add a tiny buffer to make it possible for the user to react a bit longer than the ui displays (+1000) this.deleteTimerInSeconds * 1000 + 1000, @@ -139,6 +136,14 @@ export abstract class PostingDirective implements OnInit, OnD } } + private deletePostingWithoutTimeout() { + if (this.isAnswerPost) { + this.metisService.deleteAnswerPost(this.posting); + } else { + this.metisService.deletePost(this.posting); + } + } + /** * Create a or navigate to one-to-one chat with the referenced user * From 333b9e9a8174f8aa9a43148236986b90f50d457b Mon Sep 17 00:00:00 2001 From: Julian Gassner Date: Sun, 5 Jan 2025 21:13:04 +0100 Subject: [PATCH 26/34] Programming exercises: Fix image export in combination with tooltip text (#10070) --- .../ExerciseWithSubmissionsExportService.java | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/src/main/java/de/tum/cit/aet/artemis/exercise/service/ExerciseWithSubmissionsExportService.java b/src/main/java/de/tum/cit/aet/artemis/exercise/service/ExerciseWithSubmissionsExportService.java index 1ccd03d928fc..381591ecdb85 100644 --- a/src/main/java/de/tum/cit/aet/artemis/exercise/service/ExerciseWithSubmissionsExportService.java +++ b/src/main/java/de/tum/cit/aet/artemis/exercise/service/ExerciseWithSubmissionsExportService.java @@ -43,6 +43,8 @@ public abstract class ExerciseWithSubmissionsExportService { private static final String EMBEDDED_FILE_MARKDOWN_SYNTAX_REGEX = "\\[.*] *\\(/api/files/markdown/.*\\)"; + private static final String EMBEDDED_FILE_MARKDOWN_WITH_HOVERTEXT = "\\(/api/files/markdown/.* \".*\"\\)"; + private static final String EMBEDDED_FILE_HTML_SYNTAX_REGEX = ""; private static final String API_MARKDOWN_FILE_PATH = "/api/files/markdown/"; @@ -129,7 +131,15 @@ private void copyFilesEmbeddedWithMarkdownSyntax(Exercise exercise, List for (String embeddedFile : embeddedFilesWithMarkdownSyntax) { // avoid matching other closing ] or () in the squared brackets by getting the index of the last ] String lastPartOfMatchedString = embeddedFile.substring(embeddedFile.lastIndexOf("]") + 1); - String filePath = lastPartOfMatchedString.substring(lastPartOfMatchedString.indexOf("(") + 1, lastPartOfMatchedString.indexOf(")")); + String filePath; + + if (Pattern.compile(EMBEDDED_FILE_MARKDOWN_WITH_HOVERTEXT).matcher(lastPartOfMatchedString).matches()) { + filePath = lastPartOfMatchedString.substring(lastPartOfMatchedString.indexOf("(") + 1, lastPartOfMatchedString.indexOf(" ")); + } + else { + filePath = lastPartOfMatchedString.substring(lastPartOfMatchedString.indexOf("(") + 1, lastPartOfMatchedString.indexOf(")")); + } + constructFilenameAndCopyFile(exercise, exportErrors, embeddedFilesDir, filePath); } } From 1f9326ad028d5c42097f4020f63c365b3caaaca8 Mon Sep 17 00:00:00 2001 From: Murad Talibov <56686446+muradium@users.noreply.github.com> Date: Sun, 5 Jan 2025 21:15:36 +0100 Subject: [PATCH 27/34] Development: Add playwright exam checklist e2e tests (#10050) --- .../exam-checklist.component.html | 48 ++- .../checklist-check.component.html | 4 +- .../e2e/exam/ExamAssessment.spec.ts | 124 +------ .../e2e/exam/ExamChecklists.spec.ts | 308 ++++++++++++++++++ .../e2e/exam/ExamCreationDeletion.spec.ts | 111 ++++--- .../e2e/exam/ExamParticipation.spec.ts | 70 ++-- src/test/playwright/support/fixtures.ts | 5 - .../support/pageobjects/exam/EditExamPage.ts | 13 - .../pageobjects/exam/ExamCreationPage.ts | 21 ++ .../pageobjects/exam/ExamDetailsPage.ts | 67 ++++ .../exam/ExamExerciseGroupCreationPage.ts | 37 ++- .../exam/StudentExamManagementPage.ts | 4 + .../support/requests/ExamAPIRequests.ts | 21 +- .../support/requests/ExerciseAPIRequests.ts | 11 +- src/test/playwright/support/utils.ts | 127 +++++++- 15 files changed, 718 insertions(+), 253 deletions(-) create mode 100644 src/test/playwright/e2e/exam/ExamChecklists.spec.ts delete mode 100644 src/test/playwright/support/pageobjects/exam/EditExamPage.ts diff --git a/src/main/webapp/app/exam/manage/exams/exam-checklist-component/exam-checklist.component.html b/src/main/webapp/app/exam/manage/exams/exam-checklist-component/exam-checklist.component.html index d0525ddc1928..8e773545e824 100644 --- a/src/main/webapp/app/exam/manage/exams/exam-checklist-component/exam-checklist.component.html +++ b/src/main/webapp/app/exam/manage/exams/exam-checklist-component/exam-checklist.component.html @@ -50,7 +50,10 @@

  • - +
  • @@ -63,6 +66,7 @@

  • - +
  • - +
  • - +
  • @@ -139,7 +143,7 @@

  • - +
  • @@ -150,7 +154,7 @@

    + @@ -167,14 +171,14 @@

  • - +
  • - + @@ -191,14 +195,17 @@

    @if (examChecklist) {
  • - +
  • } - + @@ -461,7 +468,7 @@

  • - +
  • @if (exam.publishResultsDate) { @@ -487,7 +494,10 @@

    - +

    @if (isEvaluatingQuizExercises) { } @else { - + } @@ -536,7 +549,10 @@

    @if (isAssessingUnsubmittedExams) { } @else { - + } @@ -580,13 +596,13 @@

    • - +
    • - +
    • diff --git a/src/main/webapp/app/shared/components/checklist-check/checklist-check.component.html b/src/main/webapp/app/shared/components/checklist-check/checklist-check.component.html index cae4278c4ea8..bc24834155d5 100644 --- a/src/main/webapp/app/shared/components/checklist-check/checklist-check.component.html +++ b/src/main/webapp/app/shared/components/checklist-check/checklist-check.component.html @@ -1,6 +1,6 @@ @if (checkAttribute) { - + } @else { - + }   diff --git a/src/test/playwright/e2e/exam/ExamAssessment.spec.ts b/src/test/playwright/e2e/exam/ExamAssessment.spec.ts index 4881c7337718..dda83bae8047 100644 --- a/src/test/playwright/e2e/exam/ExamAssessment.spec.ts +++ b/src/test/playwright/e2e/exam/ExamAssessment.spec.ts @@ -1,33 +1,19 @@ import dayjs, { Dayjs } from 'dayjs'; -import { Exercise, ExerciseType, ProgrammingExerciseAssessmentType } from '../../support/constants'; +import { Exercise, ExerciseType } from '../../support/constants'; import { admin, instructor, studentFour, studentOne, studentThree, studentTwo, tutor, users } from '../../support/users'; import { Page, expect } from '@playwright/test'; -import javaPartiallySuccessful from '../../fixtures/exercise/programming/java/partially_successful/submission.json'; - import { Course } from 'app/entities/course.model'; import { Exam } from 'app/entities/exam/exam.model'; import { Commands } from '../../support/commands'; -import { ExamAPIRequests } from '../../support/requests/ExamAPIRequests'; -import { ExamExerciseGroupCreationPage } from '../../support/pageobjects/exam/ExamExerciseGroupCreationPage'; -import { ExamParticipationPage } from '../../support/pageobjects/exam/ExamParticipationPage'; -import { ExamNavigationBar } from '../../support/pageobjects/exam/ExamNavigationBar'; -import { ExamStartEndPage } from '../../support/pageobjects/exam/ExamStartEndPage'; import { ExamManagementPage } from '../../support/pageobjects/exam/ExamManagementPage'; import { CourseAssessmentDashboardPage } from '../../support/pageobjects/assessment/CourseAssessmentDashboardPage'; import { ExerciseAssessmentDashboardPage } from '../../support/pageobjects/assessment/ExerciseAssessmentDashboardPage'; import { StudentAssessmentPage } from '../../support/pageobjects/assessment/StudentAssessmentPage'; import { ExamAssessmentPage } from '../../support/pageobjects/assessment/ExamAssessmentPage'; import { test } from '../../support/fixtures'; -import { ExerciseAPIRequests } from '../../support/requests/ExerciseAPIRequests'; -import { CoursesPage } from '../../support/pageobjects/course/CoursesPage'; -import { CourseOverviewPage } from '../../support/pageobjects/course/CourseOverviewPage'; -import { ModelingEditor } from '../../support/pageobjects/exercises/modeling/ModelingEditor'; -import { OnlineEditorPage } from '../../support/pageobjects/exercises/programming/OnlineEditorPage'; -import { MultipleChoiceQuiz } from '../../support/pageobjects/exercises/quiz/MultipleChoiceQuiz'; -import { TextEditorPage } from '../../support/pageobjects/exercises/text/TextEditorPage'; import { CourseManagementAPIRequests } from '../../support/requests/CourseManagementAPIRequests'; -import { generateUUID, newBrowserPage } from '../../support/utils'; +import { generateUUID, newBrowserPage, prepareExam, startAssessing } from '../../support/utils'; import examStatisticsSample from '../../fixtures/exam/statistics.json'; import { ExamScoresPage } from '../../support/pageobjects/exam/ExamScoresPage'; @@ -60,7 +46,7 @@ test.describe('Exam assessment', () => { test.beforeAll('Prepare exam', async ({ browser }) => { examEnd = dayjs().add(2, 'minutes'); const page = await newBrowserPage(browser); - await prepareExam(course, examEnd, ExerciseType.PROGRAMMING, page); + exam = await prepareExam(course, examEnd, ExerciseType.PROGRAMMING, page); }); test('Assess a programming exercise submission (MANUAL)', async ({ login, examManagement, examAssessment, examParticipation, courseAssessment, exerciseAssessment }) => { @@ -84,7 +70,7 @@ test.describe('Exam assessment', () => { test.beforeAll('Prepare exam', async ({ browser }) => { examEnd = dayjs().add(45, 'seconds'); const page = await newBrowserPage(browser); - await prepareExam(course, examEnd, ExerciseType.MODELING, page); + exam = await prepareExam(course, examEnd, ExerciseType.MODELING, page); }); test('Assess a modeling exercise submission', async ({ @@ -122,7 +108,7 @@ test.describe('Exam assessment', () => { test.beforeAll('Prepare exam', async ({ browser }) => { examEnd = dayjs().add(20, 'seconds'); const page = await newBrowserPage(browser); - await prepareExam(course, examEnd, ExerciseType.TEXT, page, 2); + exam = await prepareExam(course, examEnd, ExerciseType.TEXT, page, 2); }); test('Assess a text exercise submission', async ({ login, examManagement, examAssessment, examParticipation, courseAssessment, exerciseAssessment }) => { @@ -159,7 +145,7 @@ test.describe('Exam assessment', () => { examEnd = dayjs().add(30, 'seconds'); resultDate = examEnd.add(5, 'seconds'); const page = await newBrowserPage(browser); - await prepareExam(course, examEnd, ExerciseType.QUIZ, page); + exam = await prepareExam(course, examEnd, ExerciseType.QUIZ, page); }); test('Assesses quiz automatically', async ({ page, login, examManagement, courseAssessment, examParticipation }) => { @@ -294,104 +280,6 @@ test.afterAll('Delete course', async ({ browser }) => { await courseManagementAPIRequests.deleteCourse(course, admin); }); -export async function prepareExam(course: Course, end: dayjs.Dayjs, exerciseType: ExerciseType, page: Page, numberOfCorrectionRounds: number = 1): Promise { - const examAPIRequests = new ExamAPIRequests(page); - const exerciseAPIRequests = new ExerciseAPIRequests(page); - const examExerciseGroupCreation = new ExamExerciseGroupCreationPage(page, examAPIRequests, exerciseAPIRequests); - const courseList = new CoursesPage(page); - const courseOverview = new CourseOverviewPage(page); - const modelingExerciseEditor = new ModelingEditor(page); - const programmingExerciseEditor = new OnlineEditorPage(page); - const quizExerciseMultipleChoice = new MultipleChoiceQuiz(page); - const textExerciseEditor = new TextEditorPage(page); - const examNavigation = new ExamNavigationBar(page); - const examStartEnd = new ExamStartEndPage(page); - const examParticipation = new ExamParticipationPage( - courseList, - courseOverview, - examNavigation, - examStartEnd, - modelingExerciseEditor, - programmingExerciseEditor, - quizExerciseMultipleChoice, - textExerciseEditor, - page, - ); - - await Commands.login(page, admin); - const resultDate = end.add(1, 'second'); - const examConfig = { - course, - startDate: dayjs(), - endDate: end, - numberOfCorrectionRoundsInExam: numberOfCorrectionRounds, - examStudentReviewStart: resultDate, - examStudentReviewEnd: resultDate.add(1, 'minute'), - publishResultsDate: resultDate, - gracePeriod: 10, - }; - exam = await examAPIRequests.createExam(examConfig); - let additionalData = {}; - switch (exerciseType) { - case ExerciseType.PROGRAMMING: - additionalData = { submission: javaPartiallySuccessful, progExerciseAssessmentType: ProgrammingExerciseAssessmentType.SEMI_AUTOMATIC }; - break; - case ExerciseType.TEXT: - additionalData = { textFixture: 'loremIpsum-short.txt' }; - break; - case ExerciseType.QUIZ: - additionalData = { quizExerciseID: 0 }; - break; - } - - const exercise = await examExerciseGroupCreation.addGroupWithExercise(exam, exerciseType, additionalData); - await examAPIRequests.registerStudentForExam(exam, studentOne); - await examAPIRequests.generateMissingIndividualExams(exam); - await examAPIRequests.prepareExerciseStartForExam(exam); - exercise.additionalData = additionalData; - await makeExamSubmission(course, exam, exercise, page, examParticipation, examNavigation, examStartEnd); - return exam; -} - -async function makeExamSubmission( - course: Course, - exam: Exam, - exercise: Exercise, - page: Page, - examParticipation: ExamParticipationPage, - examNavigation: ExamNavigationBar, - examStartEnd: ExamStartEndPage, -) { - await examParticipation.startParticipation(studentOne, course, exam); - await examNavigation.openOrSaveExerciseByTitle(exercise.exerciseGroup!.title!); - await examParticipation.makeSubmission(exercise.id!, exercise.type!, exercise.additionalData); - await page.waitForTimeout(2000); - await examNavigation.handInEarly(); - await examStartEnd.finishExam(); -} - -async function startAssessing( - courseID: number, - examID: number, - timeout: number, - examManagement: ExamManagementPage, - courseAssessment: CourseAssessmentDashboardPage, - exerciseAssessment: ExerciseAssessmentDashboardPage, - toggleSecondRound: boolean = false, - isFirstTimeAssessing: boolean = true, -) { - await examManagement.openAssessmentDashboard(courseID, examID, timeout); - await courseAssessment.clickExerciseDashboardButton(); - if (toggleSecondRound) { - await exerciseAssessment.toggleSecondCorrectionRound(); - } - if (isFirstTimeAssessing) { - await exerciseAssessment.clickHaveReadInstructionsButton(); - } - await exerciseAssessment.clickStartNewAssessment(); - exerciseAssessment.getLockedMessage(); -} - async function handleComplaint( course: Course, exam: Exam, diff --git a/src/test/playwright/e2e/exam/ExamChecklists.spec.ts b/src/test/playwright/e2e/exam/ExamChecklists.spec.ts new file mode 100644 index 000000000000..8b1d943f863f --- /dev/null +++ b/src/test/playwright/e2e/exam/ExamChecklists.spec.ts @@ -0,0 +1,308 @@ +import { test } from '../../support/fixtures'; +import { admin, instructor, studentOne } from '../../support/users'; +import { Course } from 'app/entities/course.model'; +import { Exam } from 'app/entities/exam/exam.model'; +import { generateUUID, prepareExam, startAssessing } from '../../support/utils'; +import dayjs from 'dayjs'; +import { ExamChecklistItem } from '../../support/pageobjects/exam/ExamDetailsPage'; +import { ExerciseType } from '../../support/constants'; +import textExerciseTemplate from '../../fixtures/exercise/text/template.json'; +import { ExamExerciseGroupCreationPage } from '../../support/pageobjects/exam/ExamExerciseGroupCreationPage'; +import { ExamExerciseGroupsPage } from '../../support/pageobjects/exam/ExamExerciseGroupsPage'; +import { Page } from '@playwright/test'; +import { Commands } from '../../support/commands'; +import { ExamAPIRequests } from '../../support/requests/ExamAPIRequests'; + +test.describe('Exam Checklists', async () => { + let course: Course; + + test.beforeEach('Create course', async ({ login, courseManagementAPIRequests }) => { + await login(admin); + course = await courseManagementAPIRequests.createCourse({ customizeGroups: true }); + await courseManagementAPIRequests.addStudentToCourse(course, studentOne); + }); + + test.describe('Exercise group checks', { tag: '@fast' }, () => { + test('Instructor adds an exercise group and at least one exercise group check is marked', async ({ + page, + login, + examDetails, + examExerciseGroups, + examExerciseGroupCreation, + }) => { + const exam = await createExam(course, page); + await login(instructor); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.LEAST_ONE_EXERCISE_GROUP); + await examDetails.openExerciseGroups(); + await examExerciseGroups.clickAddExerciseGroup(); + await examExerciseGroupCreation.typeTitle('Group 1'); + await examExerciseGroupCreation.clickSave(); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemChecked(ExamChecklistItem.LEAST_ONE_EXERCISE_GROUP); + }); + + test('Instructor adds exercise groups and the number of exercise groups check is correctly reacting to changes', async ({ + page, + login, + examDetails, + examExerciseGroups, + examExerciseGroupCreation, + }) => { + const exam = await createExam(course, page); + await login(instructor); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.NUMBER_OF_EXERCISE_GROUPS); + await examDetails.openExerciseGroups(); + for (let i = 0; i < exam.numberOfExercisesInExam!; i++) { + await addExamExerciseGroup(examExerciseGroups, examExerciseGroupCreation); + } + await navigateToExamDetailsPage(page, course, exam); + await examDetails.openExerciseGroups(); + await addExamExerciseGroup(examExerciseGroups, examExerciseGroupCreation, false); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemChecked(ExamChecklistItem.NUMBER_OF_EXERCISE_GROUPS); + await examDetails.openExerciseGroups(); + await addExamExerciseGroup(examExerciseGroups, examExerciseGroupCreation); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.NUMBER_OF_EXERCISE_GROUPS); + }); + + test('Instructor adds exercise groups and each exercise group has exercises check is correctly reacting to changes', async ({ + page, + login, + examDetails, + examExerciseGroups, + examExerciseGroupCreation, + }) => { + const exam = await createExam(course, page); + await login(instructor); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.EACH_EXERCISE_GROUP_HAS_EXERCISES); + await examExerciseGroupCreation.addGroupWithExercise(exam, ExerciseType.TEXT); + await page.reload(); + await examDetails.checkItemChecked(ExamChecklistItem.EACH_EXERCISE_GROUP_HAS_EXERCISES); + await examDetails.openExerciseGroups(); + await examExerciseGroups.clickAddExerciseGroup(); + await examExerciseGroupCreation.typeTitle('Empty group'); + await examExerciseGroupCreation.clickSave(); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.EACH_EXERCISE_GROUP_HAS_EXERCISES); + }); + + test('Instructor adds exercise groups and points in exercise groups equal check is correctly reacting to changes', async ({ + page, + login, + examDetails, + examAPIRequests, + exerciseAPIRequests, + }) => { + const exam = await createExam(course, page); + await login(instructor); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.POINTS_IN_EXERCISE_GROUPS_EQUAL); + const exerciseGroup = await examAPIRequests.addExerciseGroupForExam(exam); + await exerciseAPIRequests.createTextExercise({ exerciseGroup }, 'Exercise ' + generateUUID(), textExerciseTemplate); + await page.reload(); + await examDetails.checkItemChecked(ExamChecklistItem.POINTS_IN_EXERCISE_GROUPS_EQUAL); + const maxPointsOfFirstExercise = textExerciseTemplate.maxPoints; + const exerciseTemplate = { ...textExerciseTemplate }; + exerciseTemplate.maxPoints = maxPointsOfFirstExercise - 1; + await exerciseAPIRequests.createTextExercise({ exerciseGroup }, 'Exercise ' + generateUUID(), exerciseTemplate); + await page.reload(); + await examDetails.checkItemUnchecked(ExamChecklistItem.POINTS_IN_EXERCISE_GROUPS_EQUAL); + }); + + test('Instructor adds exercise groups and total points possible check is correctly reacting to changes', async ({ + page, + login, + examDetails, + examExerciseGroupCreation, + }) => { + const exam = await createExam(course, page); + await login(instructor); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.TOTAL_POINTS_POSSIBLE); + await examExerciseGroupCreation.addGroupWithExercise(exam, ExerciseType.TEXT); + await page.reload(); + await examDetails.checkItemUnchecked(ExamChecklistItem.TOTAL_POINTS_POSSIBLE); + await examExerciseGroupCreation.addGroupWithExercise(exam, ExerciseType.TEXT, {}, false); + await page.reload(); + await examDetails.checkItemChecked(ExamChecklistItem.TOTAL_POINTS_POSSIBLE); + await examExerciseGroupCreation.addGroupWithExercise(exam, ExerciseType.TEXT); + await page.reload(); + await examDetails.checkItemChecked(ExamChecklistItem.TOTAL_POINTS_POSSIBLE); + await examExerciseGroupCreation.addGroupWithExercise(exam, ExerciseType.TEXT); + await page.reload(); + await examDetails.checkItemUnchecked(ExamChecklistItem.TOTAL_POINTS_POSSIBLE); + }); + }); + + test('Instructor registers a student to exam and at least one student check is marked', { tag: '@fast' }, async ({ page, login, examDetails, studentExamManagement }) => { + const exam = await createExam(course, page); + await login(instructor); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.LEAST_ONE_STUDENT); + await examDetails.clickStudentsToRegister(); + await studentExamManagement.clickRegisterCourseStudents(); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemChecked(ExamChecklistItem.LEAST_ONE_STUDENT); + }); + + test.describe('Individual exam generation and exam preparation checks', { tag: '@fast' }, () => { + let exam: Exam; + + test.beforeEach('Create exam', async ({ page }) => { + exam = await createExam(course, page); + }); + + test.beforeEach('Add exercise groups and register exam students', async ({ login, examExerciseGroupCreation, examAPIRequests }) => { + await login(admin); + for (let i = 0; i < exam.numberOfExercisesInExam!; i++) { + await examExerciseGroupCreation.addGroupWithExercise(exam, ExerciseType.TEXT); + } + await examAPIRequests.registerAllCourseStudentsForExam(exam); + }); + + test('Instructor generates individual exams, prepares exercises for start and corresponding checks are marked', async ({ + page, + login, + examDetails, + studentExamManagement, + }) => { + await login(instructor); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.ALL_EXAMS_GENERATED); + await examDetails.checkItemUnchecked(ExamChecklistItem.ALL_EXERCISES_PREPARED); + await examDetails.clickStudentExamsToGenerate(); + await studentExamManagement.clickGenerateStudentExams(); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemChecked(ExamChecklistItem.ALL_EXAMS_GENERATED); + await examDetails.checkItemUnchecked(ExamChecklistItem.ALL_EXERCISES_PREPARED); + await examDetails.clickStudentExamsToPrepareStart(); + await studentExamManagement.clickPrepareExerciseStart(); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemChecked(ExamChecklistItem.ALL_EXAMS_GENERATED); + await examDetails.checkItemChecked(ExamChecklistItem.ALL_EXERCISES_PREPARED); + }); + }); + + test('Instructor sets the publish results and review dates and the corresponding checks are marked', { tag: '@fast' }, async ({ page, login, examDetails, examCreation }) => { + const exam = await createExam(course, page); + await login(instructor); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.PUBLISHING_DATE_SET); + await examDetails.checkItemUnchecked(ExamChecklistItem.START_DATE_REVIEW_SET); + await examDetails.checkItemUnchecked(ExamChecklistItem.END_DATE_REVIEW_SET); + await examDetails.clickEditExamForPublishDate(); + const examEndDate = dayjs(exam.endDate! as dayjs.Dayjs); + await examCreation.setPublishResultsDate(examEndDate.add(1, 'hour')); + await examCreation.update(); + await page.waitForURL(`**/exams/${exam.id}`); + await examDetails.checkItemChecked(ExamChecklistItem.PUBLISHING_DATE_SET); + await examDetails.checkItemUnchecked(ExamChecklistItem.START_DATE_REVIEW_SET); + await examDetails.checkItemUnchecked(ExamChecklistItem.END_DATE_REVIEW_SET); + await examDetails.clickEditExamForReviewDate(); + await examCreation.setStudentReviewStartDate(examEndDate.add(2, 'hour')); + await examCreation.setStudentReviewEndDate(examEndDate.add(1, 'day')); + await examCreation.update(); + await page.waitForURL(`**/exams/${exam.id}`); + await examDetails.checkItemChecked(ExamChecklistItem.PUBLISHING_DATE_SET); + await examDetails.checkItemChecked(ExamChecklistItem.START_DATE_REVIEW_SET); + await examDetails.checkItemChecked(ExamChecklistItem.END_DATE_REVIEW_SET); + }); + + test( + 'Student makes a submission and missing assessment check is marked for instructor after assessment', + { tag: '@slow' }, + async ({ page, login, examDetails, examManagement, courseAssessment, exerciseAssessment, textExerciseAssessment, examAPIRequests }) => { + const exam = await prepareExam(course, dayjs().add(1, 'day'), ExerciseType.TEXT, page); + await login(instructor); + await examAPIRequests.finishExam(exam); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.UNFINISHED_ASSESSMENTS); + await startAssessing(course.id!, exam.id!, 60000, examManagement, courseAssessment, exerciseAssessment); + await textExerciseAssessment.addNewFeedback(5, 'OK'); + await textExerciseAssessment.submit(); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemChecked(ExamChecklistItem.UNFINISHED_ASSESSMENTS); + }, + ); + + // This test is skipped for now because it currently fails due to a known issue: + // https://github.com/ls1intum/Artemis/issues/10074 + test.skip( + 'Student makes a quiz submission and unassessed quizzes check is marked for instructor after assessment', + { tag: '@slow' }, + async ({ page, login, examDetails, examAPIRequests }) => { + const exam = await prepareExam(course, dayjs().add(1, 'day'), ExerciseType.QUIZ, page); + await login(instructor); + await examAPIRequests.finishExam(exam); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.UNASSESSED_QUIZZES); + await examDetails.clickEvaluateQuizExercises(); + await examDetails.checkItemChecked(ExamChecklistItem.UNASSESSED_QUIZZES); + }, + ); + + // This test is skipped for now because it currently fails due to a known issue: + // https://github.com/ls1intum/Artemis/issues/10076 + test.skip( + 'Student does not submit the exam on time and corresponding check is marked', + { tag: '@slow' }, + async ({ page, login, examDetails, examAPIRequests, examExerciseGroupCreation, examParticipation }) => { + const examConfig = { + startDate: dayjs(), + endDate: dayjs().add(1, 'day'), + publishResultsDate: dayjs().add(2, 'day'), + }; + const exam = await createExam(course, page, examConfig); + for (let i = 0; i < exam.numberOfExercisesInExam!; i++) { + await examExerciseGroupCreation.addGroupWithExercise(exam, ExerciseType.TEXT, { textFixture: 'loremIpsum-short.txt' }); + } + await examAPIRequests.registerStudentForExam(exam, studentOne); + await examAPIRequests.generateMissingIndividualExams(exam); + await examAPIRequests.prepareExerciseStartForExam(exam); + await examParticipation.startParticipation(studentOne, course, exam); + await login(instructor); + await examAPIRequests.finishExam(exam); + await navigateToExamDetailsPage(page, course, exam); + await examDetails.checkItemUnchecked(ExamChecklistItem.UNSUBMITTED_EXERCISES); + await examDetails.clickAssessUnsubmittedParticipations(); + await examDetails.checkItemChecked(ExamChecklistItem.UNSUBMITTED_EXERCISES); + }, + ); + + test.afterEach('Delete course', async ({ courseManagementAPIRequests }) => { + await courseManagementAPIRequests.deleteCourse(course, admin); + }); +}); + +async function createExam(course: Course, page: Page, customConfig?: any) { + const NUMBER_OF_EXERCISES = 2; + const EXAM_MAX_POINTS = NUMBER_OF_EXERCISES * 10; + + await Commands.login(page, admin); + const examConfig = { + ...customConfig, + course, + examMaxPoints: EXAM_MAX_POINTS, + numberOfExercisesInExam: NUMBER_OF_EXERCISES, + }; + + const examAPIRequests = new ExamAPIRequests(page); + return await examAPIRequests.createExam(examConfig); +} + +async function navigateToExamDetailsPage(page: Page, course: Course, exam: Exam) { + await page.goto(`/course-management/${course.id}/exams/${exam.id}`); +} + +async function addExamExerciseGroup(examExerciseGroups: ExamExerciseGroupsPage, examExerciseGroupCreation: ExamExerciseGroupCreationPage, isMandatory?: boolean) { + await examExerciseGroups.clickAddExerciseGroup(); + await examExerciseGroupCreation.typeTitle(`Group ${generateUUID()}`); + if (isMandatory !== undefined) { + await examExerciseGroupCreation.setMandatoryBox(isMandatory); + } + await examExerciseGroupCreation.clickSave(); +} diff --git a/src/test/playwright/e2e/exam/ExamCreationDeletion.spec.ts b/src/test/playwright/e2e/exam/ExamCreationDeletion.spec.ts index a17426823b20..fb0e69ef5c92 100644 --- a/src/test/playwright/e2e/exam/ExamCreationDeletion.spec.ts +++ b/src/test/playwright/e2e/exam/ExamCreationDeletion.spec.ts @@ -9,37 +9,10 @@ import { Exam } from 'app/entities/exam/exam.model'; /* * Common primitives */ -const examData = { - title: 'exam' + generateUUID(), - visibleDate: dayjs(), - startDate: dayjs().add(1, 'hour'), - endDate: dayjs().add(2, 'hour'), - numberOfExercisesInExam: 4, - examMaxPoints: 40, - startText: 'Exam start text', - endText: 'Exam end text', - confirmationStartText: 'Exam confirmation start text', - confirmationEndText: 'Exam confirmation end text', -}; - -const editedExamData = { - title: 'exam' + generateUUID(), - visibleDate: dayjs(), - startDate: dayjs().add(1, 'hour'), - endDate: dayjs().add(5, 'hour'), - numberOfExercisesInExam: 3, - examMaxPoints: 30, - startText: 'Edited exam start text', - endText: 'Edited exam end text', - confirmationStartText: 'Edited exam confirmation start text', - confirmationEndText: 'Edited exam confirmation end text', -}; - const dateFormat = 'MMM D, YYYY HH:mm'; test.describe('Exam creation/deletion', { tag: '@fast' }, () => { let course: Course; - let examId: number; test.beforeEach(async ({ login, courseManagementAPIRequests }) => { await login(admin); @@ -47,6 +20,19 @@ test.describe('Exam creation/deletion', { tag: '@fast' }, () => { }); test('Creates an exam', async ({ navigationBar, courseManagement, examManagement, examCreation }) => { + const examData = { + title: 'exam' + generateUUID(), + visibleDate: dayjs(), + startDate: dayjs().add(1, 'hour'), + endDate: dayjs().add(2, 'hour'), + numberOfExercisesInExam: 4, + examMaxPoints: 40, + startText: 'Exam start text', + endText: 'Exam end text', + confirmationStartText: 'Exam confirmation start text', + confirmationEndText: 'Exam confirmation end text', + }; + await navigationBar.openCourseManagement(); await courseManagement.openExamsOfCourse(course.id!); @@ -64,8 +50,6 @@ test.describe('Exam creation/deletion', { tag: '@fast' }, () => { await examCreation.setConfirmationEndText(examData.confirmationEndText); const response = await examCreation.submit(); - const exam: Exam = await response.json(); - examId = exam.id!; expect(response.status()).toBe(201); await expect(examManagement.getExamTitle()).toContainText(examData.title); @@ -82,44 +66,60 @@ test.describe('Exam creation/deletion', { tag: '@fast' }, () => { }); test.describe('Exam deletion', () => { + let exam: Exam; + test.beforeEach(async ({ examAPIRequests }) => { - examData.title = 'exam' + generateUUID(); const examConfig = { course, - title: examData.title, + title: 'exam' + generateUUID(), }; - const examResponse = await examAPIRequests.createExam(examConfig); - examId = examResponse.id!; + exam = await examAPIRequests.createExam(examConfig); }); test('Deletes an existing exam', async ({ navigationBar, courseManagement, examManagement, examDetails }) => { await navigationBar.openCourseManagement(); await courseManagement.openExamsOfCourse(course.id!); - await examManagement.openExam(examId); - await examDetails.deleteExam(examData.title); - await expect(examManagement.getExamSelector(examData.title)).not.toBeVisible(); + await examManagement.openExam(exam.id!); + await examDetails.deleteExam(exam.title!); + await expect(examManagement.getExamSelector(exam.title!)).not.toBeVisible(); }); }); test.describe('Edits an exam', () => { + let exam: Exam; + test.beforeEach(async ({ examAPIRequests }) => { - examData.title = 'exam' + generateUUID(); const examConfig = { course, - title: examData.title, + title: 'exam' + generateUUID(), visibleDate: dayjs(), - startDate: dayjs().add(1, 'hour'), }; - const examResponse = await examAPIRequests.createExam(examConfig); - examId = examResponse.id!; + exam = await examAPIRequests.createExam(examConfig); }); test('Edits an existing exam', async ({ navigationBar, courseManagement, examManagement, examCreation }) => { + const visibleDate = dayjs(); + const startDate = visibleDate.add(1, 'hour'); + const endDate = startDate.add(4, 'hour'); + + const editedExamData = { + title: 'exam' + generateUUID(), + visibleDate: visibleDate, + startDate: startDate, + endDate: endDate, + numberOfExercisesInExam: 3, + examMaxPoints: 30, + startText: 'Edited exam start text', + endText: 'Edited exam end text', + confirmationStartText: 'Edited exam confirmation start text', + confirmationEndText: 'Edited exam confirmation end text', + }; + await navigationBar.openCourseManagement(); await courseManagement.openExamsOfCourse(course.id!); - await examManagement.openExam(examId); + await examManagement.openExam(exam.id!); - await expect(examManagement.getExamTitle()).toContainText(examData.title); + await expect(examManagement.getExamTitle()).toContainText(exam.title!); await examManagement.clickEdit(); await examCreation.setTitle(editedExamData.title); @@ -136,19 +136,18 @@ test.describe('Exam creation/deletion', { tag: '@fast' }, () => { const response = await examCreation.update(); expect(response.status()).toBe(200); - const exam = await response.json(); - - examId = exam.id; - expect(exam.testExam).toBeFalsy(); - expect(trimDate(exam.visibleDate)).toBe(trimDate(dayjsToString(editedExamData.visibleDate))); - expect(trimDate(exam.startDate)).toBe(trimDate(dayjsToString(editedExamData.startDate))); - expect(trimDate(exam.endDate)).toBe(trimDate(dayjsToString(editedExamData.endDate))); - expect(exam.numberOfExercisesInExam).toBe(editedExamData.numberOfExercisesInExam); - expect(exam.examMaxPoints).toBe(editedExamData.examMaxPoints); - expect(exam.startText).toBe(editedExamData.startText); - expect(exam.endText).toBe(editedExamData.endText); - expect(exam.confirmationStartText).toBe(editedExamData.confirmationStartText); - expect(exam.confirmationEndText).toBe(editedExamData.confirmationEndText); + const editedExam = await response.json(); + + expect(editedExam.testExam).toBeFalsy(); + expect(trimDate(editedExam.visibleDate)).toBe(trimDate(dayjsToString(editedExamData.visibleDate))); + expect(trimDate(editedExam.startDate)).toBe(trimDate(dayjsToString(editedExamData.startDate))); + expect(trimDate(editedExam.endDate)).toBe(trimDate(dayjsToString(editedExamData.endDate))); + expect(editedExam.numberOfExercisesInExam).toBe(editedExamData.numberOfExercisesInExam); + expect(editedExam.examMaxPoints).toBe(editedExamData.examMaxPoints); + expect(editedExam.startText).toBe(editedExamData.startText); + expect(editedExam.endText).toBe(editedExamData.endText); + expect(editedExam.confirmationStartText).toBe(editedExamData.confirmationStartText); + expect(editedExam.confirmationEndText).toBe(editedExamData.confirmationEndText); await expect(examManagement.getExamTitle()).toContainText(editedExamData.title); await expect(examManagement.getExamVisibleDate()).toContainText(dayjs(editedExamData.visibleDate).format(dateFormat)); diff --git a/src/test/playwright/e2e/exam/ExamParticipation.spec.ts b/src/test/playwright/e2e/exam/ExamParticipation.spec.ts index f8a08642e70e..ca86e188d575 100644 --- a/src/test/playwright/e2e/exam/ExamParticipation.spec.ts +++ b/src/test/playwright/e2e/exam/ExamParticipation.spec.ts @@ -260,7 +260,7 @@ test.describe('Exam participation', () => { }); }); - test.describe('Exam announcements', { tag: '@slow' }, () => { + test.describe('Exam announcements', () => { let exam: Exam; const students = [studentOne, studentTwo]; let exercise: Exercise; @@ -278,42 +278,46 @@ test.describe('Exam participation', () => { await examAPIRequests.prepareExerciseStartForExam(exam); }); - test('Instructor sends an announcement message and all participants receive it', async ({ browser, login, navigationBar, courseManagement, examManagement }) => { - await login(instructor); - await navigationBar.openCourseManagement(); - await courseManagement.openExamsOfCourse(course.id!); - await examManagement.openExam(exam.id!); + test( + 'Instructor sends an announcement message and all participants receive it', + { tag: '@slow' }, + async ({ browser, login, navigationBar, courseManagement, examManagement }) => { + await login(instructor); + await navigationBar.openCourseManagement(); + await courseManagement.openExamsOfCourse(course.id!); + await examManagement.openExam(exam.id!); - const studentPages = []; + const studentPages = []; - for (const student of [studentOne, studentTwo]) { - const studentContext = await browser.newContext(); - const studentPage = await studentContext.newPage(); - studentPages.push(studentPage); + for (const student of [studentOne, studentTwo]) { + const studentContext = await browser.newContext(); + const studentPage = await studentContext.newPage(); + studentPages.push(studentPage); - await Commands.login(studentPage, student); - await studentPage.goto(`/courses/${course.id!}/exams/${exam.id!}`); - const examStartEnd = new ExamStartEndPage(studentPage); - await examStartEnd.startExam(false); - } + await Commands.login(studentPage, student); + await studentPage.goto(`/courses/${course.id!}/exams/${exam.id!}`); + const examStartEnd = new ExamStartEndPage(studentPage); + await examStartEnd.startExam(false); + } - const announcement = 'Important announcement!'; - await examManagement.openAnnouncementDialog(); - const announcementTypingTime = dayjs(); - await examManagement.typeAnnouncementMessage(announcement); - await examManagement.verifyAnnouncementContent(announcementTypingTime, announcement, instructor.username); - await examManagement.sendAnnouncement(); + const announcement = 'Important announcement!'; + await examManagement.openAnnouncementDialog(); + const announcementTypingTime = dayjs(); + await examManagement.typeAnnouncementMessage(announcement); + await examManagement.verifyAnnouncementContent(announcementTypingTime, announcement, instructor.username); + await examManagement.sendAnnouncement(); - for (const studentPage of studentPages) { - const modalDialog = new ModalDialogBox(studentPage); - await modalDialog.checkDialogTime(announcementTypingTime); - await modalDialog.checkDialogMessage(announcement); - await modalDialog.checkDialogAuthor(instructor.username); - await modalDialog.closeDialog(); - } - }); + for (const studentPage of studentPages) { + const modalDialog = new ModalDialogBox(studentPage); + await modalDialog.checkDialogTime(announcementTypingTime); + await modalDialog.checkDialogMessage(announcement); + await modalDialog.checkDialogAuthor(instructor.username); + await modalDialog.closeDialog(); + } + }, + ); - test('Instructor changes working time and all participants are informed', async ({ browser, login, navigationBar, courseManagement, examManagement }) => { + test('Instructor changes working time and all participants are informed', { tag: '@slow' }, async ({ browser, login, navigationBar, courseManagement, examManagement }) => { await login(instructor); await navigationBar.openCourseManagement(); await courseManagement.openExamsOfCourse(course.id!); @@ -354,7 +358,7 @@ test.describe('Exam participation', () => { test( 'Instructor changes problem statement and all participants are informed', { tag: '@fast' }, - async ({ browser, login, navigationBar, courseManagement, examManagement, examExerciseGroups, editExam, textExerciseCreation }) => { + async ({ browser, login, navigationBar, courseManagement, examManagement, examExerciseGroups, examDetails, textExerciseCreation }) => { await login(instructor); await navigationBar.openCourseManagement(); await courseManagement.openExamsOfCourse(course.id!); @@ -375,7 +379,7 @@ test.describe('Exam participation', () => { await examNavigation.openOrSaveExerciseByTitle(exercise.exerciseGroup!.title!); } - await editExam.openExerciseGroups(); + await examDetails.openExerciseGroups(); await examExerciseGroups.clickEditExercise(exercise.exerciseGroup!.id!, exercise.id!); const problemStatementText = textExerciseTemplate.problemStatement; diff --git a/src/test/playwright/support/fixtures.ts b/src/test/playwright/support/fixtures.ts index 00ea2c5bdef8..9d25ab252f19 100644 --- a/src/test/playwright/support/fixtures.ts +++ b/src/test/playwright/support/fixtures.ts @@ -68,7 +68,6 @@ import { QuizExerciseParticipationPage } from './pageobjects/exercises/quiz/Quiz import { ModalDialogBox } from './pageobjects/exam/ModalDialogBox'; import { ExamParticipationActions } from './pageobjects/exam/ExamParticipationActions'; import { AccountManagementAPIRequests } from './requests/AccountManagementAPIRequests'; -import { EditExamPage } from './pageobjects/exam/EditExamPage'; /* * Define custom types for fixtures @@ -98,7 +97,6 @@ export type ArtemisPageObjects = { courseCommunication: CourseCommunicationPage; lectureManagement: LectureManagementPage; lectureCreation: LectureCreationPage; - editExam: EditExamPage; examCreation: ExamCreationPage; examDetails: ExamDetailsPage; examExerciseGroupCreation: ExamExerciseGroupCreationPage; @@ -223,9 +221,6 @@ export const test = base.extend { await use(new LectureCreationPage(page)); }, - editExam: async ({ page }, use) => { - await use(new EditExamPage(page)); - }, examCreation: async ({ page }, use) => { await use(new ExamCreationPage(page)); }, diff --git a/src/test/playwright/support/pageobjects/exam/EditExamPage.ts b/src/test/playwright/support/pageobjects/exam/EditExamPage.ts deleted file mode 100644 index 69608a61522c..000000000000 --- a/src/test/playwright/support/pageobjects/exam/EditExamPage.ts +++ /dev/null @@ -1,13 +0,0 @@ -import { Page } from '@playwright/test'; - -export class EditExamPage { - private readonly page: Page; - - constructor(page: Page) { - this.page = page; - } - - async openExerciseGroups() { - await this.page.locator(`#exercises-button-groups-table`).click(); - } -} diff --git a/src/test/playwright/support/pageobjects/exam/ExamCreationPage.ts b/src/test/playwright/support/pageobjects/exam/ExamCreationPage.ts index e710822e6723..8fe6ad86c5d3 100644 --- a/src/test/playwright/support/pageobjects/exam/ExamCreationPage.ts +++ b/src/test/playwright/support/pageobjects/exam/ExamCreationPage.ts @@ -51,6 +51,27 @@ export class ExamCreationPage { await enterDate(this.page, '#endDate', date); } + /** + * @param date the date when the exam results will be published + */ + async setPublishResultsDate(date: dayjs.Dayjs) { + await enterDate(this.page, '#publishResultsDate', date); + } + + /** + * @param date the date when the exam student review starts + */ + async setStudentReviewStartDate(date: dayjs.Dayjs) { + await enterDate(this.page, '#examStudentReviewStart', date); + } + + /** + * @param date the date when the exam student review ends + */ + async setStudentReviewEndDate(date: dayjs.Dayjs) { + await enterDate(this.page, '#examStudentReviewEnd', date); + } + /** * @param time the exam working time */ diff --git a/src/test/playwright/support/pageobjects/exam/ExamDetailsPage.ts b/src/test/playwright/support/pageobjects/exam/ExamDetailsPage.ts index 29bc3b8aa8a0..33e4ef66329e 100644 --- a/src/test/playwright/support/pageobjects/exam/ExamDetailsPage.ts +++ b/src/test/playwright/support/pageobjects/exam/ExamDetailsPage.ts @@ -10,6 +10,56 @@ export class ExamDetailsPage { this.page = page; } + async openExerciseGroups() { + await this.page.locator(`#exercises-button-groups`).click(); + } + + async checkItemChecked(checklistItem: ExamChecklistItem) { + await expect( + this.getChecklistItemLocator(checklistItem).getByTestId('check-icon-checked'), + `Checklist item for \"${checklistItem}\" is not checked or not found`, + ).toBeVisible(); + } + + async checkItemUnchecked(checklistItem: ExamChecklistItem) { + await expect( + this.getChecklistItemLocator(checklistItem).getByTestId('check-icon-unchecked'), + `Checklist item for \"${checklistItem}\" is not unchecked or not found`, + ).toBeVisible(); + } + + private getChecklistItemLocator(checklistItem: ExamChecklistItem) { + return this.page.getByTestId(checklistItem); + } + + async clickStudentsToRegister() { + await this.page.getByTestId('students-button-register').click(); + } + + async clickStudentExamsToGenerate() { + await this.page.getByTestId('student-exams-button-generate').click(); + } + + async clickStudentExamsToPrepareStart() { + await this.page.getByTestId('student-exams-button-prepare-start').click(); + } + + async clickEditExamForPublishDate() { + await this.page.locator('#editButton_publish').click(); + } + + async clickEditExamForReviewDate() { + await this.page.locator('#editButton_review').click(); + } + + async clickEvaluateQuizExercises() { + await this.page.locator('#evaluateQuizExercisesButton').click(); + } + + async clickAssessUnsubmittedParticipations() { + await this.page.locator('#assessUnsubmittedExamModelingAndTextParticipationsButton').click(); + } + /** * Deletes this exam. * @param examTitle the exam title to confirm the deletion @@ -23,3 +73,20 @@ export class ExamDetailsPage { await deleteButton.click(); } } + +export enum ExamChecklistItem { + LEAST_ONE_EXERCISE_GROUP = 'check-least-one-exercise-group', + NUMBER_OF_EXERCISE_GROUPS = 'check-number-of-exercise-groups', + EACH_EXERCISE_GROUP_HAS_EXERCISES = 'check-each-exercise-group-has-exercises', + POINTS_IN_EXERCISE_GROUPS_EQUAL = 'check-points-in-exercise-groups-equal', + TOTAL_POINTS_POSSIBLE = 'check-total-points-possible', + LEAST_ONE_STUDENT = 'check-least-one-student', + ALL_EXAMS_GENERATED = 'check-all-exams-generated', + ALL_EXERCISES_PREPARED = 'check-all-exercises-prepared', + PUBLISHING_DATE_SET = 'check-publishing-date-set', + START_DATE_REVIEW_SET = 'check-start-date-review-set', + END_DATE_REVIEW_SET = 'check-end-date-review-set', + UNFINISHED_ASSESSMENTS = 'check-unfinished-assessments', + UNASSESSED_QUIZZES = 'check-unassessed-quizzes', + UNSUBMITTED_EXERCISES = 'check-unsubmitted-exercises', +} diff --git a/src/test/playwright/support/pageobjects/exam/ExamExerciseGroupCreationPage.ts b/src/test/playwright/support/pageobjects/exam/ExamExerciseGroupCreationPage.ts index 435202bf7971..8b9e504e9380 100644 --- a/src/test/playwright/support/pageobjects/exam/ExamExerciseGroupCreationPage.ts +++ b/src/test/playwright/support/pageobjects/exam/ExamExerciseGroupCreationPage.ts @@ -27,8 +27,20 @@ export class ExamExerciseGroupCreationPage { await titleField.fill(title); } + async setMandatoryBox(checked: boolean) { + if (checked) { + await this.getMandatoryBoxLocator().check(); + } else { + await this.getMandatoryBoxLocator().uncheck(); + } + } + async isMandatoryBoxShouldBeChecked() { - await this.page.locator('#isMandatory').isChecked(); + await this.getMandatoryBoxLocator().isChecked(); + } + + private getMandatoryBoxLocator() { + return this.page.locator('#isMandatory'); } async clickSave(): Promise { @@ -44,8 +56,14 @@ export class ExamExerciseGroupCreationPage { await responsePromise; } - async addGroupWithExercise(exam: Exam, exerciseType: ExerciseType, additionalData: AdditionalData = {}): Promise { - const response = await this.handleAddGroupWithExercise(exam, 'Exercise ' + generateUUID(), exerciseType, additionalData); + async addGroupWithExercise( + exam: Exam, + exerciseType: ExerciseType, + additionalData: AdditionalData = {}, + isMandatory?: boolean, + exerciseTemplate?: any, + ): Promise { + const response = await this.handleAddGroupWithExercise(exam, 'Exercise ' + generateUUID(), exerciseType, additionalData, isMandatory, exerciseTemplate); let exercise = { ...response!, additionalData }; if (exerciseType == ExerciseType.QUIZ) { const quiz = response as QuizExercise; @@ -61,11 +79,18 @@ export class ExamExerciseGroupCreationPage { return exercise; } - async handleAddGroupWithExercise(exam: Exam, title: string, exerciseType: ExerciseType, additionalData: AdditionalData): Promise { - const exerciseGroup = await this.examAPIRequests.addExerciseGroupForExam(exam); + async handleAddGroupWithExercise( + exam: Exam, + title: string, + exerciseType: ExerciseType, + additionalData: AdditionalData, + isMandatory?: boolean, + exerciseTemplate?: any, + ): Promise { + const exerciseGroup = await this.examAPIRequests.addExerciseGroupForExam(exam, 'Group ' + generateUUID(), isMandatory); switch (exerciseType) { case ExerciseType.TEXT: - return await this.exerciseAPIRequests.createTextExercise({ exerciseGroup }, title); + return await this.exerciseAPIRequests.createTextExercise({ exerciseGroup }, title, exerciseTemplate); case ExerciseType.MODELING: return await this.exerciseAPIRequests.createModelingExercise({ exerciseGroup }, title); case ExerciseType.QUIZ: diff --git a/src/test/playwright/support/pageobjects/exam/StudentExamManagementPage.ts b/src/test/playwright/support/pageobjects/exam/StudentExamManagementPage.ts index 593fc5451872..9dee95c1e573 100644 --- a/src/test/playwright/support/pageobjects/exam/StudentExamManagementPage.ts +++ b/src/test/playwright/support/pageobjects/exam/StudentExamManagementPage.ts @@ -21,6 +21,10 @@ export class StudentExamManagementPage { return await responsePromise; } + async clickPrepareExerciseStart() { + await this.page.click('#startExercisesButton'); + } + getGenerateMissingStudentExamsButton() { return this.page.locator('#generateMissingStudentExamsButton'); } diff --git a/src/test/playwright/support/requests/ExamAPIRequests.ts b/src/test/playwright/support/requests/ExamAPIRequests.ts index 34ffccb9a8fe..899e1f396e93 100644 --- a/src/test/playwright/support/requests/ExamAPIRequests.ts +++ b/src/test/playwright/support/requests/ExamAPIRequests.ts @@ -113,12 +113,18 @@ export class ExamAPIRequests { /** * Register the student for the exam - * @param exam the exam object */ async registerStudentForExam(exam: Exam, student: UserCredentials) { await this.page.request.post(`${COURSE_BASE}/${exam.course!.id}/exams/${exam.id}/students/${student.username}`); } + /** + * Register all course students for the exam + */ + async registerAllCourseStudentsForExam(exam: Exam) { + await this.page.request.post(`${COURSE_BASE}/${exam.course!.id}/exams/${exam.id}/register-course-students`); + } + /** * Creates an exam test run with the provided settings. * @param exam the exam object @@ -208,4 +214,17 @@ export class ExamAPIRequests { const response = await this.page.request.get(`${COURSE_BASE}/${exam.course!.id}/exams/${exam.id}/student-exams/${studentExam.id}/grade-summary`); return await response.json(); } + + /** + * Determines the time left until the exam ends and finishes the exam by subtracting it from the working time. + */ + async finishExam(exam: Exam) { + const examEndDate = dayjs(exam.endDate! as dayjs.Dayjs); + // Determine the time left until the exam ends and add extra minute + // to make sure the exam is finished after subtracting it from the working time + const examTimeLeftInSeconds = examEndDate.diff(dayjs(), 'seconds') + 60; + if (examTimeLeftInSeconds > 0) { + await this.page.request.patch(`${COURSE_BASE}/${exam.course!.id}/exams/${exam.id}/working-time`, { data: -examTimeLeftInSeconds }); + } + } } diff --git a/src/test/playwright/support/requests/ExerciseAPIRequests.ts b/src/test/playwright/support/requests/ExerciseAPIRequests.ts index e9c836fb3e1d..273f538abf07 100644 --- a/src/test/playwright/support/requests/ExerciseAPIRequests.ts +++ b/src/test/playwright/support/requests/ExerciseAPIRequests.ts @@ -213,10 +213,16 @@ export class ExerciseAPIRequests { * * @param body - An object containing either the course or exercise group the exercise will be added to. * @param title - The title for the text exercise (optional, default: auto-generated). + * @param exerciseTemplate - The template for the text exercise + * (optional, default: textExerciseTemplate - default template). */ - async createTextExercise(body: { course: Course } | { exerciseGroup: ExerciseGroup }, title = 'Text ' + generateUUID()): Promise { + async createTextExercise( + body: { course: Course } | { exerciseGroup: ExerciseGroup }, + title = 'Text ' + generateUUID(), + exerciseTemplate: any = textExerciseTemplate, + ): Promise { const template = { - ...textExerciseTemplate, + ...exerciseTemplate, title, channelName: 'exercise-' + titleLowercase(title), }; @@ -268,6 +274,7 @@ export class ExerciseAPIRequests { * * @param exerciseId - The ID of the text exercise for which the submission is made. * @param text - The text content of the submission. + * @param createNewSubmission - Whether to create a new submission or update an existing one (optional, default: true). */ async makeTextExerciseSubmission(exerciseId: number, text: string, createNewSubmission = true) { const url = `${EXERCISE_BASE}/${exerciseId}/text-submissions`; diff --git a/src/test/playwright/support/utils.ts b/src/test/playwright/support/utils.ts index bc464f447717..893e0d4d5adf 100644 --- a/src/test/playwright/support/utils.ts +++ b/src/test/playwright/support/utils.ts @@ -1,10 +1,30 @@ import dayjs from 'dayjs'; import utc from 'dayjs/plugin/utc'; import { v4 as uuidv4 } from 'uuid'; -import { TIME_FORMAT } from './constants'; +import { Exercise, ExerciseType, ProgrammingExerciseAssessmentType, TIME_FORMAT } from './constants'; import * as fs from 'fs'; import { dirname } from 'path'; import { Browser, Locator, Page, expect } from '@playwright/test'; +import { Course } from 'app/entities/course.model'; +import { Exam } from 'app/entities/exam/exam.model'; +import { ExamAPIRequests } from './requests/ExamAPIRequests'; +import { ExerciseAPIRequests } from './requests/ExerciseAPIRequests'; +import { ExamExerciseGroupCreationPage } from './pageobjects/exam/ExamExerciseGroupCreationPage'; +import { CoursesPage } from './pageobjects/course/CoursesPage'; +import { CourseOverviewPage } from './pageobjects/course/CourseOverviewPage'; +import { ModelingEditor } from './pageobjects/exercises/modeling/ModelingEditor'; +import { OnlineEditorPage } from './pageobjects/exercises/programming/OnlineEditorPage'; +import { MultipleChoiceQuiz } from './pageobjects/exercises/quiz/MultipleChoiceQuiz'; +import { TextEditorPage } from './pageobjects/exercises/text/TextEditorPage'; +import { ExamNavigationBar } from './pageobjects/exam/ExamNavigationBar'; +import { ExamStartEndPage } from './pageobjects/exam/ExamStartEndPage'; +import { ExamParticipationPage } from './pageobjects/exam/ExamParticipationPage'; +import { Commands } from './commands'; +import { admin, studentOne } from './users'; +import javaPartiallySuccessful from '../fixtures/exercise/programming/java/partially_successful/submission.json'; +import { ExamManagementPage } from './pageobjects/exam/ExamManagementPage'; +import { CourseAssessmentDashboardPage } from './pageobjects/assessment/CourseAssessmentDashboardPage'; +import { ExerciseAssessmentDashboardPage } from './pageobjects/assessment/ExerciseAssessmentDashboardPage'; // Add utc plugin to use the utc timezone dayjs.extend(utc); @@ -170,3 +190,108 @@ export async function drag(page: Page, draggable: Locator, droppable: Locator) { }); await page.mouse.up(); } + +/* + * Exam utility functions + */ + +export async function prepareExam(course: Course, end: dayjs.Dayjs, exerciseType: ExerciseType, page: Page, numberOfCorrectionRounds: number = 1): Promise { + const examAPIRequests = new ExamAPIRequests(page); + const exerciseAPIRequests = new ExerciseAPIRequests(page); + const examExerciseGroupCreation = new ExamExerciseGroupCreationPage(page, examAPIRequests, exerciseAPIRequests); + const courseList = new CoursesPage(page); + const courseOverview = new CourseOverviewPage(page); + const modelingExerciseEditor = new ModelingEditor(page); + const programmingExerciseEditor = new OnlineEditorPage(page); + const quizExerciseMultipleChoice = new MultipleChoiceQuiz(page); + const textExerciseEditor = new TextEditorPage(page); + const examNavigation = new ExamNavigationBar(page); + const examStartEnd = new ExamStartEndPage(page); + const examParticipation = new ExamParticipationPage( + courseList, + courseOverview, + examNavigation, + examStartEnd, + modelingExerciseEditor, + programmingExerciseEditor, + quizExerciseMultipleChoice, + textExerciseEditor, + page, + ); + + await Commands.login(page, admin); + const resultDate = end.add(1, 'second'); + const examConfig = { + course, + startDate: dayjs(), + endDate: end, + numberOfCorrectionRoundsInExam: numberOfCorrectionRounds, + examStudentReviewStart: resultDate, + examStudentReviewEnd: resultDate.add(1, 'minute'), + publishResultsDate: resultDate, + gracePeriod: 10, + }; + const exam = await examAPIRequests.createExam(examConfig); + let additionalData = {}; + switch (exerciseType) { + case ExerciseType.PROGRAMMING: + additionalData = { + submission: javaPartiallySuccessful, + progExerciseAssessmentType: ProgrammingExerciseAssessmentType.SEMI_AUTOMATIC, + }; + break; + case ExerciseType.TEXT: + additionalData = { textFixture: 'loremIpsum-short.txt' }; + break; + case ExerciseType.QUIZ: + additionalData = { quizExerciseID: 0 }; + break; + } + + const exercise = await examExerciseGroupCreation.addGroupWithExercise(exam, exerciseType, additionalData); + await examAPIRequests.registerStudentForExam(exam, studentOne); + await examAPIRequests.generateMissingIndividualExams(exam); + await examAPIRequests.prepareExerciseStartForExam(exam); + exercise.additionalData = additionalData; + await makeExamSubmission(course, exam, exercise, page, examParticipation, examNavigation, examStartEnd); + return exam; +} + +export async function makeExamSubmission( + course: Course, + exam: Exam, + exercise: Exercise, + page: Page, + examParticipation: ExamParticipationPage, + examNavigation: ExamNavigationBar, + examStartEnd: ExamStartEndPage, +) { + await examParticipation.startParticipation(studentOne, course, exam); + await examNavigation.openOrSaveExerciseByTitle(exercise.exerciseGroup!.title!); + await examParticipation.makeSubmission(exercise.id!, exercise.type!, exercise.additionalData); + await page.waitForTimeout(2000); + await examNavigation.handInEarly(); + await examStartEnd.finishExam(); +} + +export async function startAssessing( + courseID: number, + examID: number, + timeout: number, + examManagement: ExamManagementPage, + courseAssessment: CourseAssessmentDashboardPage, + exerciseAssessment: ExerciseAssessmentDashboardPage, + toggleSecondRound: boolean = false, + isFirstTimeAssessing: boolean = true, +) { + await examManagement.openAssessmentDashboard(courseID, examID, timeout); + await courseAssessment.clickExerciseDashboardButton(); + if (toggleSecondRound) { + await exerciseAssessment.toggleSecondCorrectionRound(); + } + if (isFirstTimeAssessing) { + await exerciseAssessment.clickHaveReadInstructionsButton(); + } + await exerciseAssessment.clickStartNewAssessment(); + exerciseAssessment.getLockedMessage(); +} From afc50e8fc84b27220477597cba5b5c38663e576a Mon Sep 17 00:00:00 2001 From: Ole Vester <73833780+ole-ve@users.noreply.github.com> Date: Sun, 5 Jan 2025 21:19:45 +0100 Subject: [PATCH 28/34] General: Fix an issue with course deletion summary entries (#9856) --- .../repository/PostRepository.java | 8 ++++++- .../core/dto/CourseDeletionSummaryDTO.java | 3 ++- .../artemis/core/service/CourseService.java | 18 +++++++++++++-- .../exam/repository/ExamRepository.java | 2 ++ .../exercise/dto/ExerciseTypeCountDTO.java | 13 +++++++++++ .../repository/ExerciseRepository.java | 18 +++++++++++++++ .../exercise/service/ExerciseService.java | 16 +++++++++++++ .../lecture/repository/LectureRepository.java | 2 ++ .../course-management-tab-bar.component.ts | 23 ++++++------------- .../entities/course-deletion-summary.model.ts | 7 ++++++ .../util/ConversationUtilService.java | 1 - 11 files changed, 90 insertions(+), 21 deletions(-) create mode 100644 src/main/java/de/tum/cit/aet/artemis/exercise/dto/ExerciseTypeCountDTO.java diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/repository/PostRepository.java b/src/main/java/de/tum/cit/aet/artemis/communication/repository/PostRepository.java index 1ea95f1d6657..7427a9fc1ce6 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/repository/PostRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/repository/PostRepository.java @@ -48,7 +48,13 @@ default Post findPostOrMessagePostByIdElseThrow(Long postId) throws EntityNotFou List findAllByConversationId(Long conversationId); - List findAllByCourseId(Long courseId); + @Query(""" + SELECT p + FROM Post p + LEFT JOIN p.conversation conversation + WHERE conversation.course.id = :courseId + """) + List findAllByCourseId(@Param("courseId") Long courseId); List findByIdIn(List idList); } diff --git a/src/main/java/de/tum/cit/aet/artemis/core/dto/CourseDeletionSummaryDTO.java b/src/main/java/de/tum/cit/aet/artemis/core/dto/CourseDeletionSummaryDTO.java index 2f3b8d51596c..8ee1f5982268 100644 --- a/src/main/java/de/tum/cit/aet/artemis/core/dto/CourseDeletionSummaryDTO.java +++ b/src/main/java/de/tum/cit/aet/artemis/core/dto/CourseDeletionSummaryDTO.java @@ -3,5 +3,6 @@ import com.fasterxml.jackson.annotation.JsonInclude; @JsonInclude(JsonInclude.Include.NON_EMPTY) -public record CourseDeletionSummaryDTO(long numberOfBuilds, long numberOfCommunicationPosts, long numberOfAnswerPosts) { +public record CourseDeletionSummaryDTO(long numberOfBuilds, long numberOfCommunicationPosts, long numberOfAnswerPosts, long numberProgrammingExercises, long numberTextExercises, + long numberFileUploadExercises, long numberModelingExercises, long numberQuizExercises, long numberExams, long numberLectures) { } diff --git a/src/main/java/de/tum/cit/aet/artemis/core/service/CourseService.java b/src/main/java/de/tum/cit/aet/artemis/core/service/CourseService.java index 232e86070473..6122b92810c2 100644 --- a/src/main/java/de/tum/cit/aet/artemis/core/service/CourseService.java +++ b/src/main/java/de/tum/cit/aet/artemis/core/service/CourseService.java @@ -98,6 +98,7 @@ import de.tum.cit.aet.artemis.exam.repository.ExerciseGroupRepository; import de.tum.cit.aet.artemis.exam.service.ExamDeletionService; import de.tum.cit.aet.artemis.exercise.domain.Exercise; +import de.tum.cit.aet.artemis.exercise.domain.ExerciseType; import de.tum.cit.aet.artemis.exercise.domain.IncludedInOverallScore; import de.tum.cit.aet.artemis.exercise.repository.ExerciseRepository; import de.tum.cit.aet.artemis.exercise.repository.StudentParticipationRepository; @@ -475,13 +476,26 @@ public Set findAllOnlineCoursesForPlatformForUser(String registrationId, * @return the course deletion summary */ public CourseDeletionSummaryDTO getDeletionSummary(Course course) { + Long courseId = course.getId(); + List programmingExerciseIds = course.getExercises().stream().map(Exercise::getId).toList(); long numberOfBuilds = buildJobRepository.countBuildJobsByExerciseIds(programmingExerciseIds); - List posts = postRepository.findAllByCourseId(course.getId()); + List posts = postRepository.findAllByCourseId(courseId); long numberOfCommunicationPosts = posts.size(); long numberOfAnswerPosts = answerPostRepository.countAnswerPostsByPostIdIn(posts.stream().map(Post::getId).toList()); - return new CourseDeletionSummaryDTO(numberOfBuilds, numberOfCommunicationPosts, numberOfAnswerPosts); + long numberLectures = lectureRepository.countByCourse_Id(courseId); + long numberExams = examRepository.countByCourse_Id(courseId); + + Map countByExerciseType = exerciseService.countByCourseIdGroupByType(courseId); + long numberProgrammingExercises = countByExerciseType.get(ExerciseType.PROGRAMMING); + long numberTextExercises = countByExerciseType.get(ExerciseType.TEXT); + long numberQuizExercises = countByExerciseType.get(ExerciseType.QUIZ); + long numberFileUploadExercises = countByExerciseType.get(ExerciseType.FILE_UPLOAD); + long numberModelingExercises = countByExerciseType.get(ExerciseType.MODELING); + + return new CourseDeletionSummaryDTO(numberOfBuilds, numberOfCommunicationPosts, numberOfAnswerPosts, numberProgrammingExercises, numberTextExercises, + numberFileUploadExercises, numberModelingExercises, numberQuizExercises, numberExams, numberLectures); } /** diff --git a/src/main/java/de/tum/cit/aet/artemis/exam/repository/ExamRepository.java b/src/main/java/de/tum/cit/aet/artemis/exam/repository/ExamRepository.java index a884ed754e4a..48f17f7ae26d 100644 --- a/src/main/java/de/tum/cit/aet/artemis/exam/repository/ExamRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/exam/repository/ExamRepository.java @@ -385,6 +385,8 @@ SELECT COUNT(studentExam) """) long countGeneratedStudentExamsByExamWithoutTestRuns(@Param("examId") long examId); + long countByCourse_Id(Long courseId); + /** * Returns the title of the exam with the given id. * diff --git a/src/main/java/de/tum/cit/aet/artemis/exercise/dto/ExerciseTypeCountDTO.java b/src/main/java/de/tum/cit/aet/artemis/exercise/dto/ExerciseTypeCountDTO.java new file mode 100644 index 000000000000..094ec79cd85f --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/exercise/dto/ExerciseTypeCountDTO.java @@ -0,0 +1,13 @@ +package de.tum.cit.aet.artemis.exercise.dto; + +import java.util.Objects; + +import de.tum.cit.aet.artemis.exercise.domain.Exercise; +import de.tum.cit.aet.artemis.exercise.domain.ExerciseType; + +public record ExerciseTypeCountDTO(ExerciseType exerciseType, long count) { + + public ExerciseTypeCountDTO(Class exerciseType, Long count) { + this(ExerciseType.getExerciseTypeFromClass(exerciseType.asSubclass(Exercise.class)), Objects.requireNonNullElse(count, 0L)); + } +} diff --git a/src/main/java/de/tum/cit/aet/artemis/exercise/repository/ExerciseRepository.java b/src/main/java/de/tum/cit/aet/artemis/exercise/repository/ExerciseRepository.java index 60b85bc14f79..dbe00ff1083c 100644 --- a/src/main/java/de/tum/cit/aet/artemis/exercise/repository/ExerciseRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/exercise/repository/ExerciseRepository.java @@ -25,6 +25,7 @@ import de.tum.cit.aet.artemis.core.repository.base.ArtemisJpaRepository; import de.tum.cit.aet.artemis.exam.web.ExamResource; import de.tum.cit.aet.artemis.exercise.domain.Exercise; +import de.tum.cit.aet.artemis.exercise.dto.ExerciseTypeCountDTO; import de.tum.cit.aet.artemis.exercise.dto.ExerciseTypeMetricsEntry; /** @@ -627,4 +628,21 @@ SELECT count(e) > 0 AND e.exerciseGroup IS NOT NULL """) boolean isExamExercise(@Param("exerciseId") long exerciseId); + + /** + * Returns a mapping from exercise type to count for a given course id. Note that there are way fewer courses + * than exercise, so loading the course and joining the course is way faster than vice versa. + * + * @param courseId the courseId to get the exerciseType->count mapping for + * @return a list of mappings from exercise type to count + */ + @Query(""" + SELECT new de.tum.cit.aet.artemis.exercise.dto.ExerciseTypeCountDTO(TYPE(e), COUNT(e)) + FROM Course c + JOIN c.exercises e + WHERE c.id = :courseId + AND TYPE(e) IN (ModelingExercise, TextExercise, ProgrammingExercise, QuizExercise, FileUploadExercise) + GROUP BY TYPE(e) + """) + List countByCourseIdGroupedByType(@Param("courseId") long courseId); } diff --git a/src/main/java/de/tum/cit/aet/artemis/exercise/service/ExerciseService.java b/src/main/java/de/tum/cit/aet/artemis/exercise/service/ExerciseService.java index 73fce3b0f9f1..bcaa1fcbac35 100644 --- a/src/main/java/de/tum/cit/aet/artemis/exercise/service/ExerciseService.java +++ b/src/main/java/de/tum/cit/aet/artemis/exercise/service/ExerciseService.java @@ -6,6 +6,7 @@ import static java.time.ZonedDateTime.now; import java.util.ArrayList; +import java.util.Arrays; import java.util.Collection; import java.util.Comparator; import java.util.HashMap; @@ -63,9 +64,11 @@ import de.tum.cit.aet.artemis.exam.service.ExamLiveEventsService; import de.tum.cit.aet.artemis.exercise.domain.Exercise; import de.tum.cit.aet.artemis.exercise.domain.ExerciseMode; +import de.tum.cit.aet.artemis.exercise.domain.ExerciseType; import de.tum.cit.aet.artemis.exercise.domain.Submission; import de.tum.cit.aet.artemis.exercise.domain.Team; import de.tum.cit.aet.artemis.exercise.domain.participation.StudentParticipation; +import de.tum.cit.aet.artemis.exercise.dto.ExerciseTypeCountDTO; import de.tum.cit.aet.artemis.exercise.repository.ExerciseRepository; import de.tum.cit.aet.artemis.exercise.repository.StudentParticipationRepository; import de.tum.cit.aet.artemis.exercise.repository.SubmissionRepository; @@ -829,4 +832,17 @@ public T saveWithCompetencyLinks(T exercise, Function public void reconnectCompetencyExerciseLinks(Exercise exercise) { exercise.getCompetencyLinks().forEach(link -> link.setExercise(exercise)); } + + /** + * Returns a map from exercise type to count of exercise given a course id. + * + * @param courseId the course id + * @return the mapping from exercise type to course type. If a course has no exercises for a specific type, the map contains an entry for that type with value 0. + */ + public Map countByCourseIdGroupByType(Long courseId) { + Map exerciseTypeCountMap = exerciseRepository.countByCourseIdGroupedByType(courseId).stream() + .collect(Collectors.toMap(ExerciseTypeCountDTO::exerciseType, ExerciseTypeCountDTO::count)); + + return Arrays.stream(ExerciseType.values()).collect(Collectors.toMap(type -> type, type -> exerciseTypeCountMap.getOrDefault(type, 0L))); + } } diff --git a/src/main/java/de/tum/cit/aet/artemis/lecture/repository/LectureRepository.java b/src/main/java/de/tum/cit/aet/artemis/lecture/repository/LectureRepository.java index 33ce07e7b6fb..e441c1633b23 100644 --- a/src/main/java/de/tum/cit/aet/artemis/lecture/repository/LectureRepository.java +++ b/src/main/java/de/tum/cit/aet/artemis/lecture/repository/LectureRepository.java @@ -216,4 +216,6 @@ default Lecture findByIdWithLectureUnitsAndSlidesAndAttachmentsElseThrow(long le GROUP BY l.course.id """) Set countVisibleLectures(@Param("courseIds") Set courseIds, @Param("now") ZonedDateTime now); + + long countByCourse_Id(Long courseId); } diff --git a/src/main/webapp/app/course/manage/course-management-tab-bar/course-management-tab-bar.component.ts b/src/main/webapp/app/course/manage/course-management-tab-bar/course-management-tab-bar.component.ts index e4b3865d3dbc..8e15c55fbd28 100644 --- a/src/main/webapp/app/course/manage/course-management-tab-bar/course-management-tab-bar.component.ts +++ b/src/main/webapp/app/course/manage/course-management-tab-bar/course-management-tab-bar.component.ts @@ -204,12 +204,6 @@ export class CourseManagementTabBarComponent implements OnInit, OnDestroy, After } private getExistingSummaryEntries(): EntitySummary { - const numberRepositories = - this.course?.exercises - ?.filter((exercise) => exercise.type === 'programming') - .map((exercise) => exercise?.numberOfParticipations ?? 0) - .reduce((repositorySum, numberOfParticipationsForRepository) => repositorySum + numberOfParticipationsForRepository, 0) ?? 0; - const numberOfExercisesPerType = new Map(); this.course?.exercises?.forEach((exercise) => { if (exercise.type === undefined) { @@ -219,8 +213,6 @@ export class CourseManagementTabBarComponent implements OnInit, OnDestroy, After numberOfExercisesPerType.set(exercise.type, oldValue + 1); }); - const numberExams = this.course?.numberOfExams ?? 0; - const numberLectures = this.course?.lectures?.length ?? 0; const numberStudents = this.course?.numberOfStudents ?? 0; const numberTutors = this.course?.numberOfTeachingAssistants ?? 0; const numberEditors = this.course?.numberOfEditors ?? 0; @@ -228,14 +220,6 @@ export class CourseManagementTabBarComponent implements OnInit, OnDestroy, After const isTestCourse = this.course?.testCourse; return { - 'artemisApp.course.delete.summary.numberRepositories': numberRepositories, - 'artemisApp.course.delete.summary.numberProgrammingExercises': numberOfExercisesPerType.get(ExerciseType.PROGRAMMING) ?? 0, - 'artemisApp.course.delete.summary.numberModelingExercises': numberOfExercisesPerType.get(ExerciseType.MODELING) ?? 0, - 'artemisApp.course.delete.summary.numberTextExercises': numberOfExercisesPerType.get(ExerciseType.TEXT) ?? 0, - 'artemisApp.course.delete.summary.numberFileUploadExercises': numberOfExercisesPerType.get(ExerciseType.FILE_UPLOAD) ?? 0, - 'artemisApp.course.delete.summary.numberQuizExercises': numberOfExercisesPerType.get(ExerciseType.QUIZ) ?? 0, - 'artemisApp.course.delete.summary.numberExams': numberExams, - 'artemisApp.course.delete.summary.numberLectures': numberLectures, 'artemisApp.course.delete.summary.numberStudents': numberStudents, 'artemisApp.course.delete.summary.numberTutors': numberTutors, 'artemisApp.course.delete.summary.numberEditors': numberEditors, @@ -259,6 +243,13 @@ export class CourseManagementTabBarComponent implements OnInit, OnDestroy, After return { ...this.getExistingSummaryEntries(), + 'artemisApp.course.delete.summary.numberExams': summary.numberExams, + 'artemisApp.course.delete.summary.numberLectures': summary.numberLectures, + 'artemisApp.course.delete.summary.numberProgrammingExercises': summary.numberProgrammingExercises, + 'artemisApp.course.delete.summary.numberTextExercises': summary.numberTextExercises, + 'artemisApp.course.delete.summary.numberFileUploadExercises': summary.numberFileUploadExercises, + 'artemisApp.course.delete.summary.numberQuizExercises': summary.numberQuizExercises, + 'artemisApp.course.delete.summary.numberModelingExercises': summary.numberModelingExercises, 'artemisApp.course.delete.summary.numberBuilds': summary.numberOfBuilds, 'artemisApp.course.delete.summary.numberCommunicationPosts': summary.numberOfCommunicationPosts, 'artemisApp.course.delete.summary.numberAnswerPosts': summary.numberOfAnswerPosts, diff --git a/src/main/webapp/app/entities/course-deletion-summary.model.ts b/src/main/webapp/app/entities/course-deletion-summary.model.ts index 606a9d440d49..f36eadc241d4 100644 --- a/src/main/webapp/app/entities/course-deletion-summary.model.ts +++ b/src/main/webapp/app/entities/course-deletion-summary.model.ts @@ -2,4 +2,11 @@ export interface CourseDeletionSummaryDTO { numberOfBuilds: number; numberOfCommunicationPosts: number; numberOfAnswerPosts: number; + numberProgrammingExercises: number; + numberTextExercises: number; + numberFileUploadExercises: number; + numberQuizExercises: number; + numberModelingExercises: number; + numberExams: number; + numberLectures: number; } diff --git a/src/test/java/de/tum/cit/aet/artemis/communication/util/ConversationUtilService.java b/src/test/java/de/tum/cit/aet/artemis/communication/util/ConversationUtilService.java index 1a1855651224..1d581ffc7c66 100644 --- a/src/test/java/de/tum/cit/aet/artemis/communication/util/ConversationUtilService.java +++ b/src/test/java/de/tum/cit/aet/artemis/communication/util/ConversationUtilService.java @@ -660,7 +660,6 @@ private Post createMessageWithReactionForUser(String login, String messageText, message.setAuthor(userUtilService.getUserByLogin(login)); message.setContent(messageText); message.setCreationDate(ZonedDateTime.now()); - message.setCourse(conversation.getCourse()); conversation.setCreator(message.getAuthor()); addReactionForUserToPost(login, message); conversationRepository.save(conversation); From 95257844c2a203eff16bee23d04dfed29db0938a Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Sun, 5 Jan 2025 21:46:04 +0100 Subject: [PATCH 29/34] Development: Fix client compile error after merge --- .../programming/participate/programming-repository.module.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/webapp/app/exercises/programming/participate/programming-repository.module.ts b/src/main/webapp/app/exercises/programming/participate/programming-repository.module.ts index a765439ae8c9..3728c10c582c 100644 --- a/src/main/webapp/app/exercises/programming/participate/programming-repository.module.ts +++ b/src/main/webapp/app/exercises/programming/participate/programming-repository.module.ts @@ -18,7 +18,7 @@ import { ArtemisProgrammingExerciseModule } from 'app/exercises/programming/shar import { CommitHistoryComponent } from 'app/localvc/commit-history/commit-history.component'; import { CommitDetailsViewComponent } from 'app/localvc/commit-details-view/commit-details-view.component'; import { ArtemisProgrammingExerciseActionsModule } from 'app/exercises/programming/shared/actions/programming-exercise-actions.module'; -import { GitDiffReportComponent } from 'app/exercises/programming/hestia/git-diff-report/git-diff-report.component'; +import { GitDiffReportComponent } from 'app/exercises/programming/git-diff-report/git-diff-report.component'; import { ComplaintsForTutorComponent } from 'app/complaints/complaints-for-tutor/complaints-for-tutor.component'; @NgModule({ From 271044726aa4fe012b785ce7896040c5ab4bc7f8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maximilian=20S=C3=B6lch?= Date: Sun, 5 Jan 2025 22:10:29 +0100 Subject: [PATCH 30/34] Development: Migrate client code for the grading system (#10055) --- jest.config.js | 8 ++--- .../app/exam/manage/exam-management.module.ts | 8 +++-- ...ing-exercise-instructions-render.module.ts | 3 +- .../base-grading-system.component.ts | 16 +++++----- .../grading-system/bonus/bonus.component.ts | 25 +++++++++------- .../app/grading-system/bonus/bonus.service.ts | 10 +++---- .../detailed-grading-system.component.ts | 8 +++++ .../grading-key-overview.component.ts | 20 +++++++------ .../grading-key-overview.module.ts | 3 +- .../grading-key-table.component.ts | 21 ++++++++----- .../grading-system-info-modal.component.ts | 9 ++++-- .../grading-system-presentations.component.ts | 6 ++++ .../grading-system.component.ts | 30 ++++++++++++++++--- .../grading-system/grading-system.module.ts | 13 ++++++-- .../grading-system/grading-system.service.ts | 6 ++-- .../interval-grading-system.component.ts | 21 +++++++++++++ .../shared/pipes/grade-step-bounds.pipe.ts | 1 + .../webapp/app/shared/pipes/safe-html.pipe.ts | 2 +- .../app/shared/pipes/shared-pipes.module.ts | 6 ---- .../component/bonus/bonus.component.spec.ts | 4 +-- .../exam-navigation-sidebar.component.spec.ts | 2 +- .../detailed-grading-system.component.spec.ts | 1 - .../grading-key-overview.component.spec.ts | 2 +- .../grading-key-table.component.spec.ts | 6 ++-- ...ing-system-presentations.component.spec.ts | 3 +- .../grading-system.component.spec.ts | 8 +---- .../interval-grading-system.component.spec.ts | 1 - 27 files changed, 153 insertions(+), 90 deletions(-) diff --git a/jest.config.js b/jest.config.js index efdaeff41351..eb987708aa76 100644 --- a/jest.config.js +++ b/jest.config.js @@ -105,10 +105,10 @@ module.exports = { coverageThreshold: { global: { // TODO: in the future, the following values should increase to at least 90% - statements: 87.73, - branches: 73.87, - functions: 82.44, - lines: 87.78, + statements: 87.78, + branches: 73.93, + functions: 82.47, + lines: 87.83, }, }, coverageReporters: ['clover', 'json', 'lcov', 'text-summary'], diff --git a/src/main/webapp/app/exam/manage/exam-management.module.ts b/src/main/webapp/app/exam/manage/exam-management.module.ts index e95bb9d2d5ee..d3e3677195d2 100644 --- a/src/main/webapp/app/exam/manage/exam-management.module.ts +++ b/src/main/webapp/app/exam/manage/exam-management.module.ts @@ -66,6 +66,8 @@ import { ArtemisProgrammingExerciseModule } from 'app/exercises/programming/shar import { DetailModule } from 'app/detail-overview-list/detail.module'; import { ArtemisDurationFromSecondsPipe } from 'app/shared/pipes/artemis-duration-from-seconds.pipe'; import { NoDataComponent } from 'app/shared/no-data-component'; +import { SafeHtmlPipe } from 'app/shared/pipes/safe-html.pipe'; +import { GradeStepBoundsPipe } from 'app/shared/pipes/grade-step-bounds.pipe'; import { examScoresState } from 'app/exam/exam-scores/exam-scores.route'; import { GitDiffLineStatComponent } from 'app/exercises/programming/git-diff-report/git-diff-line-stat.component'; @@ -73,7 +75,7 @@ const ENTITY_STATES = [...examManagementState, ...examScoresState]; @NgModule({ // TODO: For better modularization we could define an exercise module with the corresponding exam routes - providers: [ArtemisDurationFromSecondsPipe], + providers: [ArtemisDurationFromSecondsPipe, SafeHtmlPipe, GradeStepBoundsPipe], imports: [ RouterModule.forChild(ENTITY_STATES), ArtemisTextExerciseModule, @@ -108,6 +110,9 @@ const ENTITY_STATES = [...examManagementState, ...examScoresState]; DetailModule, NoDataComponent, GitDiffLineStatComponent, + SafeHtmlPipe, + GradeStepBoundsPipe, + BonusComponent, ], declarations: [ ExamManagementComponent, @@ -135,7 +140,6 @@ const ENTITY_STATES = [...examManagementState, ...examScoresState]; StudentExamDetailTableRowComponent, ExamImportComponent, ExamExerciseImportComponent, - BonusComponent, ExamEditWorkingTimeComponent, ExamEditWorkingTimeDialogComponent, ExamLiveAnnouncementCreateModalComponent, diff --git a/src/main/webapp/app/exercises/programming/shared/instructions-render/programming-exercise-instructions-render.module.ts b/src/main/webapp/app/exercises/programming/shared/instructions-render/programming-exercise-instructions-render.module.ts index 585b2cc832d2..fbfc110d7304 100644 --- a/src/main/webapp/app/exercises/programming/shared/instructions-render/programming-exercise-instructions-render.module.ts +++ b/src/main/webapp/app/exercises/programming/shared/instructions-render/programming-exercise-instructions-render.module.ts @@ -6,9 +6,10 @@ import { ProgrammingExerciseInstructionStepWizardComponent } from 'app/exercises import { ProgrammingExerciseInstructionTaskStatusComponent } from 'app/exercises/programming/shared/instructions-render/task/programming-exercise-instruction-task-status.component'; import { ArtemisMarkdownModule } from 'app/shared/markdown.module'; import { ExamExerciseUpdateHighlighterModule } from 'app/exam/participate/exercises/exam-exercise-update-highlighter/exam-exercise-update-highlighter.module'; +import { SafeHtmlPipe } from 'app/shared/pipes/safe-html.pipe'; @NgModule({ - imports: [ArtemisSharedModule, ArtemisResultModule, ArtemisMarkdownModule, ExamExerciseUpdateHighlighterModule], + imports: [ArtemisSharedModule, ArtemisResultModule, ArtemisMarkdownModule, ExamExerciseUpdateHighlighterModule, SafeHtmlPipe], declarations: [ProgrammingExerciseInstructionComponent, ProgrammingExerciseInstructionStepWizardComponent, ProgrammingExerciseInstructionTaskStatusComponent], exports: [ProgrammingExerciseInstructionComponent], }) diff --git a/src/main/webapp/app/grading-system/base-grading-system/base-grading-system.component.ts b/src/main/webapp/app/grading-system/base-grading-system/base-grading-system.component.ts index 21e2a3aef781..06ef364fef65 100644 --- a/src/main/webapp/app/grading-system/base-grading-system/base-grading-system.component.ts +++ b/src/main/webapp/app/grading-system/base-grading-system/base-grading-system.component.ts @@ -1,4 +1,4 @@ -import { Component, OnInit } from '@angular/core'; +import { Component, OnInit, inject } from '@angular/core'; import { GradeType, GradingScale } from 'app/entities/grading-scale.model'; import { GradeStep } from 'app/entities/grade-step.model'; import { ActivatedRoute } from '@angular/router'; @@ -39,6 +39,12 @@ export enum GradeEditMode { @Component({ template: '' }) export abstract class BaseGradingSystemComponent implements OnInit { + protected gradingSystemService = inject(GradingSystemService); + private route = inject(ActivatedRoute); + private translateService = inject(TranslateService); + private courseService = inject(CourseManagementService); + private examService = inject(ExamManagementService); + GradeType = GradeType; ButtonSize = ButtonSize; GradingScale = GradingScale; @@ -67,14 +73,6 @@ export abstract class BaseGradingSystemComponent implements OnInit { faExclamationTriangle = faExclamationTriangle; faInfo = faInfo; - constructor( - protected gradingSystemService: GradingSystemService, - private route: ActivatedRoute, - private translateService: TranslateService, - private courseService: CourseManagementService, - private examService: ExamManagementService, - ) {} - ngOnInit(): void { this.route.parent?.params.subscribe((params) => { this.isLoading = true; diff --git a/src/main/webapp/app/grading-system/bonus/bonus.component.ts b/src/main/webapp/app/grading-system/bonus/bonus.component.ts index f818de750adf..8d420789e637 100644 --- a/src/main/webapp/app/grading-system/bonus/bonus.component.ts +++ b/src/main/webapp/app/grading-system/bonus/bonus.component.ts @@ -1,4 +1,4 @@ -import { Component, OnInit } from '@angular/core'; +import { Component, OnInit, inject } from '@angular/core'; import { BonusService } from 'app/grading-system/bonus/bonus.service'; import { GradingSystemService } from 'app/grading-system/grading-system.service'; import { GradingScale } from 'app/entities/grading-scale.model'; @@ -9,10 +9,14 @@ import { faExclamationTriangle, faPlus, faQuestionCircle, faSave, faTimes } from import { GradeStep, GradeStepsDTO } from 'app/entities/grade-step.model'; import { ButtonSize } from 'app/shared/components/button.component'; import { Subject, forkJoin, of } from 'rxjs'; -import { TranslateService } from '@ngx-translate/core'; import { SearchTermPageableSearch, SortingOrder } from 'app/shared/table/pageable-table'; import { GradeEditMode } from 'app/grading-system/base-grading-system/base-grading-system.component'; import { AlertService } from 'app/core/util/alert.service'; +import { ArtemisSharedModule } from 'app/shared/shared.module'; +import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; +import { SafeHtmlPipe } from 'app/shared/pipes/safe-html.pipe'; +import { GradeStepBoundsPipe } from 'app/shared/pipes/grade-step-bounds.pipe'; +import { ArtemisModePickerModule } from 'app/exercises/shared/mode-picker/mode-picker.module'; export enum BonusStrategyOption { GRADES, @@ -28,8 +32,16 @@ export enum BonusStrategyDiscreteness { selector: 'jhi-bonus', templateUrl: './bonus.component.html', styleUrls: ['./bonus.component.scss'], + standalone: true, + imports: [ArtemisSharedModule, ArtemisSharedComponentModule, ArtemisModePickerModule, SafeHtmlPipe, GradeStepBoundsPipe], }) export class BonusComponent implements OnInit { + private bonusService = inject(BonusService); + private gradingSystemService = inject(GradingSystemService); + private route = inject(ActivatedRoute); + private router = inject(Router); + private alertService = inject(AlertService); + readonly CALCULATION_PLUS = 1; readonly CALCULATION_MINUS = -1; @@ -98,15 +110,6 @@ export class BonusComponent implements OnInit { sortedColumn: 'ID', }; - constructor( - private bonusService: BonusService, - private gradingSystemService: GradingSystemService, - private route: ActivatedRoute, - private router: Router, - private translateService: TranslateService, - private alertService: AlertService, - ) {} - ngOnInit(): void { this.isLoading = true; diff --git a/src/main/webapp/app/grading-system/bonus/bonus.service.ts b/src/main/webapp/app/grading-system/bonus/bonus.service.ts index 5da0a5ce1931..4bbf1176e80d 100644 --- a/src/main/webapp/app/grading-system/bonus/bonus.service.ts +++ b/src/main/webapp/app/grading-system/bonus/bonus.service.ts @@ -1,4 +1,4 @@ -import { Injectable } from '@angular/core'; +import { Injectable, inject } from '@angular/core'; import { HttpClient, HttpParams, HttpResponse } from '@angular/common/http'; import { Observable } from 'rxjs'; import { GradeStep, GradeStepsDTO } from 'app/entities/grade-step.model'; @@ -11,12 +11,10 @@ export type EntityResponseType = HttpResponse; @Injectable({ providedIn: 'root' }) export class BonusService { - public resourceUrl = 'api'; + private http = inject(HttpClient); + private gradingSystemService = inject(GradingSystemService); - constructor( - private http: HttpClient, - private gradingSystemService: GradingSystemService, - ) {} + public resourceUrl = 'api'; /** * Deletes the bonus. diff --git a/src/main/webapp/app/grading-system/detailed-grading-system/detailed-grading-system.component.ts b/src/main/webapp/app/grading-system/detailed-grading-system/detailed-grading-system.component.ts index cee2be24106c..103ace36cbf3 100644 --- a/src/main/webapp/app/grading-system/detailed-grading-system/detailed-grading-system.component.ts +++ b/src/main/webapp/app/grading-system/detailed-grading-system/detailed-grading-system.component.ts @@ -1,11 +1,19 @@ import { Component } from '@angular/core'; import { BaseGradingSystemComponent, CsvGradeStep } from 'app/grading-system/base-grading-system/base-grading-system.component'; import { parse } from 'papaparse'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; +import { FormsModule } from '@angular/forms'; +import { FaIconComponent } from '@fortawesome/angular-fontawesome'; +import { ArtemisSharedModule } from 'app/shared/shared.module'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; @Component({ selector: 'jhi-detailed-grading-system', templateUrl: './detailed-grading-system.component.html', styleUrls: ['./detailed-grading-system.component.scss'], + standalone: true, + imports: [TranslateDirective, ArtemisSharedComponentModule, FormsModule, FaIconComponent, ArtemisSharedModule, ArtemisTranslatePipe], }) export class DetailedGradingSystemComponent extends BaseGradingSystemComponent { /** diff --git a/src/main/webapp/app/grading-system/grading-key-overview/grading-key-overview.component.ts b/src/main/webapp/app/grading-system/grading-key-overview/grading-key-overview.component.ts index baa6d137e54e..25b01e083d2e 100644 --- a/src/main/webapp/app/grading-system/grading-key-overview/grading-key-overview.component.ts +++ b/src/main/webapp/app/grading-system/grading-key-overview/grading-key-overview.component.ts @@ -1,31 +1,33 @@ -import { Component, OnInit } from '@angular/core'; +import { Component, OnInit, inject } from '@angular/core'; import { ActivatedRoute } from '@angular/router'; -import { GradingSystemService } from 'app/grading-system/grading-system.service'; import { GradeStep } from 'app/entities/grade-step.model'; import { ArtemisNavigationUtilService } from 'app/utils/navigation.utils'; import { faChevronLeft, faPrint } from '@fortawesome/free-solid-svg-icons'; import { ThemeService } from 'app/core/theme/theme.service'; import { loadGradingKeyUrlParams } from 'app/grading-system/grading-key-overview/grading-key-helper'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { GradingKeyTableComponent } from 'app/grading-system/grading-key-overview/grading-key/grading-key-table.component'; +import { FaIconComponent } from '@fortawesome/angular-fontawesome'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; @Component({ selector: 'jhi-grade-key-overview', templateUrl: './grading-key-overview.component.html', styleUrls: ['./grading-key-overview.scss'], + standalone: true, + imports: [TranslateDirective, GradingKeyTableComponent, FaIconComponent, ArtemisTranslatePipe], }) export class GradingKeyOverviewComponent implements OnInit { + private route = inject(ActivatedRoute); + private navigationUtilService = inject(ArtemisNavigationUtilService); + private themeService = inject(ThemeService); + readonly faChevronLeft = faChevronLeft; readonly faPrint = faPrint; plagiarismGrade: string; noParticipationGrade: string; - constructor( - private route: ActivatedRoute, - private gradingSystemService: GradingSystemService, - private navigationUtilService: ArtemisNavigationUtilService, - private themeService: ThemeService, - ) {} - isExam = false; courseId?: number; diff --git a/src/main/webapp/app/grading-system/grading-key-overview/grading-key-overview.module.ts b/src/main/webapp/app/grading-system/grading-key-overview/grading-key-overview.module.ts index 9f7deffb6073..3ce1753cc341 100644 --- a/src/main/webapp/app/grading-system/grading-key-overview/grading-key-overview.module.ts +++ b/src/main/webapp/app/grading-system/grading-key-overview/grading-key-overview.module.ts @@ -5,8 +5,7 @@ import { ArtemisSharedComponentModule } from 'app/shared/components/shared-compo import { GradingKeyTableComponent } from 'app/grading-system/grading-key-overview/grading-key/grading-key-table.component'; @NgModule({ - declarations: [GradingKeyOverviewComponent, GradingKeyTableComponent], - imports: [ArtemisSharedModule, ArtemisSharedComponentModule], + imports: [ArtemisSharedModule, ArtemisSharedComponentModule, GradingKeyOverviewComponent, GradingKeyTableComponent], exports: [GradingKeyOverviewComponent, GradingKeyTableComponent], }) export class GradingKeyOverviewModule {} diff --git a/src/main/webapp/app/grading-system/grading-key-overview/grading-key/grading-key-table.component.ts b/src/main/webapp/app/grading-system/grading-key-overview/grading-key/grading-key-table.component.ts index fc5d5bd07882..bfd023f1289d 100644 --- a/src/main/webapp/app/grading-system/grading-key-overview/grading-key/grading-key-table.component.ts +++ b/src/main/webapp/app/grading-system/grading-key-overview/grading-key/grading-key-table.component.ts @@ -1,4 +1,4 @@ -import { Component, Input, OnInit } from '@angular/core'; +import { Component, Input, OnInit, inject } from '@angular/core'; import { GradingSystemService } from 'app/grading-system/grading-system.service'; import { GradeStep, GradeStepsDTO } from 'app/entities/grade-step.model'; import { GradeType, GradingScale } from 'app/entities/grading-scale.model'; @@ -11,13 +11,25 @@ import { ScoresStorageService } from 'app/course/course-scores/scores-storage.se import { ScoreType } from 'app/shared/constants/score-type.constants'; import { ActivatedRoute } from '@angular/router'; import { loadGradingKeyUrlParams } from 'app/grading-system/grading-key-overview/grading-key-helper'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; +import { GradeStepBoundsPipe } from 'app/shared/pipes/grade-step-bounds.pipe'; +import { SafeHtmlPipe } from 'app/shared/pipes/safe-html.pipe'; +import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; @Component({ selector: 'jhi-grade-key-table', templateUrl: './grading-key-table.component.html', styleUrls: ['../grading-key-overview.scss'], + standalone: true, + imports: [TranslateDirective, ArtemisTranslatePipe, GradeStepBoundsPipe, SafeHtmlPipe, ArtemisSharedComponentModule], }) export class GradingKeyTableComponent implements OnInit { + private route = inject(ActivatedRoute); + private gradingSystemService = inject(GradingSystemService); + private bonusService = inject(BonusService); + private scoresStorageService = inject(ScoresStorageService); + readonly faChevronLeft = faChevronLeft; readonly GradeEditMode = GradeEditMode; @@ -25,13 +37,6 @@ export class GradingKeyTableComponent implements OnInit { @Input() studentGradeOrBonusPointsOrGradeBonus?: string; @Input() forBonus?: boolean; - constructor( - private route: ActivatedRoute, - private gradingSystemService: GradingSystemService, - private bonusService: BonusService, - private scoresStorageService: ScoresStorageService, - ) {} - plagiarismGrade: string; noParticipationGrade: string; diff --git a/src/main/webapp/app/grading-system/grading-system-info-modal/grading-system-info-modal.component.ts b/src/main/webapp/app/grading-system/grading-system-info-modal/grading-system-info-modal.component.ts index c302881e576e..8a78ece70f82 100644 --- a/src/main/webapp/app/grading-system/grading-system-info-modal/grading-system-info-modal.component.ts +++ b/src/main/webapp/app/grading-system/grading-system-info-modal/grading-system-info-modal.component.ts @@ -1,15 +1,20 @@ -import { Component } from '@angular/core'; +import { Component, inject } from '@angular/core'; import { faQuestionCircle } from '@fortawesome/free-regular-svg-icons'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { FaIconComponent } from '@fortawesome/angular-fontawesome'; @Component({ selector: 'jhi-grading-system-info-modal', templateUrl: './grading-system-info-modal.component.html', + standalone: true, + imports: [TranslateDirective, FaIconComponent], }) export class GradingSystemInfoModalComponent { + private modalService = inject(NgbModal); + // Icons farQuestionCircle = faQuestionCircle; - constructor(private modalService: NgbModal) {} /** * Open a large modal with the given content. diff --git a/src/main/webapp/app/grading-system/grading-system-presentations/grading-system-presentations.component.ts b/src/main/webapp/app/grading-system/grading-system-presentations/grading-system-presentations.component.ts index 0aa2e3202a3e..4d0c120205cb 100644 --- a/src/main/webapp/app/grading-system/grading-system-presentations/grading-system-presentations.component.ts +++ b/src/main/webapp/app/grading-system/grading-system-presentations/grading-system-presentations.component.ts @@ -1,6 +1,10 @@ import { Component, Input, OnChanges } from '@angular/core'; import { ModePickerOption } from 'app/exercises/shared/mode-picker/mode-picker.component'; import { GradingScale } from 'app/entities/grading-scale.model'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; +import { ArtemisModePickerModule } from 'app/exercises/shared/mode-picker/mode-picker.module'; +import { FormsModule } from '@angular/forms'; export enum PresentationType { NONE = 'none', @@ -22,6 +26,8 @@ export interface PresentationsConfig { selector: 'jhi-grading-system-presentations', templateUrl: './grading-system-presentations.component.html', styleUrls: ['./grading-system-presentations.component.scss'], + standalone: true, + imports: [TranslateDirective, ArtemisSharedComponentModule, ArtemisModePickerModule, FormsModule], }) export class GradingSystemPresentationsComponent implements OnChanges { readonly NONE = PresentationType.NONE; diff --git a/src/main/webapp/app/grading-system/grading-system.component.ts b/src/main/webapp/app/grading-system/grading-system.component.ts index d947f62e340f..4bd229f5389a 100644 --- a/src/main/webapp/app/grading-system/grading-system.component.ts +++ b/src/main/webapp/app/grading-system/grading-system.component.ts @@ -1,16 +1,40 @@ -import { Component, OnInit } from '@angular/core'; -import { ActivatedRoute } from '@angular/router'; +import { Component, OnInit, inject } from '@angular/core'; +import { ActivatedRoute, RouterLink, RouterLinkActive, RouterOutlet } from '@angular/router'; import { DocumentationType } from 'app/shared/components/documentation-button/documentation-button.component'; import { GradeType } from 'app/entities/grading-scale.model'; import { BaseGradingSystemComponent } from 'app/grading-system/base-grading-system/base-grading-system.component'; import { faExclamationTriangle } from '@fortawesome/free-solid-svg-icons'; +import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { GradingSystemInfoModalComponent } from './grading-system-info-modal/grading-system-info-modal.component'; +import { FaIconComponent } from '@fortawesome/angular-fontawesome'; +import { NgbTooltip } from '@ng-bootstrap/ng-bootstrap'; +import { FormsModule } from '@angular/forms'; +import { GradingSystemPresentationsComponent } from './grading-system-presentations/grading-system-presentations.component'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; @Component({ selector: 'jhi-grading-system', templateUrl: './grading-system.component.html', styleUrls: ['./grading-system.component.scss'], + standalone: true, + imports: [ + ArtemisSharedComponentModule, + TranslateDirective, + GradingSystemInfoModalComponent, + FaIconComponent, + NgbTooltip, + FormsModule, + GradingSystemPresentationsComponent, + RouterLink, + RouterLinkActive, + RouterOutlet, + ArtemisTranslatePipe, + ], }) export class GradingSystemComponent implements OnInit { + private route = inject(ActivatedRoute); + readonly GradeType = GradeType; courseId?: number; @@ -23,8 +47,6 @@ export class GradingSystemComponent implements OnInit { // Icons readonly faExclamationTriangle = faExclamationTriangle; - constructor(private route: ActivatedRoute) {} - ngOnInit(): void { this.route.params.subscribe((params) => { this.courseId = Number(params['courseId']); diff --git a/src/main/webapp/app/grading-system/grading-system.module.ts b/src/main/webapp/app/grading-system/grading-system.module.ts index 9de479f9b748..5b9e460aa6f7 100644 --- a/src/main/webapp/app/grading-system/grading-system.module.ts +++ b/src/main/webapp/app/grading-system/grading-system.module.ts @@ -11,8 +11,17 @@ import { ArtemisSharedComponentModule } from 'app/shared/components/shared-compo import { GradingSystemPresentationsComponent } from 'app/grading-system/grading-system-presentations/grading-system-presentations.component'; @NgModule({ - declarations: [GradingSystemComponent, DetailedGradingSystemComponent, IntervalGradingSystemComponent, GradingSystemInfoModalComponent, GradingSystemPresentationsComponent], - imports: [ArtemisSharedModule, RouterModule.forChild(gradingSystemState), ArtemisModePickerModule, ArtemisSharedComponentModule], + imports: [ + ArtemisSharedModule, + RouterModule.forChild(gradingSystemState), + ArtemisModePickerModule, + ArtemisSharedComponentModule, + GradingSystemComponent, + DetailedGradingSystemComponent, + IntervalGradingSystemComponent, + GradingSystemInfoModalComponent, + GradingSystemPresentationsComponent, + ], exports: [GradingSystemComponent, GradingSystemInfoModalComponent], }) export class GradingSystemModule {} diff --git a/src/main/webapp/app/grading-system/grading-system.service.ts b/src/main/webapp/app/grading-system/grading-system.service.ts index f0f44eecd997..4b20cd46a698 100644 --- a/src/main/webapp/app/grading-system/grading-system.service.ts +++ b/src/main/webapp/app/grading-system/grading-system.service.ts @@ -1,4 +1,4 @@ -import { Injectable } from '@angular/core'; +import { Injectable, inject } from '@angular/core'; import { GradingScale } from 'app/entities/grading-scale.model'; import { HttpClient, HttpParams, HttpResponse } from '@angular/common/http'; import { Observable } from 'rxjs'; @@ -13,9 +13,9 @@ export type EntityArrayResponseType = HttpResponse; @Injectable({ providedIn: 'root' }) export class GradingSystemService { - public resourceUrl = 'api/courses'; + private http = inject(HttpClient); - constructor(private http: HttpClient) {} + public resourceUrl = 'api/courses'; /** * Store a new grading scale for course on the server diff --git a/src/main/webapp/app/grading-system/interval-grading-system/interval-grading-system.component.ts b/src/main/webapp/app/grading-system/interval-grading-system/interval-grading-system.component.ts index c185cddb890a..73a86c1feedb 100644 --- a/src/main/webapp/app/grading-system/interval-grading-system/interval-grading-system.component.ts +++ b/src/main/webapp/app/grading-system/interval-grading-system/interval-grading-system.component.ts @@ -3,11 +3,32 @@ import { GradeStep } from 'app/entities/grade-step.model'; import { ModePickerOption } from 'app/exercises/shared/mode-picker/mode-picker.component'; import { BaseGradingSystemComponent, CsvGradeStep, GradeEditMode } from 'app/grading-system/base-grading-system/base-grading-system.component'; import { parse } from 'papaparse'; +import { TranslateDirective } from 'app/shared/language/translate.directive'; +import { ArtemisModePickerModule } from 'app/exercises/shared/mode-picker/mode-picker.module'; +import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; +import { FormsModule } from '@angular/forms'; +import { FaIconComponent } from '@fortawesome/angular-fontawesome'; +import { ArtemisSharedModule } from 'app/shared/shared.module'; +import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; +import { SafeHtmlPipe } from 'app/shared/pipes/safe-html.pipe'; +import { GradeStepBoundsPipe } from 'app/shared/pipes/grade-step-bounds.pipe'; @Component({ selector: 'jhi-interval-grading-system', templateUrl: './interval-grading-system.component.html', styleUrls: ['./interval-grading-system.component.scss'], + standalone: true, + imports: [ + TranslateDirective, + ArtemisModePickerModule, + ArtemisSharedComponentModule, + FormsModule, + FaIconComponent, + ArtemisSharedModule, + ArtemisTranslatePipe, + SafeHtmlPipe, + GradeStepBoundsPipe, + ], }) export class IntervalGradingSystemComponent extends BaseGradingSystemComponent { readonly GradeEditMode = GradeEditMode; diff --git a/src/main/webapp/app/shared/pipes/grade-step-bounds.pipe.ts b/src/main/webapp/app/shared/pipes/grade-step-bounds.pipe.ts index b589911fa98b..6e9ece61b74b 100644 --- a/src/main/webapp/app/shared/pipes/grade-step-bounds.pipe.ts +++ b/src/main/webapp/app/shared/pipes/grade-step-bounds.pipe.ts @@ -6,6 +6,7 @@ import { round } from 'app/shared/util/utils'; @Pipe({ name: 'gradeStepBounds', pure: false, + standalone: true, }) export class GradeStepBoundsPipe implements PipeTransform { /** diff --git a/src/main/webapp/app/shared/pipes/safe-html.pipe.ts b/src/main/webapp/app/shared/pipes/safe-html.pipe.ts index c880d1befd71..56d9728cb387 100644 --- a/src/main/webapp/app/shared/pipes/safe-html.pipe.ts +++ b/src/main/webapp/app/shared/pipes/safe-html.pipe.ts @@ -1,7 +1,7 @@ import { Pipe, PipeTransform } from '@angular/core'; import { DomSanitizer } from '@angular/platform-browser'; -@Pipe({ name: 'safeHtml' }) +@Pipe({ name: 'safeHtml', standalone: true }) export class SafeHtmlPipe implements PipeTransform { constructor(private sanitizer: DomSanitizer) {} diff --git a/src/main/webapp/app/shared/pipes/shared-pipes.module.ts b/src/main/webapp/app/shared/pipes/shared-pipes.module.ts index 21f41cd9ff06..4ea6c175a3e2 100644 --- a/src/main/webapp/app/shared/pipes/shared-pipes.module.ts +++ b/src/main/webapp/app/shared/pipes/shared-pipes.module.ts @@ -3,20 +3,17 @@ import { SafeResourceUrlPipe } from 'app/shared/pipes/safe-resource-url.pipe'; import { KeysPipe } from 'app/shared/pipes/keys.pipe'; import { RemoveKeysPipe } from 'app/shared/pipes/remove-keys.pipe'; import { TypeCheckPipe } from 'app/shared/pipes/type-check.pipe'; -import { SafeHtmlPipe } from 'app/shared/pipes/safe-html.pipe'; import { SafeUrlPipe } from 'app/shared/pipes/safe-url.pipe'; import { TruncatePipe } from 'app/shared/pipes/truncate.pipe'; import { ExerciseTypePipe } from 'app/shared/pipes/exercise-type.pipe'; import { ExerciseCourseTitlePipe } from 'app/shared/pipes/exercise-course-title.pipe'; import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; -import { GradeStepBoundsPipe } from 'app/shared/pipes/grade-step-bounds.pipe'; import { NegatedTypeCheckPipe } from 'app/shared/pipes/negated-type-check.pipe'; import { SearchFilterPipe } from 'app/shared/pipes/search-filter.pipe'; import { AsPipe } from 'app/shared/pipes/as.pipe'; @NgModule({ declarations: [ - SafeHtmlPipe, SafeUrlPipe, SafeResourceUrlPipe, RemoveKeysPipe, @@ -26,12 +23,10 @@ import { AsPipe } from 'app/shared/pipes/as.pipe'; TypeCheckPipe, NegatedTypeCheckPipe, TruncatePipe, - GradeStepBoundsPipe, SearchFilterPipe, AsPipe, ], exports: [ - SafeHtmlPipe, SafeUrlPipe, RemoveKeysPipe, ExerciseCourseTitlePipe, @@ -41,7 +36,6 @@ import { AsPipe } from 'app/shared/pipes/as.pipe'; NegatedTypeCheckPipe, TruncatePipe, SafeResourceUrlPipe, - GradeStepBoundsPipe, SearchFilterPipe, AsPipe, ], diff --git a/src/test/javascript/spec/component/bonus/bonus.component.spec.ts b/src/test/javascript/spec/component/bonus/bonus.component.spec.ts index 367bfcff0184..6a1a1bcf1f6e 100644 --- a/src/test/javascript/spec/component/bonus/bonus.component.spec.ts +++ b/src/test/javascript/spec/component/bonus/bonus.component.spec.ts @@ -16,7 +16,7 @@ import { GradeStepBoundsPipe } from 'app/shared/pipes/grade-step-bounds.pipe'; import { GradeStep, GradeStepsDTO } from 'app/entities/grade-step.model'; import { HttpResponse } from '@angular/common/http'; import { DeleteDialogComponent } from 'app/shared/delete-dialog/delete-dialog.component'; -import { FormsModule, NgModel, ReactiveFormsModule } from '@angular/forms'; +import { FormsModule, ReactiveFormsModule } from '@angular/forms'; import { NgbTooltipModule } from '@ng-bootstrap/ng-bootstrap'; import { DeleteButtonDirective } from 'app/shared/delete-dialog/delete-button.directive'; @@ -252,13 +252,11 @@ describe('BonusComponent', () => { await TestBed.configureTestingModule({ imports: [ArtemisTestModule, FormsModule, ReactiveFormsModule, MockModule(NgbTooltipModule)], declarations: [ - BonusComponent, MockPipe(ArtemisTranslatePipe), MockPipe(SafeHtmlPipe), MockComponent(ModePickerComponent), MockPipe(GradeStepBoundsPipe), MockComponent(DeleteDialogComponent), - MockDirective(NgModel), MockDirective(DeleteButtonDirective), ], providers: [ diff --git a/src/test/javascript/spec/component/exam/participate/exam-navigation-sidebar.component.spec.ts b/src/test/javascript/spec/component/exam/participate/exam-navigation-sidebar.component.spec.ts index 2645f07b9ce2..356fc70b98d8 100644 --- a/src/test/javascript/spec/component/exam/participate/exam-navigation-sidebar.component.spec.ts +++ b/src/test/javascript/spec/component/exam/participate/exam-navigation-sidebar.component.spec.ts @@ -36,7 +36,7 @@ describe('ExamNavigationSidebarComponent', () => { beforeEach(() => { TestBed.configureTestingModule({ imports: [ArtemisTestModule, TranslateTestingModule, MockModule(NgbTooltipModule), MockModule(ArtemisSharedCommonModule)], - declarations: [ExamNavigationSidebarComponent, MockComponent(ExamTimerComponent), MockComponent(ExamLiveEventsButtonComponent), MockComponent(SearchFilterComponent)], + declarations: [MockComponent(ExamTimerComponent), MockComponent(ExamLiveEventsButtonComponent), MockComponent(SearchFilterComponent)], providers: [ ExamParticipationService, { provide: ExamExerciseUpdateService, useValue: mockExamExerciseUpdateService }, diff --git a/src/test/javascript/spec/component/grading-system/detailed-grading-system.component.spec.ts b/src/test/javascript/spec/component/grading-system/detailed-grading-system.component.spec.ts index 7ecabd855b07..d393fe92dfbb 100644 --- a/src/test/javascript/spec/component/grading-system/detailed-grading-system.component.spec.ts +++ b/src/test/javascript/spec/component/grading-system/detailed-grading-system.component.spec.ts @@ -82,7 +82,6 @@ describe('Detailed Grading System Component', () => { declarations: [ MockDirective(NgModel), MockDirective(NgSelectOption), - DetailedGradingSystemComponent, MockComponent(GradingSystemInfoModalComponent), MockComponent(HelpIconComponent), MockDirective(DeleteButtonDirective), diff --git a/src/test/javascript/spec/component/grading-system/grading-key-overview.component.spec.ts b/src/test/javascript/spec/component/grading-system/grading-key-overview.component.spec.ts index f1e4c6e0739b..644aa227dd36 100644 --- a/src/test/javascript/spec/component/grading-system/grading-key-overview.component.spec.ts +++ b/src/test/javascript/spec/component/grading-system/grading-key-overview.component.spec.ts @@ -38,7 +38,7 @@ describe('GradingKeyOverviewComponent', () => { } as ActivatedRoute; TestBed.configureTestingModule({ - declarations: [ + imports: [ GradingKeyOverviewComponent, MockComponent(GradingKeyTableComponent), MockComponent(FaIconComponent), diff --git a/src/test/javascript/spec/component/grading-system/grading-key-table.component.spec.ts b/src/test/javascript/spec/component/grading-system/grading-key-table.component.spec.ts index 6110d07dde4f..7168efba6da5 100644 --- a/src/test/javascript/spec/component/grading-system/grading-key-table.component.spec.ts +++ b/src/test/javascript/spec/component/grading-system/grading-key-table.component.spec.ts @@ -1,8 +1,7 @@ import { GradingSystemService } from 'app/grading-system/grading-system.service'; import { ComponentFixture, TestBed } from '@angular/core/testing'; import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; -import { MockComponent, MockDirective, MockModule, MockPipe, MockProvider } from 'ng-mocks'; -import { NgbModule } from '@ng-bootstrap/ng-bootstrap'; +import { MockComponent, MockDirective, MockPipe, MockProvider } from 'ng-mocks'; import { ActivatedRoute, ActivatedRouteSnapshot, Params, Router } from '@angular/router'; import { of } from 'rxjs'; import { FaIconComponent } from '@fortawesome/angular-fontawesome'; @@ -71,8 +70,7 @@ describe('GradingKeyTableComponent', () => { } as ActivatedRoute; return TestBed.configureTestingModule({ - imports: [MockModule(NgbModule)], - declarations: [ + imports: [ GradingKeyTableComponent, MockComponent(FaIconComponent), MockPipe(ArtemisTranslatePipe), diff --git a/src/test/javascript/spec/component/grading-system/grading-system-presentations.component.spec.ts b/src/test/javascript/spec/component/grading-system/grading-system-presentations.component.spec.ts index 6900bfbb3363..2e57665dfd3b 100644 --- a/src/test/javascript/spec/component/grading-system/grading-system-presentations.component.spec.ts +++ b/src/test/javascript/spec/component/grading-system/grading-system-presentations.component.spec.ts @@ -2,7 +2,6 @@ import { ComponentFixture, TestBed } from '@angular/core/testing'; import { GradingScale } from 'app/entities/grading-scale.model'; import { GradingSystemPresentationsComponent, PresentationType } from 'app/grading-system/grading-system-presentations/grading-system-presentations.component'; import { Course } from 'app/entities/course.model'; -import { ModePickerComponent } from 'app/exercises/shared/mode-picker/mode-picker.component'; import { HelpIconComponent } from 'app/shared/components/help-icon.component'; import { MockComponent } from 'ng-mocks'; @@ -44,7 +43,7 @@ describe('Grading System Presentations Component', () => { beforeEach(() => { return TestBed.configureTestingModule({ - declarations: [GradingSystemPresentationsComponent, MockComponent(HelpIconComponent), MockComponent(ModePickerComponent)], + declarations: [MockComponent(HelpIconComponent)], }) .compileComponents() .then(() => { diff --git a/src/test/javascript/spec/component/grading-system/grading-system.component.spec.ts b/src/test/javascript/spec/component/grading-system/grading-system.component.spec.ts index 31fad3a99ef5..2f45c6ee0466 100644 --- a/src/test/javascript/spec/component/grading-system/grading-system.component.spec.ts +++ b/src/test/javascript/spec/component/grading-system/grading-system.component.spec.ts @@ -21,13 +21,7 @@ describe('Grading System Component', () => { beforeEach(() => { TestBed.configureTestingModule({ imports: [ArtemisTestModule, RouterModule.forRoot([])], - declarations: [ - GradingSystemComponent, - MockDirective(NgModel), - MockComponent(DocumentationButtonComponent), - MockComponent(GradingSystemInfoModalComponent), - MockPipe(ArtemisTranslatePipe), - ], + declarations: [MockDirective(NgModel), MockComponent(DocumentationButtonComponent), MockComponent(GradingSystemInfoModalComponent), MockPipe(ArtemisTranslatePipe)], providers: [{ provide: ActivatedRoute, useValue: route }], }) .compileComponents() diff --git a/src/test/javascript/spec/component/grading-system/interval-grading-system.component.spec.ts b/src/test/javascript/spec/component/grading-system/interval-grading-system.component.spec.ts index 531b3f1f2064..fa0f490c3e09 100644 --- a/src/test/javascript/spec/component/grading-system/interval-grading-system.component.spec.ts +++ b/src/test/javascript/spec/component/grading-system/interval-grading-system.component.spec.ts @@ -80,7 +80,6 @@ describe('Interval Grading System Component', () => { declarations: [ MockDirective(NgModel), MockDirective(NgSelectOption), - IntervalGradingSystemComponent, MockComponent(GradingSystemInfoModalComponent), MockComponent(HelpIconComponent), MockComponent(ModePickerComponent), From e3f72f18f12a246bfaf2b46f2760cf9c18927b26 Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Sun, 5 Jan 2025 22:21:30 +0100 Subject: [PATCH 31/34] Development: Adapt server test coverage --- gradle/test.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/gradle/test.gradle b/gradle/test.gradle index 52ef3fe25ea5..5005a226659b 100644 --- a/gradle/test.gradle +++ b/gradle/test.gradle @@ -114,7 +114,7 @@ jacocoTestCoverageVerification { counter = "CLASS" value = "MISSEDCOUNT" // TODO: in the future the following value should become less than 10 - maximum = 55 + maximum = 57 } } } From 3448c7390099ddb9a1a04fe223d7afd0deaaca54 Mon Sep 17 00:00:00 2001 From: Florian Glombik <63976129+florian-glombik@users.noreply.github.com> Date: Sun, 5 Jan 2025 22:28:09 +0100 Subject: [PATCH 32/34] Lectures: Replace guided mode with status bar edit and create view (#9655) --- jest.config.js | 8 +- .../webapp/app/entities/attachment.model.ts | 2 - .../close-edit-lecture-modal.component.html | 4 +- .../close-edit-lecture-modal.component.ts | 0 .../lecture/hasLectureUnsavedChanges.guard.ts | 4 +- .../lecture/lecture-attachments.component.ts | 4 +- .../lecture-period.component.html | 4 +- .../lecture-unit-management.component.ts | 68 +++--- .../lecture-units.component.html} | 8 +- .../lecture-units.component.ts} | 30 ++- .../app/lecture/lecture-update.component.html | 219 +++++++++--------- .../app/lecture/lecture-update.component.ts | 128 ++++++---- .../webapp/app/lecture/lecture.component.ts | 10 +- src/main/webapp/app/lecture/lecture.module.ts | 17 +- .../webapp/app/lecture/lecture.service.ts | 2 + .../lecture-update-wizard-step.component.html | 16 -- .../lecture-update-wizard-step.component.ts | 24 -- .../lecture-update-wizard.component.html | 54 ----- .../lecture-update-wizard.component.scss | 70 ------ .../lecture-update-wizard.component.ts | 104 --------- .../lecture-wizard-attachments.component.html | 7 - .../lecture-wizard-attachments.component.ts | 13 -- .../lecture-wizard-title.component.html | 11 - .../lecture-wizard-title.component.ts | 14 -- .../exercise-filter-modal.component.html | 5 - src/main/webapp/i18n/de/lecture.json | 32 +-- src/main/webapp/i18n/en/lecture.json | 32 +-- ...close-edit-lecture-modal.component.spec.ts | 2 +- ...pec.ts => lecture-units.component.spec.ts} | 16 +- .../lecture/lecture-update.component.spec.ts | 176 +++++++------- ...cture-wizard-attachments.component.spec.ts | 34 --- .../lecture-wizard-title.component.spec.ts | 44 ---- .../lecture-wizard.component.spec.ts | 218 ----------------- .../image-cropper/keyboard.util.spec.ts | 71 ++++++ .../javascript/spec/util/regex.util.spec.ts | 34 +++ .../e2e/lecture/LectureManagement.spec.ts | 11 +- 36 files changed, 524 insertions(+), 972 deletions(-) rename src/main/webapp/app/lecture/{close-edit-lecture-dialog => close-edit-lecture-modal}/close-edit-lecture-modal.component.html (87%) rename src/main/webapp/app/lecture/{close-edit-lecture-dialog => close-edit-lecture-modal}/close-edit-lecture-modal.component.ts (100%) rename src/main/webapp/app/lecture/{wizard-mode/lecture-wizard-units.component.html => lecture-units/lecture-units.component.html} (87%) rename src/main/webapp/app/lecture/{wizard-mode/lecture-wizard-units.component.ts => lecture-units/lecture-units.component.ts} (89%) delete mode 100644 src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard-step.component.html delete mode 100644 src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard-step.component.ts delete mode 100644 src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.html delete mode 100644 src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.scss delete mode 100644 src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.ts delete mode 100644 src/main/webapp/app/lecture/wizard-mode/lecture-wizard-attachments.component.html delete mode 100644 src/main/webapp/app/lecture/wizard-mode/lecture-wizard-attachments.component.ts delete mode 100644 src/main/webapp/app/lecture/wizard-mode/lecture-wizard-title.component.html delete mode 100644 src/main/webapp/app/lecture/wizard-mode/lecture-wizard-title.component.ts rename src/test/javascript/spec/component/lecture/{wizard-mode/lecture-wizard-units.component.spec.ts => lecture-units.component.spec.ts} (98%) delete mode 100644 src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-attachments.component.spec.ts delete mode 100644 src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-title.component.spec.ts delete mode 100644 src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard.component.spec.ts create mode 100644 src/test/javascript/spec/component/shared/image-cropper/keyboard.util.spec.ts create mode 100644 src/test/javascript/spec/util/regex.util.spec.ts diff --git a/jest.config.js b/jest.config.js index eb987708aa76..ab2c159af355 100644 --- a/jest.config.js +++ b/jest.config.js @@ -105,10 +105,10 @@ module.exports = { coverageThreshold: { global: { // TODO: in the future, the following values should increase to at least 90% - statements: 87.78, - branches: 73.93, - functions: 82.47, - lines: 87.83, + statements: 87.81, + branches: 73.97, + functions: 82.50, + lines: 87.86, }, }, coverageReporters: ['clover', 'json', 'lcov', 'text-summary'], diff --git a/src/main/webapp/app/entities/attachment.model.ts b/src/main/webapp/app/entities/attachment.model.ts index c070019513ee..78e6b639da8d 100644 --- a/src/main/webapp/app/entities/attachment.model.ts +++ b/src/main/webapp/app/entities/attachment.model.ts @@ -20,6 +20,4 @@ export class Attachment implements BaseEntity { lecture?: Lecture; exercise?: Exercise; attachmentUnit?: AttachmentUnit; - - constructor() {} } diff --git a/src/main/webapp/app/lecture/close-edit-lecture-dialog/close-edit-lecture-modal.component.html b/src/main/webapp/app/lecture/close-edit-lecture-modal/close-edit-lecture-modal.component.html similarity index 87% rename from src/main/webapp/app/lecture/close-edit-lecture-dialog/close-edit-lecture-modal.component.html rename to src/main/webapp/app/lecture/close-edit-lecture-modal/close-edit-lecture-modal.component.html index afc317c054bc..d554326acf9e 100644 --- a/src/main/webapp/app/lecture/close-edit-lecture-dialog/close-edit-lecture-modal.component.html +++ b/src/main/webapp/app/lecture/close-edit-lecture-modal/close-edit-lecture-modal.component.html @@ -13,12 +13,12 @@
    diff --git a/src/main/webapp/app/lecture/close-edit-lecture-dialog/close-edit-lecture-modal.component.ts b/src/main/webapp/app/lecture/close-edit-lecture-modal/close-edit-lecture-modal.component.ts similarity index 100% rename from src/main/webapp/app/lecture/close-edit-lecture-dialog/close-edit-lecture-modal.component.ts rename to src/main/webapp/app/lecture/close-edit-lecture-modal/close-edit-lecture-modal.component.ts diff --git a/src/main/webapp/app/lecture/hasLectureUnsavedChanges.guard.ts b/src/main/webapp/app/lecture/hasLectureUnsavedChanges.guard.ts index b292d2413830..f608b63a1cb2 100644 --- a/src/main/webapp/app/lecture/hasLectureUnsavedChanges.guard.ts +++ b/src/main/webapp/app/lecture/hasLectureUnsavedChanges.guard.ts @@ -3,10 +3,10 @@ import { CanDeactivateFn } from '@angular/router'; import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap'; import { LectureUpdateComponent } from 'app/lecture/lecture-update.component'; import { Observable, from, of } from 'rxjs'; -import { CloseEditLectureModalComponent } from 'app/lecture/close-edit-lecture-dialog/close-edit-lecture-modal.component'; +import { CloseEditLectureModalComponent } from 'app/lecture/close-edit-lecture-modal/close-edit-lecture-modal.component'; export const hasLectureUnsavedChangesGuard: CanDeactivateFn = (component: LectureUpdateComponent): Observable => { - if (!component.shouldDisplayDismissWarning || component.isShowingWizardMode) { + if (!component.shouldDisplayDismissWarning) { return of(true); } diff --git a/src/main/webapp/app/lecture/lecture-attachments.component.ts b/src/main/webapp/app/lecture/lecture-attachments.component.ts index fd9d7633baa4..95af8f8c3357 100644 --- a/src/main/webapp/app/lecture/lecture-attachments.component.ts +++ b/src/main/webapp/app/lecture/lecture-attachments.component.ts @@ -78,7 +78,9 @@ export class LectureAttachmentsComponent implements OnDestroy { }); private readonly statusChanges = toSignal(this.form.statusChanges ?? 'INVALID'); - isFormValid = computed(() => this.statusChanges() === 'VALID' && this.isFileSelectionValid() && this.datePickerComponent()?.isValid()); + isFormValid = computed( + () => !this.attachmentToBeUpdatedOrCreated() || (this.statusChanges() === 'VALID' && this.isFileSelectionValid() && this.datePickerComponent()?.isValid()), + ); constructor() { effect( diff --git a/src/main/webapp/app/lecture/lecture-period/lecture-period.component.html b/src/main/webapp/app/lecture/lecture-period/lecture-period.component.html index 5434a21491a2..918658de63a7 100644 --- a/src/main/webapp/app/lecture/lecture-period/lecture-period.component.html +++ b/src/main/webapp/app/lecture/lecture-period/lecture-period.component.html @@ -1,6 +1,6 @@
    -

    -

    +

    +

    ; viewButtonAvailable: Record = {}; - updateOrderSubjectSubscription: Subscription; - navigationEndSubscription: Subscription; + private updateOrderSubjectSubscription: Subscription; + private navigationEndSubscription: Subscription; + private activatedRouteSubscription?: Subscription; + private profileInfoSubscription: Subscription; - readonly LectureUnitType = LectureUnitType; - readonly ActionType = ActionType; private dialogErrorSource = new Subject(); dialogError$ = this.dialogErrorSource.asObservable(); - private profileInfoSubscription: Subscription; irisEnabled = false; lectureIngestionEnabled = false; routerEditLinksBase: { [key: string]: string } = { @@ -56,40 +74,13 @@ export class LectureUnitManagementComponent implements OnInit, OnDestroy { [LectureUnitType.ONLINE]: 'online-units', }; - // Icons - readonly faTrash = faTrash; - readonly faPencilAlt = faPencilAlt; - readonly faEye = faEye; - readonly faFileExport = faFileExport; - readonly faRepeat = faRepeat; - readonly faCheckCircle = faCheckCircle; - readonly faSpinner = faSpinner; - - constructor( - private activatedRoute: ActivatedRoute, - private router: Router, - private lectureService: LectureService, - private alertService: AlertService, - public lectureUnitService: LectureUnitService, - private profileService: ProfileService, - private irisSettingsService: IrisSettingsService, - ) {} - - ngOnDestroy(): void { - this.updateOrder(); - this.updateOrderSubjectSubscription.unsubscribe(); - this.dialogErrorSource.unsubscribe(); - this.navigationEndSubscription.unsubscribe(); - this.profileInfoSubscription?.unsubscribe(); - } - ngOnInit(): void { this.navigationEndSubscription = this.router.events.pipe(filter((value) => value instanceof NavigationEnd)).subscribe(() => { this.loadData(); }); this.updateOrderSubject = new Subject(); - this.activatedRoute.parent!.params.subscribe((params) => { + this.activatedRouteSubscription = this.activatedRoute?.parent?.params.subscribe((params) => { this.lectureId ??= +params['lectureId']; if (this.lectureId) { // TODO: the lecture (without units) is already available through the lecture.route.ts resolver, it's not really good that we load it twice @@ -103,6 +94,15 @@ export class LectureUnitManagementComponent implements OnInit, OnDestroy { }); } + ngOnDestroy(): void { + this.updateOrder(); + this.updateOrderSubjectSubscription.unsubscribe(); + this.dialogErrorSource.unsubscribe(); + this.navigationEndSubscription.unsubscribe(); + this.profileInfoSubscription?.unsubscribe(); + this.activatedRouteSubscription?.unsubscribe(); + } + loadData() { this.isLoading = true; // TODO: we actually would like to have the lecture with all units! Posts and competencies are not required here diff --git a/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-units.component.html b/src/main/webapp/app/lecture/lecture-units/lecture-units.component.html similarity index 87% rename from src/main/webapp/app/lecture/wizard-mode/lecture-wizard-units.component.html rename to src/main/webapp/app/lecture/lecture-units/lecture-units.component.html index ef7e6a217539..97c984fe428d 100644 --- a/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-units.component.html +++ b/src/main/webapp/app/lecture/lecture-units/lecture-units.component.html @@ -1,6 +1,6 @@
    -

    -

    +

    +

    @if (!isEditingLectureUnit) { -

    +

    } @else { -

    +

    } @if (isTextUnitFormOpen()) { { + return ( + (this.textUnitForm()?.isFormValid() || !this.isTextUnitFormOpen()) && + (this.videoUnitForm()?.isFormValid() || !this.isVideoUnitFormOpen()) && + (this.onlineUnitForm()?.isFormValid() || !this.isOnlineUnitFormOpen()) && + (this.attachmentUnitForm()?.isFormValid() || !this.isAttachmentUnitFormOpen()) + ); + }); + isEditingLectureUnit: boolean; isTextUnitFormOpen = signal(false); isExerciseUnitFormOpen = signal(false); diff --git a/src/main/webapp/app/lecture/lecture-update.component.html b/src/main/webapp/app/lecture/lecture-update.component.html index 332e6b86f4a1..c73c3178bfe2 100644 --- a/src/main/webapp/app/lecture/lecture-update.component.html +++ b/src/main/webapp/app/lecture/lecture-update.component.html @@ -1,131 +1,130 @@
    - @if (isShowingWizardMode) { - - } - @if (!isShowingWizardMode) { -
    -
    -
    -
    -

    - -
    -
    -
    -
    - -
    + +
    +
    +
    +

    +
    +
    + +
    -

    -

    - -
    - - -
    - - @if (lecture().course) { +

    +

    + @if (courseTitle()) {
    - +
    } + + @if (lecture()) { +
    + + +
    + }
    -
    - - - + +
    + +
    - @if (processUnitMode) { -
    -
      -
    • -
    • -
    • -
    -
    - } - @if (processUnitMode) { -
    - - - -
    -
    - -
    -
    - @if (!fileName && fileInputTouched) { -
    - } + @if (isEditMode()) { +
    +

    +

    + +

    +
    } -
    -
    - - +
    + @if (isEditMode()) { +
    + +
    + + +
    - @if (processUnitMode) { -
    -
    - -
    +
    + } + @if (processUnitMode) { +
    +
      +
    • +
    • +
    • +
    +
    + } + @if (processUnitMode) { +
    + + + +
    +
    +
    +
    + @if (!fileName && fileInputTouched) { +
    }
    - - } + } +
    + @if (processUnitMode) { +
    +
    + +
    +
    + } +
    +
    diff --git a/src/main/webapp/app/lecture/lecture-update.component.ts b/src/main/webapp/app/lecture/lecture-update.component.ts index d4653758c979..092e1348c536 100644 --- a/src/main/webapp/app/lecture/lecture-update.component.ts +++ b/src/main/webapp/app/lecture/lecture-update.component.ts @@ -1,4 +1,4 @@ -import { Component, OnDestroy, OnInit, ViewChild, effect, inject, signal, viewChild } from '@angular/core'; +import { Component, OnDestroy, OnInit, computed, effect, inject, model, signal, viewChild } from '@angular/core'; import { ActivatedRoute, Router } from '@angular/router'; import { HttpErrorResponse, HttpResponse } from '@angular/common/http'; import { Observable, Subscription } from 'rxjs'; @@ -9,8 +9,7 @@ import { Course } from 'app/entities/course.model'; import { onError } from 'app/shared/util/global.utils'; import { ArtemisNavigationUtilService } from 'app/utils/navigation.utils'; import { DocumentationType } from 'app/shared/components/documentation-button/documentation-button.component'; -import { faBan, faHandshakeAngle, faPuzzlePiece, faQuestionCircle, faSave } from '@fortawesome/free-solid-svg-icons'; -import { LectureUpdateWizardComponent } from 'app/lecture/wizard-mode/lecture-update-wizard.component'; +import { faBan, faPuzzlePiece, faQuestionCircle, faSave } from '@fortawesome/free-solid-svg-icons'; import { ACCEPTED_FILE_EXTENSIONS_FILE_BROWSER, ALLOWED_FILE_EXTENSIONS_HUMAN_READABLE } from 'app/shared/constants/file-extensions.constants'; import { FormulaAction } from 'app/shared/monaco-editor/model/actions/formula.action'; import { LectureTitleChannelNameComponent } from './lecture-title-channel-name.component'; @@ -18,6 +17,9 @@ import { LectureUpdatePeriodComponent } from 'app/lecture/lecture-period/lecture import dayjs, { Dayjs } from 'dayjs/esm'; import { FormDateTimePickerComponent } from 'app/shared/date-time-picker/date-time-picker.component'; import cloneDeep from 'lodash-es/cloneDeep'; +import { FormSectionStatus, FormStatusBarComponent } from 'app/forms/form-status-bar/form-status-bar.component'; +import { LectureAttachmentsComponent } from 'app/lecture/lecture-attachments.component'; +import { LectureUpdateUnitsComponent } from 'app/lecture/lecture-units/lecture-units.component'; @Component({ selector: 'jhi-lecture-update', @@ -30,28 +32,31 @@ export class LectureUpdateComponent implements OnInit, OnDestroy { protected readonly faSave = faSave; protected readonly faPuzzleProcess = faPuzzlePiece; protected readonly faBan = faBan; - protected readonly faHandShakeAngle = faHandshakeAngle; protected readonly allowedFileExtensions = ALLOWED_FILE_EXTENSIONS_HUMAN_READABLE; protected readonly acceptedFileExtensionsFileBrowser = ACCEPTED_FILE_EXTENSIONS_FILE_BROWSER; - @ViewChild(LectureUpdateWizardComponent, { static: false }) wizardComponent: LectureUpdateWizardComponent; - private readonly alertService = inject(AlertService); private readonly lectureService = inject(LectureService); private readonly activatedRoute = inject(ActivatedRoute); private readonly navigationUtilService = inject(ArtemisNavigationUtilService); private readonly router = inject(Router); - titleSection = viewChild(LectureTitleChannelNameComponent); - lecturePeriodSection = viewChild(LectureUpdatePeriodComponent); + titleSection = viewChild.required(LectureTitleChannelNameComponent); + lecturePeriodSection = viewChild.required(LectureUpdatePeriodComponent); + attachmentsSection = viewChild(LectureAttachmentsComponent); + unitSection = viewChild(LectureUpdateUnitsComponent); + formStatusBar = viewChild(FormStatusBarComponent); + courseTitle = model(''); lecture = signal(new Lecture()); lectureOnInit: Lecture; + isEditMode = signal(false); isSaving: boolean; isProcessing: boolean; processUnitMode: boolean; - isShowingWizardMode: boolean; + + formStatusSections: FormSectionStatus[]; courses: Course[]; @@ -60,12 +65,20 @@ export class LectureUpdateComponent implements OnInit, OnDestroy { fileName: string; fileInputTouched = false; - toggleModeFunction = () => this.toggleWizardMode(); - saveLectureFunction = () => this.save(); + isNewlyCreatedExercise = false; isChangeMadeToTitleOrPeriodSection = false; shouldDisplayDismissWarning = true; + areSectionsValid = computed(() => { + return ( + this.titleSection().titleChannelNameComponent().isFormValidSignal() && + this.lecturePeriodSection().isPeriodSectionValid() && + (this.unitSection()?.isUnitConfigurationValid() ?? true) && + (this.attachmentsSection()?.isFormValid() ?? true) + ); + }); + private subscriptions = new Subscription(); constructor() { @@ -96,13 +109,25 @@ export class LectureUpdateComponent implements OnInit, OnDestroy { ); } }); + + effect(() => { + this.updateFormStatusBar(); + }); + + effect( + function scrollToLastSectionAfterLectureCreation() { + if (this.unitSection() && this.isNewlyCreatedExercise) { + this.isNewlyCreatedExercise = false; + this.formStatusBar()?.scrollToHeadline('artemisApp.lecture.sections.period'); + } + }.bind(this), + ); } ngOnInit() { this.isSaving = false; this.processUnitMode = false; this.isProcessing = false; - this.isShowingWizardMode = false; this.activatedRoute.parent!.data.subscribe((data) => { // Create a new lecture to use unless we fetch an existing lecture const lecture = data['lecture']; @@ -113,19 +138,45 @@ export class LectureUpdateComponent implements OnInit, OnDestroy { } }); - this.activatedRoute.queryParams.subscribe((params) => { - if (params.shouldBeInWizardMode) { - this.isShowingWizardMode = params.shouldBeInWizardMode; - } - }); - + this.isEditMode.set(!this.router.url.endsWith('/new')); this.lectureOnInit = cloneDeep(this.lecture()); + this.courseTitle.set(this.lecture().course?.title ?? ''); } ngOnDestroy() { this.subscriptions.unsubscribe(); } + updateFormStatusBar() { + const updatedFormStatusSections: FormSectionStatus[] = []; + + updatedFormStatusSections.push( + { + title: 'artemisApp.lecture.sections.title', + valid: Boolean(this.titleSection().titleChannelNameComponent().isFormValidSignal()), + }, + { + title: 'artemisApp.lecture.sections.period', + valid: Boolean(this.lecturePeriodSection().isPeriodSectionValid()), + }, + ); + + if (this.isEditMode()) { + updatedFormStatusSections.push( + { + title: 'artemisApp.lecture.sections.attachments', + valid: Boolean(this.attachmentsSection()?.isFormValid()), + }, + { + title: 'artemisApp.lecture.sections.units', + valid: Boolean(this.unitSection()?.isUnitConfigurationValid()), + }, + ); + } + + this.formStatusSections = updatedFormStatusSections; + } + isChangeMadeToTitleSection() { return ( this.lecture().title !== this.lectureOnInit.title || @@ -182,14 +233,6 @@ export class LectureUpdateComponent implements OnInit, OnDestroy { } } - /** - * Activate or deactivate the wizard mode for easier lecture creation. - * This function is called by pressing "Switch to guided mode" when creating a new lecture - */ - toggleWizardMode() { - this.isShowingWizardMode = !this.isShowingWizardMode; - } - proceedToUnitSplit() { this.save(); } @@ -227,25 +270,30 @@ export class LectureUpdateComponent implements OnInit, OnDestroy { * Action on successful lecture creation or edit */ protected onSaveSuccess(lecture: Lecture) { - if (this.isShowingWizardMode && !this.lecture().id) { - this.lectureService.findWithDetails(lecture.id!).subscribe({ - next: (response: HttpResponse) => { - this.isSaving = false; - this.lecture.set(response.body!); - this.alertService.success(`Lecture with title ${lecture.title} was successfully created.`); - this.wizardComponent.onLectureCreationSucceeded(); - }, - }); - } else if (this.processUnitMode) { - this.isSaving = false; + this.isSaving = false; + + if (!lecture.course?.id) { + console.error('Lecture has no course id', lecture); + return; + } + + if (this.processUnitMode) { this.isProcessing = false; this.alertService.success(`Lecture with title ${lecture.title} was successfully ${this.lecture().id !== undefined ? 'updated' : 'created'}.`); - this.router.navigate(['course-management', lecture.course!.id, 'lectures', lecture.id, 'unit-management', 'attachment-units', 'process'], { + this.router.navigate(['course-management', lecture.course.id, 'lectures', lecture.id, 'unit-management', 'attachment-units', 'process'], { state: { file: this.file, fileName: this.fileName }, }); + } else if (this.isEditMode()) { + this.router.navigate(['course-management', lecture.course.id, 'lectures', lecture.id]); } else { - this.isSaving = false; - this.router.navigate(['course-management', lecture.course!.id, 'lectures', lecture.id]); + // after create we stay on the edit page, as now attachments and lecture units are available (we need the lecture id to save them) + this.isNewlyCreatedExercise = true; + this.isEditMode.set(true); + this.lectureOnInit = cloneDeep(lecture); + this.lecture.set(lecture); + this.updateIsChangesMadeToTitleOrPeriodSection(); + window.history.replaceState({}, '', `course-management/${lecture.course.id}/lectures/${lecture.id}/edit`); + this.shouldDisplayDismissWarning = true; } } diff --git a/src/main/webapp/app/lecture/lecture.component.ts b/src/main/webapp/app/lecture/lecture.component.ts index 19c4dc417eb1..7bbe6d1b18b3 100644 --- a/src/main/webapp/app/lecture/lecture.component.ts +++ b/src/main/webapp/app/lecture/lecture.component.ts @@ -124,6 +124,12 @@ export class LectureComponent implements OnInit, OnDestroy { ); } + private deleteLectureFromDisplayedLectures(lectureId: number) { + this.dialogErrorSource.next(''); + this.lectures = this.lectures.filter((lecture) => lecture.id !== lectureId); + this.applyFilters(); + } + /** * Deletes Lecture * @param lectureId the id of the lecture @@ -131,9 +137,7 @@ export class LectureComponent implements OnInit, OnDestroy { deleteLecture(lectureId: number) { this.lectureService.delete(lectureId).subscribe({ next: () => { - this.dialogErrorSource.next(''); - this.lectures = this.lectures.filter((lecture) => lecture.id !== lectureId); - this.applyFilters(); + this.deleteLectureFromDisplayedLectures(lectureId); }, error: (error: HttpErrorResponse) => this.dialogErrorSource.next(error.message), }); diff --git a/src/main/webapp/app/lecture/lecture.module.ts b/src/main/webapp/app/lecture/lecture.module.ts index db1380abd69d..83afd92a0e56 100644 --- a/src/main/webapp/app/lecture/lecture.module.ts +++ b/src/main/webapp/app/lecture/lecture.module.ts @@ -1,12 +1,10 @@ import { NgModule } from '@angular/core'; import { RouterModule } from '@angular/router'; - import { ArtemisSharedModule } from 'app/shared/shared.module'; import { FormDateTimePickerModule } from 'app/shared/date-time-picker/date-time-picker.module'; import { ArtemisSharedComponentModule } from 'app/shared/components/shared-component.module'; import { ArtemisMarkdownEditorModule } from 'app/shared/markdown-editor/markdown-editor.module'; import { LectureUpdateComponent } from 'app/lecture/lecture-update.component'; -import { LectureUpdateWizardComponent } from 'app/lecture/wizard-mode/lecture-update-wizard.component'; import { LectureComponent } from 'app/lecture/lecture.component'; import { LectureDetailComponent } from 'app/lecture/lecture-detail.component'; import { LectureAttachmentsComponent } from 'app/lecture/lecture-attachments.component'; @@ -15,14 +13,12 @@ import { ArtemisLectureUnitManagementModule } from 'app/lecture/lecture-unit/lec import { ArtemisMarkdownModule } from 'app/shared/markdown.module'; import { LectureImportComponent } from 'app/lecture/lecture-import.component'; import { ArtemisCompetenciesModule } from 'app/course/competencies/competency.module'; -import { LectureUpdateWizardTitleComponent } from 'app/lecture/wizard-mode/lecture-wizard-title.component'; -import { LectureUpdateWizardAttachmentsComponent } from 'app/lecture/wizard-mode/lecture-wizard-attachments.component'; -import { LectureUpdateWizardUnitsComponent } from 'app/lecture/wizard-mode/lecture-wizard-units.component'; -import { LectureUpdateWizardStepComponent } from 'app/lecture/wizard-mode/lecture-update-wizard-step.component'; import { TitleChannelNameModule } from 'app/shared/form/title-channel-name/title-channel-name.module'; import { LectureTitleChannelNameComponent } from 'app/lecture/lecture-title-channel-name.component'; import { DetailModule } from 'app/detail-overview-list/detail.module'; import { CompetencyFormComponent } from 'app/course/competencies/forms/competency/competency-form.component'; +import { FormsModule } from 'app/forms/forms.module'; +import { LectureUpdateUnitsComponent } from 'app/lecture/lecture-units/lecture-units.component'; import { LectureUpdatePeriodComponent } from 'app/lecture/lecture-period/lecture-period.component'; const ENTITY_STATES = [...lectureRoute]; @@ -40,20 +36,17 @@ const ENTITY_STATES = [...lectureRoute]; TitleChannelNameModule, DetailModule, CompetencyFormComponent, + FormsModule, ], declarations: [ LectureComponent, LectureDetailComponent, LectureImportComponent, LectureUpdateComponent, - LectureUpdateWizardComponent, LectureAttachmentsComponent, - LectureUpdateWizardTitleComponent, - LectureUpdatePeriodComponent, - LectureUpdateWizardAttachmentsComponent, - LectureUpdateWizardUnitsComponent, - LectureUpdateWizardStepComponent, LectureTitleChannelNameComponent, + LectureUpdateUnitsComponent, + LectureUpdatePeriodComponent, ], }) export class ArtemisLectureModule {} diff --git a/src/main/webapp/app/lecture/lecture.service.ts b/src/main/webapp/app/lecture/lecture.service.ts index e6b508e6695a..118ef77172d8 100644 --- a/src/main/webapp/app/lecture/lecture.service.ts +++ b/src/main/webapp/app/lecture/lecture.service.ts @@ -112,6 +112,7 @@ export class LectureService { tap((res: EntityArrayResponseType) => res?.body?.forEach(this.sendTitlesToEntityTitleService.bind(this))), ); } + /** * triggers the ingestion of All the lectures inside the course specified or one lecture inside of the course * @@ -128,6 +129,7 @@ export class LectureService { observe: 'response', }); } + /** * Fetch the ingestion state of all the lectures inside the course specified * @param courseId diff --git a/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard-step.component.html b/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard-step.component.html deleted file mode 100644 index fdaee7117471..000000000000 --- a/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard-step.component.html +++ /dev/null @@ -1,16 +0,0 @@ -
    -
    - @if (isPerformed) { - - } - @if (!isPerformed) { - - } -
    -
    - Current step - @if (descriptionTranslationKey) { - - } -
    -
    diff --git a/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard-step.component.ts b/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard-step.component.ts deleted file mode 100644 index 9c8d6973a233..000000000000 --- a/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard-step.component.ts +++ /dev/null @@ -1,24 +0,0 @@ -import { Component, Input } from '@angular/core'; -import { faCheck, faDotCircle } from '@fortawesome/free-solid-svg-icons'; - -@Component({ - selector: 'jhi-lecture-update-wizard-step', - templateUrl: './lecture-update-wizard-step.component.html', - styleUrls: ['./lecture-update-wizard.component.scss'], -}) -export class LectureUpdateWizardStepComponent { - @Input() - currentlySelected: boolean; - - @Input() - isPerformed: boolean; - - @Input() - labelTranslationKey: string; - - @Input() - descriptionTranslationKey: string; - - protected readonly faCheck = faCheck; - protected readonly faDotCircle = faDotCircle; -} diff --git a/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.html b/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.html deleted file mode 100644 index 1e228e1c0747..000000000000 --- a/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.html +++ /dev/null @@ -1,54 +0,0 @@ -
    - @if (currentStep >= LECTURE_UPDATE_WIZARD_TITLE_STEP) { - - } - @if (currentStep >= LECTURE_UPDATE_WIZARD_PERIOD_STEP) { - - } - @if (currentStep >= LECTURE_UPDATE_WIZARD_ATTACHMENT_STEP) { - - } - @if (currentStep >= LECTURE_UPDATE_WIZARD_UNIT_STEP) { - - } - -
    -
    - -
    -
    -
    - -
    - -
    - -
    - -
    -
    -
    - -
    -
    -
    diff --git a/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.scss b/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.scss deleted file mode 100644 index 4c77040d9087..000000000000 --- a/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.scss +++ /dev/null @@ -1,70 +0,0 @@ -.form-step + .form-step::before { - content: ''; - display: block; - height: 1px; - background: gray; - margin-bottom: 1rem; -} - -.code-hint-generation-wrapper { - .planned, - .finished, - .check { - color: var(--success); - } - - .unset { - color: var(--secondary); - } - - .current { - color: var(--warning); - } - - li { - display: list-item !important; - } - - .connector { - width: 100%; - min-width: 50px; - border-top: 2px solid; - margin-top: 1em; - } -} - -.status-step { - width: 25px; - overflow-x: visible; - justify-content: flex-start; - display: flex; - flex-direction: column; - align-items: center; - - .status-step-content { - min-width: fit-content; - white-space: nowrap; - - .selected-label { - font-weight: bold; - } - } - - .header-icon { - z-index: 2; - font-size: 2em; - - & > .step-icon { - cursor: pointer; - background-color: var(--programming-exercise-instruction-step-wizard-card-header-background); - width: 30px; - height: 30px; - text-align: center; - padding: 6px 0; - font-size: 12px; - line-height: 1.428571429; - border-radius: 15px; - border-color: var(--programming-exercise-instruction-step-wizard-btn-border-color); - } - } -} diff --git a/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.ts b/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.ts deleted file mode 100644 index d360ffc962ca..000000000000 --- a/src/main/webapp/app/lecture/wizard-mode/lecture-update-wizard.component.ts +++ /dev/null @@ -1,104 +0,0 @@ -import { Component, Input, OnInit, ViewChild } from '@angular/core'; -import { ActivatedRoute, Router } from '@angular/router'; -import { CourseManagementService } from 'app/course/manage/course-management.service'; -import { Lecture } from 'app/entities/lecture.model'; -import { ArtemisNavigationUtilService } from 'app/utils/navigation.utils'; -import { faArrowRight, faCheck, faHandshakeAngle } from '@fortawesome/free-solid-svg-icons'; -import { LectureUpdateWizardUnitsComponent } from 'app/lecture/wizard-mode/lecture-wizard-units.component'; -import { take } from 'rxjs/operators'; - -@Component({ - selector: 'jhi-lecture-update-wizard', - templateUrl: './lecture-update-wizard.component.html', - styleUrls: ['./lecture-update-wizard.component.scss'], -}) -export class LectureUpdateWizardComponent implements OnInit { - @Input() toggleModeFunction: () => void; - @Input() saveLectureFunction: () => void; - @Input() validateDatesFunction: () => void; - @Input() lecture: Lecture; - @Input() isSaving: boolean; - - @ViewChild(LectureUpdateWizardUnitsComponent, { static: false }) unitsComponent: LectureUpdateWizardUnitsComponent; - - readonly LECTURE_UPDATE_WIZARD_TITLE_STEP = 1; - readonly LECTURE_UPDATE_WIZARD_PERIOD_STEP = 2; - readonly LECTURE_UPDATE_WIZARD_ATTACHMENT_STEP = 3; - readonly LECTURE_UPDATE_WIZARD_UNIT_STEP = 4; - - currentStep: number; - - // Icons - faCheck = faCheck; - faHandShakeAngle = faHandshakeAngle; - faArrowRight = faArrowRight; - - constructor( - protected courseService: CourseManagementService, - protected activatedRoute: ActivatedRoute, - private navigationUtilService: ArtemisNavigationUtilService, - private router: Router, - ) {} - - /** - * Life cycle hook called by Angular to indicate that Angular is done creating the component - */ - ngOnInit() { - this.isSaving = false; - - this.activatedRoute.queryParams.pipe(take(1)).subscribe((params) => { - if (params.step && !isNaN(+params.step)) { - this.currentStep = +params.step; - } else { - if (this.lecture.id) { - this.currentStep = this.LECTURE_UPDATE_WIZARD_UNIT_STEP; - } else if (this.lecture.startDate === undefined && this.lecture.endDate === undefined) { - this.currentStep = this.LECTURE_UPDATE_WIZARD_TITLE_STEP; - } else if (!this.lecture.id) { - this.currentStep = this.LECTURE_UPDATE_WIZARD_PERIOD_STEP; - } - } - - this.router.navigate([], { - relativeTo: this.activatedRoute, - queryParamsHandling: '', - replaceUrl: true, - }); - }); - } - - /** - * Progress to the next step of the wizard mode - */ - next() { - if (this.currentStep === this.LECTURE_UPDATE_WIZARD_PERIOD_STEP || this.currentStep === this.LECTURE_UPDATE_WIZARD_UNIT_STEP) { - this.saveLectureFunction(); - return; - } - - this.currentStep++; - } - - /** - * Called when the lecture has been successfully created in the parent component to advance in the wizard - */ - onLectureCreationSucceeded() { - this.currentStep++; - } - - getNextIcon() { - return this.currentStep < this.LECTURE_UPDATE_WIZARD_UNIT_STEP ? faArrowRight : faCheck; - } - - getNextText() { - return this.currentStep < this.LECTURE_UPDATE_WIZARD_UNIT_STEP ? 'artemisApp.lecture.home.nextStepLabel' : 'entity.action.finish'; - } - - toggleWizardMode() { - if (this.currentStep <= this.LECTURE_UPDATE_WIZARD_PERIOD_STEP) { - this.toggleModeFunction(); - } else { - this.router.navigate(['course-management', this.lecture.course!.id, 'lectures', this.lecture.id]); - } - } -} diff --git a/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-attachments.component.html b/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-attachments.component.html deleted file mode 100644 index 5b58c99bb28f..000000000000 --- a/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-attachments.component.html +++ /dev/null @@ -1,7 +0,0 @@ -
    -

    -

    - @if (currentStep >= 3) { - - } -
    diff --git a/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-attachments.component.ts b/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-attachments.component.ts deleted file mode 100644 index 867ad557e393..000000000000 --- a/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-attachments.component.ts +++ /dev/null @@ -1,13 +0,0 @@ -import { Component, Input } from '@angular/core'; -import { Lecture } from 'app/entities/lecture.model'; - -@Component({ - selector: 'jhi-lecture-update-wizard-attachments', - templateUrl: './lecture-wizard-attachments.component.html', -}) -export class LectureUpdateWizardAttachmentsComponent { - @Input() currentStep: number; - @Input() lecture: Lecture; - - constructor() {} -} diff --git a/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-title.component.html b/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-title.component.html deleted file mode 100644 index 9100d8436184..000000000000 --- a/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-title.component.html +++ /dev/null @@ -1,11 +0,0 @@ -
    -

    -

    -
    - - -
    - - -
    -
    diff --git a/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-title.component.ts b/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-title.component.ts deleted file mode 100644 index e354df72e629..000000000000 --- a/src/main/webapp/app/lecture/wizard-mode/lecture-wizard-title.component.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { Component, Input } from '@angular/core'; -import { Lecture } from 'app/entities/lecture.model'; -import { FormulaAction } from 'app/shared/monaco-editor/model/actions/formula.action'; - -@Component({ - selector: 'jhi-lecture-update-wizard-title', - templateUrl: './lecture-wizard-title.component.html', -}) -export class LectureUpdateWizardTitleComponent { - @Input() currentStep: number; - @Input() lecture: Lecture; - - domainActionsDescription = [new FormulaAction()]; -} diff --git a/src/main/webapp/app/shared/exercise-filter/exercise-filter-modal.component.html b/src/main/webapp/app/shared/exercise-filter/exercise-filter-modal.component.html index 1a87237a4d46..218a925805ca 100644 --- a/src/main/webapp/app/shared/exercise-filter/exercise-filter-modal.component.html +++ b/src/main/webapp/app/shared/exercise-filter/exercise-filter-modal.component.html @@ -70,11 +70,6 @@
    } - - - - - @if (difficultyFilter?.isDisplayed && difficultyFilter) { diff --git a/src/main/webapp/i18n/de/lecture.json b/src/main/webapp/i18n/de/lecture.json index 09357210a6cb..2415d32b6f19 100644 --- a/src/main/webapp/i18n/de/lecture.json +++ b/src/main/webapp/i18n/de/lecture.json @@ -4,6 +4,7 @@ "home": { "title": "Vorlesungen", "createLabel": "Vorlesung erstellen", + "editLabel": "Vorlesung bearbeiten", "filterLabel": "Filter", "filterOptions": { "filterPast": "Vergangene", @@ -12,9 +13,6 @@ "filterUnspecifiedDates": "Ohne Zeitangaben" }, "createOrEditLabel": "Vorlesung erstellen oder bearbeiten", - "switchToGuidedModeLabel": "Zum geführten Modus wechseln", - "switchToTraditionalModeLabel": "Zum normalen Modus wechseln", - "nextStepLabel": "Weiter", "ingestLecturesInPyrisLabel": "Vorlesungen an Iris senden" }, "created": "Vorlesung erstellt mit ID {{ param }}", @@ -71,24 +69,18 @@ "loading": "Lade...", "label": "Vorlesung importieren" }, - "wizardMode": { - "steps": { - "titleStepTitle": "Titel", - "titleStepMessage": "Gib einen Titel und eine aussagekräftige Beschreibung für die neue Vorlesung ein.", - "periodStepTitle": "Zeitraum", - "periodStepMessage": "Lege das Start- und Enddatum der Vorlesung fest.", - "attachmentsStepTitle": "Anhänge", - "attachmentsStepMessage": "Lade Anhänge für die Vorlesung hoch.", - "unitsStepTitle": "Vorlesungseinheiten", - "unitsStepMessage": "Füge Inhalte zur Vorlesung hinzu, indem du Vorlesungseinheiten erstellst." - }, - "newLectureUnit": "Neue Vorlesungseinheit", - "editLectureUnit": "Vorlesungseinheit bearbeiten", - "newExercise": "Neue Übung erstellen", - "competencyTitle": "Titel", - "competencyConnectedUnits": "Verknüpfte Einheiten", - "competencyNoConnectedUnits": "Keine verknüpften Einheiten" + "sections": { + "title": "Titel", + "titleDescription": "Gib einen Titel und eine aussagekräftige Beschreibung für die Vorlesung ein.", + "period": "Zeitraum", + "periodDescription": "Lege das Start- und Enddatum der Vorlesung fest.", + "attachments": "Anhänge", + "attachmentsDescription": "Lade Anhänge für die Vorlesung hoch.", + "units": "Vorlesungseinheiten", + "unitsDescription": "Füge Inhalte zur Vorlesung hinzu, indem du Vorlesungseinheiten erstellst." }, + "newLectureUnit": "Neue Vorlesungseinheit", + "editLectureUnit": "Vorlesungseinheit bearbeiten", "dismissChangesModal": { "title": "Ungespeicherte Änderungen der Vorlesung verwerfen?", "message": "Bist du sicher, dass du die ungespeicherten Änderungen verwerfen willst?", diff --git a/src/main/webapp/i18n/en/lecture.json b/src/main/webapp/i18n/en/lecture.json index 9c07a5530c0f..cf23b9cea407 100644 --- a/src/main/webapp/i18n/en/lecture.json +++ b/src/main/webapp/i18n/en/lecture.json @@ -4,6 +4,7 @@ "home": { "title": "Lectures", "createLabel": "Create Lecture", + "editLabel": "Edit Lecture", "filterLabel": "Filter", "filterOptions": { "filterPast": "Past", @@ -12,9 +13,6 @@ "filterUnspecifiedDates": "Unspecified Date(s)" }, "createOrEditLabel": "Create or Edit Lecture", - "switchToGuidedModeLabel": "Switch to guided mode", - "switchToTraditionalModeLabel": "Switch to normal mode", - "nextStepLabel": "Next", "ingestLecturesInPyrisLabel": "Send Lectures to Iris" }, "created": "Created new lecture with identifier {{ param }}", @@ -71,24 +69,18 @@ "loading": "Loading...", "label": "Import Lecture" }, - "wizardMode": { - "steps": { - "titleStepTitle": "Title", - "titleStepMessage": "Add a title and meaningful description to the new lecture.", - "periodStepTitle": "Period", - "periodStepMessage": "Specify the begin and end dates of the lecture.", - "attachmentsStepTitle": "Attachments", - "attachmentsStepMessage": "Upload attachments to this lecture.", - "unitsStepTitle": "Units", - "unitsStepMessage": "Add content to the lecture by creating different kinds of lecture units." - }, - "newLectureUnit": "New Lecture Unit", - "editLectureUnit": "Edit Lecture Unit", - "newExercise": "Create new exercise", - "competencyTitle": "Title", - "competencyConnectedUnits": "Connected Units", - "competencyNoConnectedUnits": "No connected units" + "sections": { + "title": "Title", + "titleDescription": "Add a title and meaningful description to the lecture.", + "period": "Period", + "periodDescription": "Specify the begin and end dates of the lecture.", + "attachments": "Attachments", + "attachmentsDescription": "Upload attachments to this lecture.", + "units": "Units", + "unitsDescription": "Add content to the lecture by creating different kinds of lecture units." }, + "newLectureUnit": "New Lecture Unit", + "editLectureUnit": "Edit Lecture Unit", "dismissChangesModal": { "title": "Discard unsaved lecture changes", "message": "Are you sure you want to discard your unsaved changes?", diff --git a/src/test/javascript/spec/component/lecture/close-edit-lecture-modal.component.spec.ts b/src/test/javascript/spec/component/lecture/close-edit-lecture-modal.component.spec.ts index 93b92f1d9627..e35dfed6baa2 100644 --- a/src/test/javascript/spec/component/lecture/close-edit-lecture-modal.component.spec.ts +++ b/src/test/javascript/spec/component/lecture/close-edit-lecture-modal.component.spec.ts @@ -1,6 +1,6 @@ import { ComponentFixture, TestBed } from '@angular/core/testing'; import { ArtemisTestModule } from '../../test.module'; -import { CloseEditLectureModalComponent } from '../../../../../main/webapp/app/lecture/close-edit-lecture-dialog/close-edit-lecture-modal.component'; +import { CloseEditLectureModalComponent } from '../../../../../main/webapp/app/lecture/close-edit-lecture-modal/close-edit-lecture-modal.component'; describe('CloseEditLectureModalComponent', () => { let component: CloseEditLectureModalComponent; diff --git a/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-units.component.spec.ts b/src/test/javascript/spec/component/lecture/lecture-units.component.spec.ts similarity index 98% rename from src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-units.component.spec.ts rename to src/test/javascript/spec/component/lecture/lecture-units.component.spec.ts index c850a83f6ecc..fd8a0190e1f9 100644 --- a/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-units.component.spec.ts +++ b/src/test/javascript/spec/component/lecture/lecture-units.component.spec.ts @@ -5,13 +5,12 @@ import { VideoUnitService } from 'app/lecture/lecture-unit/lecture-unit-manageme import { MockComponent, MockProvider } from 'ng-mocks'; import { AlertService } from 'app/core/util/alert.service'; import { ActivatedRoute, Router } from '@angular/router'; -import { MockRouter } from '../../../helpers/mocks/mock-router'; +import { MockRouter } from '../../helpers/mocks/mock-router'; import { of, throwError } from 'rxjs'; import { VideoUnit } from 'app/entities/lecture-unit/videoUnit.model'; import dayjs from 'dayjs/esm'; import { HttpResponse } from '@angular/common/http'; import { By } from '@angular/platform-browser'; -import { LectureUpdateWizardUnitsComponent } from 'app/lecture/wizard-mode/lecture-wizard-units.component'; import { Lecture } from 'app/entities/lecture.model'; import { TextUnitService } from 'app/lecture/lecture-unit/lecture-unit-management/textUnit.service'; import { OnlineUnitService } from 'app/lecture/lecture-unit/lecture-unit-management/onlineUnit.service'; @@ -27,7 +26,8 @@ import { Attachment, AttachmentType } from 'app/entities/attachment.model'; import { AttachmentUnit } from 'app/entities/lecture-unit/attachmentUnit.model'; import { objectToJsonBlob } from 'app/utils/blob-util'; import { CreateExerciseUnitComponent } from 'app/lecture/lecture-unit/lecture-unit-management/create-exercise-unit/create-exercise-unit.component'; -import { CompetencyLectureUnitLink } from '../../../../../../main/webapp/app/entities/competency.model'; +import { LectureUpdateUnitsComponent } from 'app/lecture/lecture-units/lecture-units.component'; +import { CompetencyLectureUnitLink } from 'app/entities/competency.model'; @Component({ selector: 'jhi-video-unit-form', template: '' }) class VideoUnitFormStubComponent { @@ -41,9 +41,9 @@ class UnitCreationCardStubComponent { @Output() onUnitCreationCardClicked: EventEmitter = new EventEmitter(); } -describe('LectureWizardUnitComponent', () => { - let wizardUnitComponentFixture: ComponentFixture; - let wizardUnitComponent: LectureUpdateWizardUnitsComponent; +describe('LectureUpdateUnitsComponent', () => { + let wizardUnitComponentFixture: ComponentFixture; + let wizardUnitComponent: LectureUpdateUnitsComponent; beforeEach(() => { TestBed.configureTestingModule({ @@ -51,7 +51,7 @@ describe('LectureWizardUnitComponent', () => { declarations: [ VideoUnitFormStubComponent, UnitCreationCardStubComponent, - LectureUpdateWizardUnitsComponent, + LectureUpdateUnitsComponent, MockComponent(CreateExerciseUnitComponent), MockComponent(LectureUnitManagementComponent), ], @@ -72,7 +72,7 @@ describe('LectureWizardUnitComponent', () => { }) .compileComponents() .then(() => { - wizardUnitComponentFixture = TestBed.createComponent(LectureUpdateWizardUnitsComponent); + wizardUnitComponentFixture = TestBed.createComponent(LectureUpdateUnitsComponent); wizardUnitComponent = wizardUnitComponentFixture.componentInstance; wizardUnitComponent.lecture = new Lecture(); wizardUnitComponent.lecture.id = 1; diff --git a/src/test/javascript/spec/component/lecture/lecture-update.component.spec.ts b/src/test/javascript/spec/component/lecture/lecture-update.component.spec.ts index 384ec2002b9c..754da6e66712 100644 --- a/src/test/javascript/spec/component/lecture/lecture-update.component.spec.ts +++ b/src/test/javascript/spec/component/lecture/lecture-update.component.spec.ts @@ -8,7 +8,6 @@ import { TranslateService } from '@ngx-translate/core'; import { Lecture } from 'app/entities/lecture.model'; import { LectureUpdateComponent } from 'app/lecture/lecture-update.component'; import { LectureService } from 'app/lecture/lecture.service'; -import { LectureUpdateWizardComponent } from 'app/lecture/wizard-mode/lecture-update-wizard.component'; import { FormDateTimePickerComponent } from 'app/shared/date-time-picker/date-time-picker.component'; import { ArtemisDatePipe } from 'app/shared/pipes/artemis-date.pipe'; import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; @@ -23,16 +22,19 @@ import { ArtemisTestModule } from '../../test.module'; import { DocumentationButtonComponent } from 'app/shared/components/documentation-button/documentation-button.component'; import { LectureTitleChannelNameComponent } from 'app/lecture/lecture-title-channel-name.component'; import { MarkdownEditorMonacoComponent } from 'app/shared/markdown-editor/monaco/markdown-editor-monaco.component'; -import { CustomNotIncludedInValidatorDirective } from '../../../../../main/webapp/app/shared/validators/custom-not-included-in-validator.directive'; +import { CustomNotIncludedInValidatorDirective } from 'app/shared/validators/custom-not-included-in-validator.directive'; import { OwlDateTimeModule } from '@danielmoncada/angular-datetime-picker'; -import { TitleChannelNameComponent } from '../../../../../main/webapp/app/shared/form/title-channel-name/title-channel-name.component'; -import { LectureUpdatePeriodComponent } from '../../../../../main/webapp/app/lecture/lecture-period/lecture-period.component'; -import { LectureUnitManagementComponent } from '../../../../../main/webapp/app/lecture/lecture-unit/lecture-unit-management/lecture-unit-management.component'; +import { TitleChannelNameComponent } from 'app/shared/form/title-channel-name/title-channel-name.component'; +import { LectureUpdatePeriodComponent } from 'app/lecture/lecture-period/lecture-period.component'; +import { LectureUnitManagementComponent } from 'app/lecture/lecture-unit/lecture-unit-management/lecture-unit-management.component'; +import { FormStatusBarComponent } from 'app/forms/form-status-bar/form-status-bar.component'; +import { ArtemisSharedModule } from 'app/shared/shared.module'; +import { LectureAttachmentsComponent } from 'app/lecture/lecture-attachments.component'; +import { LectureUpdateUnitsComponent } from 'app/lecture/lecture-units/lecture-units.component'; +import { UnitCreationCardComponent } from 'app/lecture/lecture-unit/lecture-unit-management/unit-creation-card/unit-creation-card.component'; +import { signal } from '@angular/core'; describe('LectureUpdateComponent', () => { - let lectureUpdateWizardComponentFixture: ComponentFixture; - let lectureUpdateWizardComponent: LectureUpdateWizardComponent; - let lectureService: LectureService; let lectureUpdateComponentFixture: ComponentFixture; let lectureUpdateComponent: LectureUpdateComponent; @@ -50,21 +52,24 @@ describe('LectureUpdateComponent', () => { pastLecture.endDate = yesterday; TestBed.configureTestingModule({ - imports: [ArtemisTestModule, FormsModule, MockModule(NgbTooltipModule), MockModule(OwlDateTimeModule)], + imports: [ArtemisTestModule, MockModule(ArtemisSharedModule), FormsModule, MockModule(NgbTooltipModule), MockModule(OwlDateTimeModule)], declarations: [ LectureUpdateComponent, LectureTitleChannelNameComponent, TitleChannelNameComponent, FormDateTimePickerComponent, + LectureAttachmentsComponent, + LectureUpdateUnitsComponent, LectureUpdatePeriodComponent, - MockComponent(LectureUpdateWizardComponent), MockComponent(LectureUnitManagementComponent), + MockComponent(FormStatusBarComponent), MockComponent(MarkdownEditorMonacoComponent), MockComponent(DocumentationButtonComponent), MockPipe(ArtemisTranslatePipe), MockPipe(ArtemisDatePipe), MockPipe(HtmlForMarkdownPipe), MockRouterLinkDirective, + MockComponent(UnitCreationCardComponent), MockDirective(CustomNotIncludedInValidatorDirective), ], providers: [ @@ -93,9 +98,6 @@ describe('LectureUpdateComponent', () => { lectureUpdateComponentFixture = TestBed.createComponent(LectureUpdateComponent); lectureUpdateComponent = lectureUpdateComponentFixture.componentInstance; - lectureUpdateWizardComponentFixture = TestBed.createComponent(LectureUpdateWizardComponent); - lectureUpdateWizardComponent = lectureUpdateWizardComponentFixture.componentInstance; - lectureService = TestBed.inject(LectureService); router = TestBed.get(Router); activatedRoute = TestBed.inject(ActivatedRoute); @@ -108,7 +110,6 @@ describe('LectureUpdateComponent', () => { it('should create lecture', () => { lectureUpdateComponent.lecture.set({ title: 'test1', channelName: 'test1' } as Lecture); - const navigateSpy = jest.spyOn(router, 'navigate'); const createSpy = jest.spyOn(lectureService, 'create').mockReturnValue( of( @@ -127,58 +128,13 @@ describe('LectureUpdateComponent', () => { lectureUpdateComponent.save(); lectureUpdateComponentFixture.detectChanges(); - const expectedPath = ['course-management', 1, 'lectures', 3]; - expect(navigateSpy).toHaveBeenCalledWith(expectedPath); - expect(createSpy).toHaveBeenCalledOnce(); expect(createSpy).toHaveBeenCalledWith({ title: 'test1', channelName: 'test1' }); }); - it('should create lecture in wizard mode', () => { - lectureUpdateComponent.lecture.set({ title: '', channelName: '' } as Lecture); - lectureUpdateComponent.isShowingWizardMode = true; - lectureUpdateComponent.wizardComponent = lectureUpdateWizardComponent; - - const createSpy = jest.spyOn(lectureService, 'create').mockReturnValue( - of>( - new HttpResponse({ - body: { - title: 'test1', - course: { - id: 1, - }, - } as Lecture, - }), - ), - ); - - const findSpy = jest.spyOn(lectureService, 'findWithDetails').mockReturnValue( - of>( - new HttpResponse({ - body: { - id: 3, - title: 'test1', - course: { - id: 1, - }, - } as Lecture, - }), - ), - ); - - const onLectureCreationSucceededSpy = jest.spyOn(lectureUpdateWizardComponent, 'onLectureCreationSucceeded'); - - lectureUpdateComponent.save(); - - expect(createSpy).toHaveBeenCalledOnce(); - expect(createSpy).toHaveBeenCalledWith({ title: '', channelName: '' }); - - expect(findSpy).toHaveBeenCalledOnce(); - expect(onLectureCreationSucceededSpy).toHaveBeenCalledOnce(); - }); - it('should edit a lecture', fakeAsync(() => { activatedRoute.parent!.data = of({ course: { id: 1 }, lecture: { id: 6 } }); + const navigateSpy = jest.spyOn(router, 'navigate'); lectureUpdateComponentFixture.detectChanges(); lectureUpdateComponent.lecture.set({ id: 6, title: 'test1Updated', channelName: 'test1Updated' } as Lecture); @@ -201,32 +157,13 @@ describe('LectureUpdateComponent', () => { tick(); lectureUpdateComponentFixture.detectChanges(); + const expectedPath = ['course-management', 1, 'lectures', 6]; + expect(navigateSpy).toHaveBeenCalledWith(expectedPath); + expect(updateSpy).toHaveBeenCalledOnce(); expect(updateSpy).toHaveBeenCalledWith({ id: 6, title: 'test1Updated', channelName: 'test1Updated' }); })); - it('should switch to wizard mode', fakeAsync(() => { - lectureUpdateComponent.isShowingWizardMode = false; - const wizardModeButton = jest.spyOn(lectureUpdateComponent, 'toggleWizardMode'); - lectureUpdateComponent.toggleWizardMode(); - tick(); - expect(wizardModeButton).toHaveBeenCalledOnce(); - expect(lectureUpdateComponent.isShowingWizardMode).toBeTrue(); - })); - - it('should be in wizard mode', fakeAsync(() => { - activatedRoute = TestBed.inject(ActivatedRoute); - activatedRoute.queryParams = of({ - shouldBeInWizardMode: true, - }); - - lectureUpdateComponent.ngOnInit(); - lectureUpdateComponentFixture.detectChanges(); - tick(); - - expect(lectureUpdateComponent.isShowingWizardMode).toBeTrue(); - })); - it('should select process units checkbox', fakeAsync(() => { lectureUpdateComponent.processUnitMode = false; const selectProcessUnit = jest.spyOn(lectureUpdateComponent, 'onSelectProcessUnit'); @@ -416,4 +353,79 @@ describe('LectureUpdateComponent', () => { expect(lectureUpdateComponent.isChangeMadeToPeriodSection()).toBeFalse(); }); }); + + describe('updateFormStatusBar', () => { + it('should update form status bar correctly in edit mode', () => { + lectureUpdateComponent.isEditMode.set(true); + lectureUpdateComponent.titleSection = signal({ + titleChannelNameComponent: () => ({ + isFormValidSignal: () => true, + }), + } as any); + lectureUpdateComponent.lecturePeriodSection = signal({ + isPeriodSectionValid: () => true, + } as any); + lectureUpdateComponent.attachmentsSection = signal({ + isFormValid: () => true, + } as any); + lectureUpdateComponent.unitSection = signal({ + isUnitConfigurationValid: () => true, + } as any); + + lectureUpdateComponent.updateFormStatusBar(); + + expect(lectureUpdateComponent.formStatusSections).toEqual([ + { title: 'artemisApp.lecture.sections.title', valid: true }, + { title: 'artemisApp.lecture.sections.period', valid: true }, + { title: 'artemisApp.lecture.sections.attachments', valid: true }, + { title: 'artemisApp.lecture.sections.units', valid: true }, + ]); + }); + + it('should update form status bar correctly in create mode', () => { + lectureUpdateComponent.isEditMode.set(false); + lectureUpdateComponent.titleSection = signal({ + titleChannelNameComponent: () => ({ + isFormValidSignal: () => false, + }), + } as any); + lectureUpdateComponent.lecturePeriodSection = signal({ + isPeriodSectionValid: () => true, + } as any); + + lectureUpdateComponent.updateFormStatusBar(); + + expect(lectureUpdateComponent.formStatusSections).toEqual([ + { title: 'artemisApp.lecture.sections.title', valid: false }, + { title: 'artemisApp.lecture.sections.period', valid: true }, + ]); + }); + + it('should handle invalid sections correctly', () => { + lectureUpdateComponent.isEditMode.set(true); + lectureUpdateComponent.titleSection = signal({ + titleChannelNameComponent: () => ({ + isFormValidSignal: () => false, + }), + } as any); + lectureUpdateComponent.lecturePeriodSection = signal({ + isPeriodSectionValid: () => false, + } as any); + lectureUpdateComponent.attachmentsSection = signal({ + isFormValid: () => false, + } as any); + lectureUpdateComponent.unitSection = signal({ + isUnitConfigurationValid: () => false, + } as any); + + lectureUpdateComponent.updateFormStatusBar(); + + expect(lectureUpdateComponent.formStatusSections).toEqual([ + { title: 'artemisApp.lecture.sections.title', valid: false }, + { title: 'artemisApp.lecture.sections.period', valid: false }, + { title: 'artemisApp.lecture.sections.attachments', valid: false }, + { title: 'artemisApp.lecture.sections.units', valid: false }, + ]); + }); + }); }); diff --git a/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-attachments.component.spec.ts b/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-attachments.component.spec.ts deleted file mode 100644 index 0e3f9909ad52..000000000000 --- a/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-attachments.component.spec.ts +++ /dev/null @@ -1,34 +0,0 @@ -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { MockPipe } from 'ng-mocks'; -import { Lecture } from 'app/entities/lecture.model'; -import { ArtemisTranslatePipe } from 'app/shared/pipes/artemis-translate.pipe'; -import { LectureUpdateWizardAttachmentsComponent } from 'app/lecture/wizard-mode/lecture-wizard-attachments.component'; - -describe('LectureWizardAttachmentsComponent', () => { - let wizardAttachmentsComponentFixture: ComponentFixture; - let wizardAttachmentsComponent: LectureUpdateWizardAttachmentsComponent; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [], - declarations: [LectureUpdateWizardAttachmentsComponent, MockPipe(ArtemisTranslatePipe)], - providers: [], - schemas: [], - }) - .compileComponents() - .then(() => { - wizardAttachmentsComponentFixture = TestBed.createComponent(LectureUpdateWizardAttachmentsComponent); - wizardAttachmentsComponent = wizardAttachmentsComponentFixture.componentInstance; - wizardAttachmentsComponent.lecture = new Lecture(); - }); - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should initialize', () => { - wizardAttachmentsComponentFixture.detectChanges(); - expect(wizardAttachmentsComponent).not.toBeNull(); - }); -}); diff --git a/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-title.component.spec.ts b/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-title.component.spec.ts deleted file mode 100644 index de4b7fc35dc7..000000000000 --- a/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard-title.component.spec.ts +++ /dev/null @@ -1,44 +0,0 @@ -import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { LectureUpdateWizardTitleComponent } from 'app/lecture/wizard-mode/lecture-wizard-title.component'; -import { Lecture } from 'app/entities/lecture.model'; -import { MockComponent, MockDirective, MockModule } from 'ng-mocks'; -import { FormsModule } from '@angular/forms'; -import { MarkdownEditorMonacoComponent } from '../../../../../../main/webapp/app/shared/markdown-editor/monaco/markdown-editor-monaco.component'; -import { LectureTitleChannelNameComponent } from '../../../../../../main/webapp/app/lecture/lecture-title-channel-name.component'; -import { CustomNotIncludedInValidatorDirective } from '../../../../../../main/webapp/app/shared/validators/custom-not-included-in-validator.directive'; -import { TitleChannelNameComponent } from '../../../../../../main/webapp/app/shared/form/title-channel-name/title-channel-name.component'; - -describe('LectureWizardTitleComponent', () => { - let wizardTitleComponentFixture: ComponentFixture; - let wizardTitleComponent: LectureUpdateWizardTitleComponent; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [MockModule(FormsModule)], - declarations: [ - LectureUpdateWizardTitleComponent, - LectureTitleChannelNameComponent, - TitleChannelNameComponent, - MockComponent(MarkdownEditorMonacoComponent), - MockDirective(CustomNotIncludedInValidatorDirective), - ], - providers: [], - schemas: [], - }) - .compileComponents() - .then(() => { - wizardTitleComponentFixture = TestBed.createComponent(LectureUpdateWizardTitleComponent); - wizardTitleComponent = wizardTitleComponentFixture.componentInstance; - wizardTitleComponent.lecture = new Lecture(); - }); - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should initialize', () => { - wizardTitleComponentFixture.detectChanges(); - expect(wizardTitleComponent).not.toBeNull(); - }); -}); diff --git a/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard.component.spec.ts b/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard.component.spec.ts deleted file mode 100644 index e01eea78ab89..000000000000 --- a/src/test/javascript/spec/component/lecture/wizard-mode/lecture-wizard.component.spec.ts +++ /dev/null @@ -1,218 +0,0 @@ -import { ComponentFixture, TestBed, fakeAsync, tick } from '@angular/core/testing'; -import { MockComponent, MockModule, MockProvider } from 'ng-mocks'; -import { ActivatedRoute, Router } from '@angular/router'; -import { MockRouter } from '../../../helpers/mocks/mock-router'; -import { of } from 'rxjs'; -import { Lecture } from 'app/entities/lecture.model'; -import { MockTranslateService } from '../../../helpers/mocks/service/mock-translate.service'; -import { TranslateService } from '@ngx-translate/core'; -import { Course } from 'app/entities/course.model'; -import { LectureUpdateWizardComponent } from 'app/lecture/wizard-mode/lecture-update-wizard.component'; -import { ArtemisNavigationUtilService } from 'app/utils/navigation.utils'; -import { CourseManagementService } from 'app/course/manage/course-management.service'; -import { LectureUpdateWizardStepComponent } from 'app/lecture/wizard-mode/lecture-update-wizard-step.component'; -import { LectureUpdateWizardUnitsComponent } from 'app/lecture/wizard-mode/lecture-wizard-units.component'; -import { LectureUpdateWizardAttachmentsComponent } from 'app/lecture/wizard-mode/lecture-wizard-attachments.component'; -import { LectureUpdateWizardTitleComponent } from 'app/lecture/wizard-mode/lecture-wizard-title.component'; -import dayjs from 'dayjs/esm'; -import { LectureUpdatePeriodComponent } from '../../../../../../main/webapp/app/lecture/lecture-period/lecture-period.component'; -import { ArtemisTestModule } from '../../../test.module'; -import { ArtemisSharedModule } from '../../../../../../main/webapp/app/shared/shared.module'; -import { FormDateTimePickerComponent } from '../../../../../../main/webapp/app/shared/date-time-picker/date-time-picker.component'; - -describe('LectureWizardComponent', () => { - let wizardComponentFixture: ComponentFixture; - let wizardComponent: LectureUpdateWizardComponent; - - beforeEach(() => { - TestBed.configureTestingModule({ - imports: [ArtemisTestModule, MockModule(ArtemisSharedModule)], - declarations: [ - LectureUpdateWizardComponent, - LectureUpdatePeriodComponent, - MockComponent(FormDateTimePickerComponent), - MockComponent(LectureUpdateWizardTitleComponent), - MockComponent(LectureUpdateWizardStepComponent), - MockComponent(LectureUpdateWizardUnitsComponent), - MockComponent(LectureUpdateWizardAttachmentsComponent), - ], - providers: [ - MockProvider(ArtemisNavigationUtilService), - MockProvider(CourseManagementService), - { provide: TranslateService, useClass: MockTranslateService }, - { provide: Router, useClass: MockRouter }, - { - provide: ActivatedRoute, - useValue: { queryParams: of({}) }, - }, - ], - schemas: [], - }) - .compileComponents() - .then(() => { - wizardComponentFixture = TestBed.createComponent(LectureUpdateWizardComponent); - wizardComponent = wizardComponentFixture.componentInstance; - - const course = new Course(); - course.id = 2; - - wizardComponent.lecture = new Lecture(); - wizardComponent.lecture.id = 1; - wizardComponent.lecture.course = course; - }); - }); - - afterEach(() => { - jest.restoreAllMocks(); - }); - - it('should initialize and set step with given lecture', fakeAsync(() => { - wizardComponentFixture.detectChanges(); - expect(wizardComponent).not.toBeNull(); - - wizardComponentFixture.whenStable().then(() => { - expect(wizardComponent.currentStep).toBe(4); - }); - })); - - it('should initialize and set step without given lecture', fakeAsync(() => { - wizardComponent.lecture.id = undefined; - - wizardComponentFixture.detectChanges(); - expect(wizardComponent).not.toBeNull(); - - tick(); - - wizardComponentFixture.whenStable().then(() => { - expect(wizardComponent.currentStep).toBe(1); - }); - })); - - it('should initialize and set step without given lecture but preset date', fakeAsync(() => { - wizardComponent.lecture.id = undefined; - wizardComponent.lecture.startDate = dayjs().year(2010).month(3).date(5); - - wizardComponentFixture.detectChanges(); - expect(wizardComponent).not.toBeNull(); - - tick(); - - wizardComponentFixture.whenStable().then(() => { - expect(wizardComponent.currentStep).toBe(2); - }); - })); - - it('should initialize and set given step', fakeAsync(() => { - const route = TestBed.inject(ActivatedRoute); - route.queryParams = of({ step: 3 }); - - wizardComponentFixture.detectChanges(); - expect(wizardComponent).not.toBeNull(); - - wizardComponentFixture.whenStable().then(() => { - expect(wizardComponent.currentStep).toBe(3); - }); - })); - - it('should increase the step when clicked', fakeAsync(() => { - const route = TestBed.inject(ActivatedRoute); - route.queryParams = of({ step: 1 }); - - wizardComponentFixture.detectChanges(); - - wizardComponentFixture.whenStable().then(() => { - expect(wizardComponent.currentStep).toBe(1); - wizardComponent.next(); - expect(wizardComponent.currentStep).toBe(2); - }); - })); - - it('should save the lecture when finishing the last step', fakeAsync(() => { - wizardComponent.saveLectureFunction = () => {}; - const saveStub = jest.spyOn(wizardComponent, 'saveLectureFunction'); - - wizardComponentFixture.detectChanges(); - - wizardComponentFixture.whenStable().then(() => { - expect(wizardComponent.currentStep).toBe(4); - wizardComponent.next(); - expect(saveStub).toHaveBeenCalledOnce(); - }); - })); - - it('should increase the step after lecture created', fakeAsync(() => { - wizardComponentFixture.detectChanges(); - - wizardComponentFixture.whenStable().then(() => { - wizardComponent.currentStep = 2; - wizardComponent.onLectureCreationSucceeded(); - expect(wizardComponent.currentStep).toBe(3); - }); - })); - - it('should return correct icon for last step', fakeAsync(() => { - wizardComponentFixture.detectChanges(); - - wizardComponentFixture.whenStable().then(() => { - wizardComponent.currentStep = 5; - const result = wizardComponent.getNextIcon(); - expect(result).toBe(wizardComponent.faCheck); - }); - })); - - it('should return correct icon for not last step', fakeAsync(() => { - wizardComponentFixture.detectChanges(); - - wizardComponentFixture.whenStable().then(() => { - wizardComponent.currentStep = 3; - const result = wizardComponent.getNextIcon(); - expect(result).toBe(wizardComponent.faArrowRight); - }); - })); - - it('should return correct text for last step', fakeAsync(() => { - wizardComponentFixture.detectChanges(); - - wizardComponentFixture.whenStable().then(() => { - wizardComponent.currentStep = 5; - const result = wizardComponent.getNextText(); - expect(result).toBe('entity.action.finish'); - }); - })); - - it('should return correct text for not last step', fakeAsync(() => { - wizardComponentFixture.detectChanges(); - - wizardComponentFixture.whenStable().then(() => { - wizardComponent.currentStep = 3; - const result = wizardComponent.getNextText(); - expect(result).toBe('artemisApp.lecture.home.nextStepLabel'); - }); - })); - - it('should toggle wizard when lecture not created', fakeAsync(() => { - wizardComponent.toggleModeFunction = () => {}; - const toggleStub = jest.spyOn(wizardComponent, 'toggleModeFunction'); - - wizardComponentFixture.detectChanges(); - - wizardComponentFixture.whenStable().then(() => { - wizardComponent.currentStep = 1; - wizardComponent.toggleWizardMode(); - expect(toggleStub).toHaveBeenCalledOnce(); - }); - })); - - it('should navigate when toggling wizard after lecture was created', fakeAsync(() => { - const router = TestBed.inject(Router); - const navigateStub = jest.spyOn(router, 'navigate'); - - wizardComponentFixture.detectChanges(); - - wizardComponentFixture.whenStable().then(() => { - wizardComponent.currentStep = 3; - wizardComponent.toggleWizardMode(); - expect(navigateStub).toHaveBeenCalledTimes(2); // 1 from init to clear the params and 1 from toggling - }); - })); -}); diff --git a/src/test/javascript/spec/component/shared/image-cropper/keyboard.util.spec.ts b/src/test/javascript/spec/component/shared/image-cropper/keyboard.util.spec.ts new file mode 100644 index 000000000000..19b6070aef3d --- /dev/null +++ b/src/test/javascript/spec/component/shared/image-cropper/keyboard.util.spec.ts @@ -0,0 +1,71 @@ +import { getEventForKey, getInvertedPositionForKey, getPositionForKey } from 'app/shared/image-cropper/utils/keyboard.utils'; + +describe('Keyboard Utils', () => { + describe('getPositionForKey', () => { + it('should return correct position for ArrowUp', () => { + expect(getPositionForKey('ArrowUp')).toBe('top'); + }); + + it('should return correct position for ArrowRight', () => { + expect(getPositionForKey('ArrowRight')).toBe('right'); + }); + + it('should return correct position for ArrowDown', () => { + expect(getPositionForKey('ArrowDown')).toBe('bottom'); + }); + + it('should return correct position for ArrowLeft', () => { + expect(getPositionForKey('ArrowLeft')).toBe('left'); + }); + + it('should return default position for unknown key', () => { + expect(getPositionForKey('UnknownKey')).toBe('left'); + }); + }); + + describe('getInvertedPositionForKey', () => { + it('should return correct inverted position for ArrowUp', () => { + expect(getInvertedPositionForKey('ArrowUp')).toBe('bottom'); + }); + + it('should return correct inverted position for ArrowRight', () => { + expect(getInvertedPositionForKey('ArrowRight')).toBe('left'); + }); + + it('should return correct inverted position for ArrowDown', () => { + expect(getInvertedPositionForKey('ArrowDown')).toBe('top'); + }); + + it('should return correct inverted position for ArrowLeft', () => { + expect(getInvertedPositionForKey('ArrowLeft')).toBe('right'); + }); + + it('should return default inverted position for unknown key', () => { + expect(getInvertedPositionForKey('UnknownKey')).toBe('right'); + }); + }); + + describe('getEventForKey', () => { + const stepSize = 10; + + it('should return correct event for ArrowUp', () => { + expect(getEventForKey('ArrowUp', stepSize)).toEqual({ clientX: 0, clientY: -stepSize }); + }); + + it('should return correct event for ArrowRight', () => { + expect(getEventForKey('ArrowRight', stepSize)).toEqual({ clientX: stepSize, clientY: 0 }); + }); + + it('should return correct event for ArrowDown', () => { + expect(getEventForKey('ArrowDown', stepSize)).toEqual({ clientX: 0, clientY: stepSize }); + }); + + it('should return correct event for ArrowLeft', () => { + expect(getEventForKey('ArrowLeft', stepSize)).toEqual({ clientX: -stepSize, clientY: 0 }); + }); + + it('should return default event for unknown key', () => { + expect(getEventForKey('UnknownKey', stepSize)).toEqual({ clientX: -stepSize, clientY: 0 }); + }); + }); +}); diff --git a/src/test/javascript/spec/util/regex.util.spec.ts b/src/test/javascript/spec/util/regex.util.spec.ts new file mode 100644 index 000000000000..1501b95a9e4a --- /dev/null +++ b/src/test/javascript/spec/util/regex.util.spec.ts @@ -0,0 +1,34 @@ +import { matchesRegexFully } from 'app/utils/regex.util'; + +describe('matchesRegexFully', () => { + it('should return true if regex is undefined', () => { + expect(matchesRegexFully('test', undefined)).toBeTrue(); + }); + + it('should return false if input is undefined', () => { + expect(matchesRegexFully(undefined, 'test')).toBeFalse(); + }); + + it('should return true for a full match', () => { + expect(matchesRegexFully('test', 'test')).toBeTrue(); + }); + + it('should return false for a partial match', () => { + expect(matchesRegexFully('testing', 'test')).toBeFalse(); + }); + + it('should return true for a match with regex special characters', () => { + expect(matchesRegexFully('test123', 'test\\d+')).toBeTrue(); + }); + + it('should return false for no match', () => { + expect(matchesRegexFully('test', 'no-match')).toBeFalse(); + }); + + it('should handle regex without ^ and $', () => { + expect(matchesRegexFully('test', 'test')).toBeTrue(); + expect(matchesRegexFully('test', '^test')).toBeTrue(); + expect(matchesRegexFully('test', 'test$')).toBeTrue(); + expect(matchesRegexFully('test', '^test$')).toBeTrue(); + }); +}); diff --git a/src/test/playwright/e2e/lecture/LectureManagement.spec.ts b/src/test/playwright/e2e/lecture/LectureManagement.spec.ts index 74590f2f5e14..c229c40f701d 100644 --- a/src/test/playwright/e2e/lecture/LectureManagement.spec.ts +++ b/src/test/playwright/e2e/lecture/LectureManagement.spec.ts @@ -41,10 +41,17 @@ test.describe('Lecture management', { tag: '@fast' }, () => { const lectureResponse = await lectureCreation.save(); const lecture: Lecture = await lectureResponse.json(); expect(lectureResponse.status()).toBe(201); + await expect(page).toHaveURL(`/course-management/${course.id}/lectures/${lecture.id}/edit`); + + const adjustedDescription = description! + 'change to enable save button again'; + await lectureCreation.typeDescription(adjustedDescription); + const lectureResponseFromEdit = await lectureCreation.save(); + const lectureFromEdit: Lecture = await lectureResponseFromEdit.json(); + expect(lectureResponseFromEdit.status()).toBe(200); + await page.waitForURL(`**/${course.id}/lectures/${lectureFromEdit.id}`); - await page.waitForURL(`**/${course.id}/lectures/${lecture.id}`); await expect(lectureManagement.getLectureTitle()).toContainText(lectureData.title); - await expect(lectureManagement.getLectureDescription()).toContainText(description!); + await expect(lectureManagement.getLectureDescription()).toContainText(adjustedDescription!); await expect(lectureManagement.getLectureVisibleDate()).toContainText(lectureData.visibleDate!.format(dateFormat)); await expect(lectureManagement.getLectureStartDate()).toContainText(lectureData.startDate!.format(dateFormat)); await expect(lectureManagement.getLectureEndDate()).toContainText(lectureData.endDate!.format(dateFormat)); From 1798b257ff1b547aface3c7795d46113e08dad78 Mon Sep 17 00:00:00 2001 From: Stephan Krusche Date: Mon, 6 Jan 2025 09:23:00 +0100 Subject: [PATCH 33/34] Development: Bump version to 7.8.3 (bugfix update) --- README.md | 2 +- build.gradle | 2 +- package-lock.json | 4 ++-- package.json | 2 +- 4 files changed, 5 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index 0c776b37df5c..99248761d635 100644 --- a/README.md +++ b/README.md @@ -193,7 +193,7 @@ Refer to [Using JHipster in production](http://www.jhipster.tech/production) for The following command can automate the deployment to a server. The example shows the deployment to the main Artemis test server (which runs a virtual machine): ```shell -./artemis-server-cli deploy username@artemis-test0.artemis.in.tum.de -w build/libs/Artemis-7.8.2.war +./artemis-server-cli deploy username@artemis-test0.artemis.in.tum.de -w build/libs/Artemis-7.8.3.war ``` ## Architecture diff --git a/build.gradle b/build.gradle index 339abaed3836..7d3099e888c4 100644 --- a/build.gradle +++ b/build.gradle @@ -25,7 +25,7 @@ plugins { } group = "de.tum.cit.aet.artemis" -version = "7.8.2" +version = "7.8.3" description = "Interactive Learning with Individual Feedback" java { diff --git a/package-lock.json b/package-lock.json index 319cdead9f47..a543cb1fb222 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "artemis", - "version": "7.8.2", + "version": "7.8.3", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "artemis", - "version": "7.8.2", + "version": "7.8.3", "hasInstallScript": true, "license": "MIT", "dependencies": { diff --git a/package.json b/package.json index 37a9cdd0a97a..d392de262aa4 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "artemis", - "version": "7.8.2", + "version": "7.8.3", "description": "Interactive Learning with Individual Feedback", "private": true, "license": "MIT", From 599c92e4aed3d44bf1d246968db909e06a48fa10 Mon Sep 17 00:00:00 2001 From: Marcel Gaupp Date: Thu, 9 Jan 2025 19:27:17 +0100 Subject: [PATCH 34/34] Programming exercises: Fix programming language feature flags (#10110) --- ...kinsProgrammingLanguageFeatureService.java | 4 +-- ...alCIProgrammingLanguageFeatureService.java | 34 +++++++++---------- 2 files changed, 19 insertions(+), 19 deletions(-) diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsProgrammingLanguageFeatureService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsProgrammingLanguageFeatureService.java index 6ad7b4c65aec..a858a171efd7 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsProgrammingLanguageFeatureService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/jenkins/JenkinsProgrammingLanguageFeatureService.java @@ -44,9 +44,9 @@ public JenkinsProgrammingLanguageFeatureService() { programmingLanguageFeatures.put(GO, new ProgrammingLanguageFeature(GO, false, false, true, true, false, List.of(), false)); programmingLanguageFeatures.put(HASKELL, new ProgrammingLanguageFeature(HASKELL, false, false, false, false, true, List.of(), false)); programmingLanguageFeatures.put(JAVA, - new ProgrammingLanguageFeature(JAVA, true, true, true, true, false, List.of(PLAIN_GRADLE, GRADLE_GRADLE, PLAIN_MAVEN, MAVEN_MAVEN, MAVEN_BLACKBOX), true)); + new ProgrammingLanguageFeature(JAVA, true, true, true, true, false, List.of(PLAIN_GRADLE, GRADLE_GRADLE, PLAIN_MAVEN, MAVEN_MAVEN, MAVEN_BLACKBOX), false)); programmingLanguageFeatures.put(JAVASCRIPT, new ProgrammingLanguageFeature(JAVASCRIPT, false, false, true, false, false, List.of(), false)); - programmingLanguageFeatures.put(KOTLIN, new ProgrammingLanguageFeature(KOTLIN, true, false, true, true, false, List.of(), true)); + programmingLanguageFeatures.put(KOTLIN, new ProgrammingLanguageFeature(KOTLIN, true, false, true, true, false, List.of(), false)); programmingLanguageFeatures.put(PYTHON, new ProgrammingLanguageFeature(PYTHON, false, false, true, false, false, List.of(), false)); programmingLanguageFeatures.put(R, new ProgrammingLanguageFeature(R, false, false, true, false, false, List.of(), false)); programmingLanguageFeatures.put(RUST, new ProgrammingLanguageFeature(RUST, false, false, true, false, false, List.of(), false)); diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIProgrammingLanguageFeatureService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIProgrammingLanguageFeatureService.java index b8478cf65daf..0c46a202574b 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIProgrammingLanguageFeatureService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/localci/LocalCIProgrammingLanguageFeatureService.java @@ -43,23 +43,23 @@ public class LocalCIProgrammingLanguageFeatureService extends ProgrammingLanguag public LocalCIProgrammingLanguageFeatureService() { // Must be extended once a new programming language is added - programmingLanguageFeatures.put(EMPTY, new ProgrammingLanguageFeature(EMPTY, false, false, false, false, false, List.of(), false)); - programmingLanguageFeatures.put(ASSEMBLER, new ProgrammingLanguageFeature(ASSEMBLER, false, false, false, false, false, List.of(), false)); - programmingLanguageFeatures.put(C, new ProgrammingLanguageFeature(C, false, true, true, false, false, List.of(FACT, GCC), false)); - programmingLanguageFeatures.put(C_PLUS_PLUS, new ProgrammingLanguageFeature(C_PLUS_PLUS, false, false, true, false, false, List.of(), false)); - programmingLanguageFeatures.put(C_SHARP, new ProgrammingLanguageFeature(C_SHARP, false, false, true, false, false, List.of(), false)); - programmingLanguageFeatures.put(GO, new ProgrammingLanguageFeature(GO, false, false, true, true, false, List.of(), false)); - programmingLanguageFeatures.put(HASKELL, new ProgrammingLanguageFeature(HASKELL, true, false, false, false, true, List.of(), false)); + programmingLanguageFeatures.put(EMPTY, new ProgrammingLanguageFeature(EMPTY, false, false, false, false, false, List.of(), true)); + programmingLanguageFeatures.put(ASSEMBLER, new ProgrammingLanguageFeature(ASSEMBLER, false, false, false, false, false, List.of(), true)); + programmingLanguageFeatures.put(C, new ProgrammingLanguageFeature(C, false, true, true, false, false, List.of(FACT, GCC), true)); + programmingLanguageFeatures.put(C_PLUS_PLUS, new ProgrammingLanguageFeature(C_PLUS_PLUS, false, false, true, false, false, List.of(), true)); + programmingLanguageFeatures.put(C_SHARP, new ProgrammingLanguageFeature(C_SHARP, false, false, true, false, false, List.of(), true)); + programmingLanguageFeatures.put(GO, new ProgrammingLanguageFeature(GO, false, false, true, true, false, List.of(), true)); + programmingLanguageFeatures.put(HASKELL, new ProgrammingLanguageFeature(HASKELL, true, false, false, false, true, List.of(), true)); programmingLanguageFeatures.put(JAVA, - new ProgrammingLanguageFeature(JAVA, true, true, true, true, false, List.of(PLAIN_GRADLE, GRADLE_GRADLE, PLAIN_MAVEN, MAVEN_MAVEN), false)); - programmingLanguageFeatures.put(JAVASCRIPT, new ProgrammingLanguageFeature(JAVASCRIPT, false, false, true, false, false, List.of(), false)); - programmingLanguageFeatures.put(KOTLIN, new ProgrammingLanguageFeature(KOTLIN, false, false, true, true, false, List.of(), false)); - programmingLanguageFeatures.put(OCAML, new ProgrammingLanguageFeature(OCAML, false, false, false, false, true, List.of(), false)); - programmingLanguageFeatures.put(PYTHON, new ProgrammingLanguageFeature(PYTHON, false, false, true, false, false, List.of(), false)); - programmingLanguageFeatures.put(R, new ProgrammingLanguageFeature(R, false, false, true, false, false, List.of(), false)); - programmingLanguageFeatures.put(RUST, new ProgrammingLanguageFeature(RUST, false, false, true, false, false, List.of(), false)); - programmingLanguageFeatures.put(SWIFT, new ProgrammingLanguageFeature(SWIFT, false, false, true, true, false, List.of(PLAIN), false)); - programmingLanguageFeatures.put(TYPESCRIPT, new ProgrammingLanguageFeature(TYPESCRIPT, false, false, true, false, false, List.of(), false)); - programmingLanguageFeatures.put(VHDL, new ProgrammingLanguageFeature(VHDL, false, false, false, false, false, List.of(), false)); + new ProgrammingLanguageFeature(JAVA, true, true, true, true, false, List.of(PLAIN_GRADLE, GRADLE_GRADLE, PLAIN_MAVEN, MAVEN_MAVEN), true)); + programmingLanguageFeatures.put(JAVASCRIPT, new ProgrammingLanguageFeature(JAVASCRIPT, false, false, true, false, false, List.of(), true)); + programmingLanguageFeatures.put(KOTLIN, new ProgrammingLanguageFeature(KOTLIN, false, false, true, true, false, List.of(), true)); + programmingLanguageFeatures.put(OCAML, new ProgrammingLanguageFeature(OCAML, false, false, false, false, true, List.of(), true)); + programmingLanguageFeatures.put(PYTHON, new ProgrammingLanguageFeature(PYTHON, false, true, true, false, false, List.of(), true)); + programmingLanguageFeatures.put(R, new ProgrammingLanguageFeature(R, false, false, true, false, false, List.of(), true)); + programmingLanguageFeatures.put(RUST, new ProgrammingLanguageFeature(RUST, false, false, true, false, false, List.of(), true)); + programmingLanguageFeatures.put(SWIFT, new ProgrammingLanguageFeature(SWIFT, false, false, true, true, false, List.of(PLAIN), true)); + programmingLanguageFeatures.put(TYPESCRIPT, new ProgrammingLanguageFeature(TYPESCRIPT, false, false, true, false, false, List.of(), true)); + programmingLanguageFeatures.put(VHDL, new ProgrammingLanguageFeature(VHDL, false, false, false, false, false, List.of(), true)); } }