From bee69786631e0eef3cee17e12edf967b43b2a214 Mon Sep 17 00:00:00 2001 From: entholzer Date: Sat, 21 Dec 2024 12:18:20 +0100 Subject: [PATCH 1/8] added translation strings --- src/main/resources/i18n/messages.properties | 9 +++++++++ src/main/resources/i18n/messages_de.properties | 8 ++++++++ src/main/resources/i18n/messages_en.properties | 8 ++++++++ src/main/webapp/i18n/de/notification.json | 8 ++++++-- src/main/webapp/i18n/en/notification.json | 8 ++++++-- 5 files changed, 37 insertions(+), 4 deletions(-) diff --git a/src/main/resources/i18n/messages.properties b/src/main/resources/i18n/messages.properties index 6580e702bd3c..81e506aa51c7 100644 --- a/src/main/resources/i18n/messages.properties +++ b/src/main/resources/i18n/messages.properties @@ -137,6 +137,12 @@ email.notification.sshKeyExpiry.renew = You can renew your SSH key here: email.notification.sshKeyExpiry.sshKeyHash = SSH key hash: email.notification.sshKeyExpiry.sshKeyLabel = SSH key label: +# VCS Token Settings +email.notification.vcsAccessTokenAdded.title = A new VCS access token was added to your account. +email.notification.vcsAccessTokenAdded.ifMistake = If you believe this token was added in error, you can remove it and disable access at the following location: +email.notification.vcsAccessTokenExpiry.title = Your personal VCS access token has expired. +email.notification.vcsAccessTokenExpiry.renew = You can renew it here: + # Email Subjects # The reason for the format artemisApp.{notificationCategory}.title.{notificicationType} is that these placeholders are also used in the client and this is the format used there artemisApp.groupNotification.title.attachmentChange = Attachment updated @@ -166,3 +172,6 @@ artemisApp.singleUserNotification.title.dataExportFailed = Your Artemis data exp artemisApp.singleUserNotification.title.sshKeyAdded = New SSH key added to account artemisApp.singleUserNotification.title.sshKeyExpiresSoon = SSH key expires soon artemisApp.singleUserNotification.title.sshKeyHasExpired = SSH key has expired +artemisApp.singleUserNotification.title.vcsAccessTokenAdded = New VCS access token added to account +artemisApp.singleUserNotification.title.vcsAccessTokenExpired = VCS access token has expired + diff --git a/src/main/resources/i18n/messages_de.properties b/src/main/resources/i18n/messages_de.properties index 0960e6223fec..ea364a60fec5 100644 --- a/src/main/resources/i18n/messages_de.properties +++ b/src/main/resources/i18n/messages_de.properties @@ -137,6 +137,12 @@ email.notification.sshKeyExpiry.renew = Hier kannst du deinen SSH-Schlüssel akt email.notification.sshKeyExpiry.sshKeyHash = SSH-Schlüssel-Hash: email.notification.sshKeyExpiry.sshKeyLabel = SSH-Schlüssel-Label: +# VCS access token User Settings +email.notification.vcsAccessTokenAdded.title = Ein neues VCS Zugriffstoken wurde zu deinem Benutzerkonto hinzugefügt. +email.notification.vcsAccessTokenAdded.ifMistake = Wenn du glaubst, dass dieses Zugriffstoken irrtümlich hinzugefügt wurde, kannst du es entfernen und den Zugriff an folgender Stelle deaktivieren: +email.notification.vcsAccessTokenExpiry.title = Your personal VCS access token has expired. +email.notification.vcsAccessTokenExpiry.renew = You can renew it here: + # Email Subjects # The reason for the format artemisApp.{notificationCategory}.title.{notificicationType} is that these placeholders are also used in the client and this is the format used there artemisApp.groupNotification.title.attachmentChange = Anhang aktualisiert @@ -166,3 +172,5 @@ artemisApp.singleUserNotification.title.dataExportFailed = Dein Artemis Datenexp artemisApp.singleUserNotification.title.sshKeyAdded = Neuer SSH-Schlüssel hinzugefügt artemisApp.singleUserNotification.title.sshKeyExpiresSoon = SSH-Schlüssel läuft bald ab artemisApp.singleUserNotification.title.sshKeyHasExpired = SSH-Schlüssel ist abgelaufen +artemisApp.singleUserNotification.title.vcsAccessTokenAdded = Neues VCS Zugriffstoken hinzugefügt +artemisApp.singleUserNotification.title.vcsAccessTokenExpired = VCS Zugriffstoken ist abgelaufen diff --git a/src/main/resources/i18n/messages_en.properties b/src/main/resources/i18n/messages_en.properties index 01e56d22ce6a..f3acae3d9d6c 100644 --- a/src/main/resources/i18n/messages_en.properties +++ b/src/main/resources/i18n/messages_en.properties @@ -137,6 +137,12 @@ email.notification.sshKeyExpiry.renew = You can renew your SSH key here: email.notification.sshKeyExpiry.sshKeyHash = SSH key hash: email.notification.sshKeyExpiry.sshKeyLabel = SSH key label: +# VCS Token Settings +email.notification.vcsAccessTokenAdded.title = A new VCS access token was added to your account. +email.notification.vcsAccessTokenAdded.ifMistake = If you believe this token was added in error, you can remove it and disable access at the following location: +email.notification.vcsAccessTokenExpiry.title = Your personal VCS access token has expired. +email.notification.vcsAccessTokenExpiry.renew = You can renew it here: + # Email Subjects # The reason for the format artemisApp.{notificationCategory}.title.{notificicationType} is that these placeholders are also used in the client and this is the format used there artemisApp.groupNotification.title.attachmentChange = Attachment updated @@ -165,3 +171,5 @@ artemisApp.singleUserNotification.title.dataExportFailed = Your Artemis data exp artemisApp.singleUserNotification.title.sshKeyAdded = New SSH key added to account artemisApp.singleUserNotification.title.sshKeyExpiresSoon = SSH key expires soon artemisApp.singleUserNotification.title.sshKeyHasExpired = SSH key has expired +artemisApp.singleUserNotification.title.vcsAccessTokenAdded = New VCS access token added to account +artemisApp.singleUserNotification.title.vcsAccessTokenExpired = VCS access token has expired diff --git a/src/main/webapp/i18n/de/notification.json b/src/main/webapp/i18n/de/notification.json index 0f5dd8ff6261..d98ca682a349 100644 --- a/src/main/webapp/i18n/de/notification.json +++ b/src/main/webapp/i18n/de/notification.json @@ -71,7 +71,9 @@ "courseArchiveFailed": "Kursarchivierung fehlgeschlagen", "examArchiveStarted": "Klausurarchivierung gestartet", "examArchiveFinished": "Klausurarchivierung beendet", - "examArchiveFailed": "Klausurarchivierung fehlgeschlagen" + "examArchiveFailed": "Klausurarchivierung fehlgeschlagen", + "vcsAccessTokenAdded": "Neues VCS Zugriffstoken hinzugefügt", + "vcsAccessTokenExpired": "VCS Zugriffstoken abgelaufen" }, "text": { "attachmentChange": "Der Anhang \"{{ placeholderValues.1 }}\" von \"{{ placeholderValues.2 }}\" im Kurs \"{{ placeholderValues.0}}\" wurde aktualisiert.", @@ -97,7 +99,9 @@ "examArchiveStarted": "Die Klausur \"{{ placeholderValues.1 }}\" wird archiviert.", "examArchiveFinishedWithoutErrors": "Die Klausur \"{{ placeholderValues.1 }}\" wurde archiviert.", "examArchiveFinishedWithErrors": "Die Klausur \"{{ placeholderValues.1 }}\" wurde archiviert. Einige Aufgaben konnten nicht in das Archiv aufgenommen werden:

{{ placeholderValues.2 }}", - "examArchiveFailed": "Es gab ein Problem beim Archivieren der Klausur \"{{ placeholderValues.1 }}\":

{{ placeholderValues.2 }}" + "examArchiveFailed": "Es gab ein Problem beim Archivieren der Klausur \"{{ placeholderValues.1 }}\":

{{ placeholderValues.2 }}", + "vcsAccessTokenAdded": "Du hast erfolgreich ein persönliches Zugriffstoken hinzugefügt.", + "vcsAccessTokenExpired": "Dein persönliches Zugriffstoken ist abgelaufen." } }, "singleUserNotification": { diff --git a/src/main/webapp/i18n/en/notification.json b/src/main/webapp/i18n/en/notification.json index e4ba3381e92c..3a3b27b699e6 100644 --- a/src/main/webapp/i18n/en/notification.json +++ b/src/main/webapp/i18n/en/notification.json @@ -126,7 +126,9 @@ "dataExportFailed": "Data export failed", "sshKeyAdded": "New SSH key added", "sshKeyExpiresSoon": "SSH key will expire soon", - "sshKeyHasExpired": "SSH key has expired" + "sshKeyHasExpired": "SSH key has expired", + "vcsAccessTokenAdded": "New VCS access token added", + "vcsAccessTokenExpired": "VCS access token has expired" }, "text": { "newReplyForExercisePost": "Your post regarding exercise \"{{ placeholderValues.8 }}\" in the course \"{{ placeholderValues.0 }}\" got a new reply: \"{{ placeholderValues.5 }}\"", @@ -156,7 +158,9 @@ "dataExportFailed": "Your data export could not be created. Please try again later.", "sshKeyAdded": "You have successfully added a new SSH key", "sshKeyExpiresSoon": "Your SSH key with the label \"{{ placeholderValues.0}}\" will expire on {{ placeholderValues.1 }}.", - "sshKeyHasExpired": "Your SSH key with the label \"{{ placeholderValues.0}}\" has expired on {{ placeholderValues.1 }}." + "sshKeyHasExpired": "Your SSH key with the label \"{{ placeholderValues.0}}\" has expired on {{ placeholderValues.1 }}.", + "vcsAccessTokenAdded": "You have successfully added a new VCS access token", + "vcsAccessTokenExpired": "Your personal version control access token has expired." } }, "tutorialGroupNotification": { From 773f84c2be92f274dd5bcc0abaf764117fc55790 Mon Sep 17 00:00:00 2001 From: entholzer Date: Sat, 21 Dec 2024 12:18:35 +0100 Subject: [PATCH 2/8] added email templates --- .../vcsAccessTokenAddedEmail.html | 28 +++++++++++++++++++ .../vcsAccessTokenExpiredEmail.html | 28 +++++++++++++++++++ 2 files changed, 56 insertions(+) create mode 100644 src/main/resources/templates/mail/notification/vcsAccessTokenAddedEmail.html create mode 100644 src/main/resources/templates/mail/notification/vcsAccessTokenExpiredEmail.html diff --git a/src/main/resources/templates/mail/notification/vcsAccessTokenAddedEmail.html b/src/main/resources/templates/mail/notification/vcsAccessTokenAddedEmail.html new file mode 100644 index 000000000000..54a6a2c8dd7e --- /dev/null +++ b/src/main/resources/templates/mail/notification/vcsAccessTokenAddedEmail.html @@ -0,0 +1,28 @@ + + + + + + + + + +
+ + +

+ The following SSH key was added to your account: +

+ +

+ If adding it was by mistake, remove it here: + + Login + link +

+ + +
+ + + diff --git a/src/main/resources/templates/mail/notification/vcsAccessTokenExpiredEmail.html b/src/main/resources/templates/mail/notification/vcsAccessTokenExpiredEmail.html new file mode 100644 index 000000000000..71194367aa37 --- /dev/null +++ b/src/main/resources/templates/mail/notification/vcsAccessTokenExpiredEmail.html @@ -0,0 +1,28 @@ + + + + + + + + + +
+ + +

+ Your VCS access token has expired +

+ +

+ You can renew it here + + Login + link +

+ + +
+ + + From 62980dde4ca856e0b349dd865fe042d3199b7eb4 Mon Sep 17 00:00:00 2001 From: entholzer Date: Sat, 21 Dec 2024 12:18:45 +0100 Subject: [PATCH 3/8] added tests --- .../SingleUserNotificationServiceTest.java | 58 +++++++++++++++++++ 1 file changed, 58 insertions(+) diff --git a/src/test/java/de/tum/cit/aet/artemis/communication/notification/SingleUserNotificationServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/communication/notification/SingleUserNotificationServiceTest.java index 0dbdfae61e53..956696bfd53b 100644 --- a/src/test/java/de/tum/cit/aet/artemis/communication/notification/SingleUserNotificationServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/communication/notification/SingleUserNotificationServiceTest.java @@ -34,6 +34,8 @@ import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.TUTORIAL_GROUP_REGISTRATION_TUTOR_TITLE; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.TUTORIAL_GROUP_UNASSIGNED_TEXT; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.TUTORIAL_GROUP_UNASSIGNED_TITLE; +import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.VCS_ACCESS_TOKEN_ADDED_TEXT; +import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.VCS_ACCESS_TOKEN_EXPIRED_TEXT; import static de.tum.cit.aet.artemis.communication.service.notifications.NotificationSettingsService.NOTIFICATION_USER_NOTIFICATION_DATA_EXPORT_CREATED; import static de.tum.cit.aet.artemis.communication.service.notifications.NotificationSettingsService.NOTIFICATION_USER_NOTIFICATION_DATA_EXPORT_FAILED; import static de.tum.cit.aet.artemis.communication.service.notifications.NotificationSettingsService.NOTIFICATION__EXERCISE_NOTIFICATION__EXERCISE_SUBMISSION_ASSESSED; @@ -111,6 +113,7 @@ import de.tum.cit.aet.artemis.programming.dto.UserSshPublicKeyDTO; import de.tum.cit.aet.artemis.programming.service.sshuserkeys.UserSshPublicKeyExpiryNotificationService; import de.tum.cit.aet.artemis.programming.service.sshuserkeys.UserSshPublicKeyService; +import de.tum.cit.aet.artemis.programming.service.tokens.UserTokenExpiryNotificationService; import de.tum.cit.aet.artemis.shared.base.AbstractSpringIntegrationIndependentTest; import de.tum.cit.aet.artemis.text.domain.TextExercise; import de.tum.cit.aet.artemis.text.util.TextExerciseFactory; @@ -147,6 +150,9 @@ class SingleUserNotificationServiceTest extends AbstractSpringIntegrationIndepen @Autowired private UserSshPublicKeyExpiryNotificationService userSshPublicKeyExpiryNotificationService; + @Autowired + private UserTokenExpiryNotificationService userTokenExpiryNotificationService; + @Autowired private UserSshPublicKeyService userSshPublicKeyService; @@ -503,6 +509,58 @@ void checkFirstNotification() { } } + // User VCS access token related (expiry warning and newly added token) + + @Nested + class UserTokenExpiryNotification { + + List sentNotifications; + + @AfterEach + void tearDown() throws Exception { + user.setVcsAccessTokenExpiryDate(null); + user.setVcsAccessToken(null); + userTestRepository.save(user); + } + + @Test + void shouldNotifyUserAboutNewlyAddedVcsAccessToken() { + singleUserNotificationService.notifyUserAboutNewlyAddedVcsAccessToken(user); + + sentNotifications = notificationRepository.findAll(); + assertThat(sentNotifications.getFirst()).isInstanceOf(SingleUserNotification.class); + assertThat(((SingleUserNotification) sentNotifications.getFirst()).getRecipient()).isEqualTo(user); + assertThat((sentNotifications.getFirst()).getText()).isEqualTo(VCS_ACCESS_TOKEN_ADDED_TEXT); + } + + @Test + void shouldNotifyUserAboutExpiredVcsAccessToken() { + user.setVcsAccessToken("token"); + user.setVcsAccessTokenExpiryDate(ZonedDateTime.now().minusHours(5)); + userTestRepository.save(user); + + userTokenExpiryNotificationService.sendTokenExpirationNotifications(); + + sentNotifications = notificationRepository.findAll(); + assertThat(sentNotifications).hasSize(1); + assertThat(sentNotifications.getFirst()).isInstanceOf(SingleUserNotification.class); + assertThat(((SingleUserNotification) sentNotifications.getFirst()).getRecipient()).isEqualTo(user); + assertThat((sentNotifications.getFirst()).getText()).isEqualTo(VCS_ACCESS_TOKEN_EXPIRED_TEXT); + } + + @Test + void shouldNotNotifyUserAboutVcsAccessTokenExpiryWhenTokenIsNotExpired() { + user.setVcsAccessToken("token"); + user.setVcsAccessTokenExpiryDate(ZonedDateTime.now().plusDays(5)); + userTestRepository.save(user); + + userTokenExpiryNotificationService.sendTokenExpirationNotifications(); + + sentNotifications = notificationRepository.findAll(); + assertThat(sentNotifications).hasSize(0); + } + } + // Plagiarism related /** From 17cff21522041e7f70bae04ee6474ffe1d0ae77a Mon Sep 17 00:00:00 2001 From: entholzer Date: Sat, 21 Dec 2024 12:19:23 +0100 Subject: [PATCH 4/8] added Notifications for new tokens and token expirz --- .../domain/NotificationType.java | 2 +- .../notification/NotificationConstants.java | 13 +++- .../SingleUserNotificationFactory.java | 25 ++++++++ .../service/notifications/MailService.java | 2 + .../NotificationSettingsService.java | 16 ++++- .../SingleUserNotificationService.java | 23 ++++++- .../core/repository/UserRepository.java | 2 + .../aet/artemis/core/web/AccountResource.java | 9 ++- .../UserTokenExpiryNotificationService.java | 64 +++++++++++++++++++ 9 files changed, 148 insertions(+), 8 deletions(-) create mode 100644 src/main/java/de/tum/cit/aet/artemis/programming/service/tokens/UserTokenExpiryNotificationService.java diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/domain/NotificationType.java b/src/main/java/de/tum/cit/aet/artemis/communication/domain/NotificationType.java index 2a85412663bb..b05165088c28 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/domain/NotificationType.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/domain/NotificationType.java @@ -10,5 +10,5 @@ public enum NotificationType { TUTORIAL_GROUP_UNASSIGNED, CONVERSATION_NEW_MESSAGE, CONVERSATION_NEW_REPLY_MESSAGE, CONVERSATION_USER_MENTIONED, CONVERSATION_CREATE_ONE_TO_ONE_CHAT, CONVERSATION_CREATE_GROUP_CHAT, CONVERSATION_ADD_USER_GROUP_CHAT, CONVERSATION_ADD_USER_CHANNEL, CONVERSATION_REMOVE_USER_GROUP_CHAT, CONVERSATION_REMOVE_USER_CHANNEL, CONVERSATION_DELETE_CHANNEL, DATA_EXPORT_CREATED, DATA_EXPORT_FAILED, PROGRAMMING_REPOSITORY_LOCKS, PROGRAMMING_BUILD_RUN_UPDATE, SSH_KEY_ADDED, SSH_KEY_EXPIRES_SOON, - SSH_KEY_HAS_EXPIRED + SSH_KEY_HAS_EXPIRED, VCS_ACCESS_TOKEN_ADDED, VCS_ACCESS_TOKEN_EXPIRED } diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/domain/notification/NotificationConstants.java b/src/main/java/de/tum/cit/aet/artemis/communication/domain/notification/NotificationConstants.java index b9c8b651c3e3..eb1e60262243 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/domain/notification/NotificationConstants.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/domain/notification/NotificationConstants.java @@ -55,6 +55,8 @@ import static de.tum.cit.aet.artemis.communication.domain.NotificationType.TUTORIAL_GROUP_REGISTRATION_TUTOR; import static de.tum.cit.aet.artemis.communication.domain.NotificationType.TUTORIAL_GROUP_UNASSIGNED; import static de.tum.cit.aet.artemis.communication.domain.NotificationType.TUTORIAL_GROUP_UPDATED; +import static de.tum.cit.aet.artemis.communication.domain.NotificationType.VCS_ACCESS_TOKEN_ADDED; +import static de.tum.cit.aet.artemis.communication.domain.NotificationType.VCS_ACCESS_TOKEN_EXPIRED; import com.google.common.collect.BiMap; import com.google.common.collect.ImmutableBiMap; @@ -156,6 +158,10 @@ public class NotificationConstants { public static final String SSH_KEY_HAS_EXPIRED_TITLE = "artemisApp.singleUserNotification.title.sshKeyHasExpired"; + public static final String VCS_ACCESS_TOKEN_ADDED_TITLE = "artemisApp.singleUserNotification.title.vcsAccessTokenAdded"; + + public static final String VCS_ACCESS_TOKEN_EXPIRED_TITLE = "artemisApp.singleUserNotification.title.vcsAccessTokenExpired"; + // Texts public static final String LIVE_EXAM_EXERCISE_UPDATE_NOTIFICATION_TEXT = "artemisApp.groupNotification.text.liveExamExerciseUpdate"; @@ -295,6 +301,10 @@ public class NotificationConstants { public static final String SSH_KEY_HAS_EXPIRED_TEXT = "artemisApp.singleUserNotification.text.sshKeyHasExpired"; + public static final String VCS_ACCESS_TOKEN_ADDED_TEXT = "artemisApp.singleUserNotification.text.vcsAccessTokenAdded"; + + public static final String VCS_ACCESS_TOKEN_EXPIRED_TEXT = "artemisApp.singleUserNotification.text.vcsAccessTokenExpired"; + // bidirectional map private static final BiMap NOTIFICATION_TYPE_AND_TITLE_MAP = new ImmutableBiMap.Builder() .put(EXERCISE_SUBMISSION_ASSESSED, EXERCISE_SUBMISSION_ASSESSED_TITLE).put(ATTACHMENT_CHANGE, ATTACHMENT_CHANGE_TITLE).put(EXERCISE_RELEASED, EXERCISE_RELEASED_TITLE) @@ -321,7 +331,8 @@ public class NotificationConstants { .put(CONVERSATION_REMOVE_USER_CHANNEL, CONVERSATION_REMOVE_USER_CHANNEL_TITLE).put(CONVERSATION_DELETE_CHANNEL, CONVERSATION_DELETE_CHANNEL_TITLE) .put(DATA_EXPORT_CREATED, DATA_EXPORT_CREATED_TITLE).put(DATA_EXPORT_FAILED, DATA_EXPORT_FAILED_TITLE) .put(PROGRAMMING_REPOSITORY_LOCKS, PROGRAMMING_REPOSITORY_LOCKS_TITLE).put(PROGRAMMING_BUILD_RUN_UPDATE, PROGRAMMING_BUILD_RUN_UPDATE_TITLE) - .put(SSH_KEY_ADDED, SSH_KEY_ADDED_TITLE).put(SSH_KEY_EXPIRES_SOON, SSH_KEY_EXPIRES_SOON_TITLE).put(SSH_KEY_HAS_EXPIRED, SSH_KEY_HAS_EXPIRED_TITLE).build(); + .put(SSH_KEY_ADDED, SSH_KEY_ADDED_TITLE).put(SSH_KEY_EXPIRES_SOON, SSH_KEY_EXPIRES_SOON_TITLE).put(SSH_KEY_HAS_EXPIRED, SSH_KEY_HAS_EXPIRED_TITLE) + .put(VCS_ACCESS_TOKEN_ADDED, VCS_ACCESS_TOKEN_ADDED_TITLE).put(VCS_ACCESS_TOKEN_EXPIRED, VCS_ACCESS_TOKEN_EXPIRED_TITLE).build(); /** * Finds the corresponding NotificationType for the provided notification title diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/domain/notification/SingleUserNotificationFactory.java b/src/main/java/de/tum/cit/aet/artemis/communication/domain/notification/SingleUserNotificationFactory.java index 89df0016e4da..deb80dcdcca3 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/domain/notification/SingleUserNotificationFactory.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/domain/notification/SingleUserNotificationFactory.java @@ -64,6 +64,10 @@ import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.TUTORIAL_GROUP_REGISTRATION_STUDENT_TEXT; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.TUTORIAL_GROUP_REGISTRATION_TUTOR_TEXT; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.TUTORIAL_GROUP_UNASSIGNED_TEXT; +import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.VCS_ACCESS_TOKEN_ADDED_TEXT; +import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.VCS_ACCESS_TOKEN_ADDED_TITLE; +import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.VCS_ACCESS_TOKEN_EXPIRED_TEXT; +import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.VCS_ACCESS_TOKEN_EXPIRED_TITLE; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.findCorrespondingNotificationTitleOrThrow; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationTargetFactory.createConversationCreationTarget; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationTargetFactory.createConversationDeletionTarget; @@ -191,6 +195,27 @@ public static SingleUserNotification createNotification(UserSshPublicKey key, No } } + /** + * Creates a user notification based on the given SSH key and notification type. + * + * @param vcsAccessToken The access token of the user + * @param notificationType The type of notification to create (e.g., key added, expiring, or expired). + * @param recipient The user who will receive the notification. + * @return A configured {@link SingleUserNotification}. + * @throws UnsupportedOperationException if the notification type is unsupported. + */ + public static SingleUserNotification createNotification(String vcsAccessToken, NotificationType notificationType, User recipient) { + switch (notificationType) { + case VCS_ACCESS_TOKEN_ADDED -> { + return new SingleUserNotification(recipient, VCS_ACCESS_TOKEN_ADDED_TITLE, VCS_ACCESS_TOKEN_ADDED_TEXT, true, new String[] {}); + } + case VCS_ACCESS_TOKEN_EXPIRED -> { + return new SingleUserNotification(recipient, VCS_ACCESS_TOKEN_EXPIRED_TITLE, VCS_ACCESS_TOKEN_EXPIRED_TEXT, true, new String[] {}); + } + default -> throw new UnsupportedOperationException("Unsupported NotificationType: " + notificationType); + } + } + /** * Creates an instance of SingleUserNotification based on plagiarisms. * diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/MailService.java b/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/MailService.java index 2d7b4214aa58..8ea46c7998d2 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/MailService.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/MailService.java @@ -408,6 +408,8 @@ private String createContentForNotificationEmailByType(NotificationType notifica case SSH_KEY_ADDED -> templateEngine.process("mail/notification/sshKeyAddedEmail", context); case SSH_KEY_EXPIRES_SOON -> templateEngine.process("mail/notification/sshKeyExpiresSoonEmail", context); case SSH_KEY_HAS_EXPIRED -> templateEngine.process("mail/notification/sshKeyHasExpiredEmail", context); + case VCS_ACCESS_TOKEN_ADDED -> templateEngine.process("mail/notification/vcsAccessTokenAddedEmail", context); + case VCS_ACCESS_TOKEN_EXPIRED -> templateEngine.process("mail/notification/vcsAccessTokenExpiredEmail", context); default -> throw new UnsupportedOperationException("Unsupported NotificationType: " + notificationType); }; diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/NotificationSettingsService.java b/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/NotificationSettingsService.java index c953c3700dae..b5bf3adf9aac 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/NotificationSettingsService.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/NotificationSettingsService.java @@ -44,6 +44,8 @@ import static de.tum.cit.aet.artemis.communication.domain.NotificationType.TUTORIAL_GROUP_REGISTRATION_TUTOR; import static de.tum.cit.aet.artemis.communication.domain.NotificationType.TUTORIAL_GROUP_UNASSIGNED; import static de.tum.cit.aet.artemis.communication.domain.NotificationType.TUTORIAL_GROUP_UPDATED; +import static de.tum.cit.aet.artemis.communication.domain.NotificationType.VCS_ACCESS_TOKEN_ADDED; +import static de.tum.cit.aet.artemis.communication.domain.NotificationType.VCS_ACCESS_TOKEN_EXPIRED; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.findCorrespondingNotificationType; import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_CORE; @@ -144,6 +146,10 @@ public class NotificationSettingsService { public static final String NOTIFICATION_USER_NOTIFICATION_SSH_KEY_HAS_EXPIRED = "notification.user-notification.ssh-key-has-expired"; + public static final String NOTIFICATION_USER_NOTIFICATION_VCS_ACCESS_TOKEN_ADDED = "notification.user-notification.vcs-access-token-added"; + + public static final String NOTIFICATION_USER_NOTIFICATION_VCS_ACCESS_TOKEN_EXPIRED = "notification.user-notification.vcs-access-token-expired"; + // if webapp or email is not explicitly set for a specific setting -> no support for this communication channel for this setting // this has to match the properties in the notification settings structure file on the client that hides the related UI elements public static final Set DEFAULT_NOTIFICATION_SETTINGS = new HashSet<>(Arrays.asList( @@ -188,7 +194,9 @@ public class NotificationSettingsService { new NotificationSetting(true, true, true, NOTIFICATION_USER_NOTIFICATION_DATA_EXPORT_CREATED), new NotificationSetting(true, true, false, NOTIFICATION_USER_NOTIFICATION_SSH_KEY_ADDED), new NotificationSetting(true, true, false, NOTIFICATION_USER_NOTIFICATION_SSH_KEY_EXPIRES_SOON), - new NotificationSetting(true, true, false, NOTIFICATION_USER_NOTIFICATION_SSH_KEY_HAS_EXPIRED))); + new NotificationSetting(true, true, false, NOTIFICATION_USER_NOTIFICATION_SSH_KEY_HAS_EXPIRED), + new NotificationSetting(true, true, false, NOTIFICATION_USER_NOTIFICATION_VCS_ACCESS_TOKEN_ADDED), + new NotificationSetting(true, true, false, NOTIFICATION_USER_NOTIFICATION_VCS_ACCESS_TOKEN_EXPIRED))); /** * This is the place where the mapping between SettingId and NotificationTypes happens on the server side @@ -225,7 +233,9 @@ public class NotificationSettingsService { Map.entry(NOTIFICATION__USER_NOTIFICATION__USER_MENTION, new NotificationType[] { CONVERSATION_USER_MENTIONED }), Map.entry(NOTIFICATION_USER_NOTIFICATION_SSH_KEY_ADDED, new NotificationType[] { SSH_KEY_ADDED }), Map.entry(NOTIFICATION_USER_NOTIFICATION_SSH_KEY_EXPIRES_SOON, new NotificationType[] { SSH_KEY_EXPIRES_SOON }), - Map.entry(NOTIFICATION_USER_NOTIFICATION_SSH_KEY_HAS_EXPIRED, new NotificationType[] { SSH_KEY_HAS_EXPIRED })); + Map.entry(NOTIFICATION_USER_NOTIFICATION_SSH_KEY_HAS_EXPIRED, new NotificationType[] { SSH_KEY_HAS_EXPIRED }), + Map.entry(NOTIFICATION_USER_NOTIFICATION_VCS_ACCESS_TOKEN_ADDED, new NotificationType[] { VCS_ACCESS_TOKEN_ADDED }), + Map.entry(NOTIFICATION_USER_NOTIFICATION_VCS_ACCESS_TOKEN_EXPIRED, new NotificationType[] { VCS_ACCESS_TOKEN_EXPIRED })); // This set has to equal the UI configuration in the client notification settings structure file! // More information on supported notification types can be found here: https://docs.artemis.cit.tum.de/user/notifications/ @@ -236,7 +246,7 @@ public class NotificationSettingsService { TUTORIAL_GROUP_DEREGISTRATION_STUDENT, TUTORIAL_GROUP_DEREGISTRATION_TUTOR, TUTORIAL_GROUP_DELETED, TUTORIAL_GROUP_UPDATED, TUTORIAL_GROUP_ASSIGNED, TUTORIAL_GROUP_UNASSIGNED, NEW_EXERCISE_POST, NEW_LECTURE_POST, NEW_REPLY_FOR_LECTURE_POST, NEW_COURSE_POST, NEW_REPLY_FOR_COURSE_POST, NEW_REPLY_FOR_EXERCISE_POST, QUIZ_EXERCISE_STARTED, DATA_EXPORT_CREATED, DATA_EXPORT_FAILED, CONVERSATION_NEW_MESSAGE, CONVERSATION_NEW_REPLY_MESSAGE, SSH_KEY_ADDED, SSH_KEY_EXPIRES_SOON, - SSH_KEY_HAS_EXPIRED); + SSH_KEY_HAS_EXPIRED, VCS_ACCESS_TOKEN_ADDED, VCS_ACCESS_TOKEN_EXPIRED); // More information on supported notification types can be found here: https://docs.artemis.cit.tum.de/user/notifications/ // Please adapt the above docs if you change the supported notification types diff --git a/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/SingleUserNotificationService.java b/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/SingleUserNotificationService.java index f052c6362dea..06382c8dd458 100644 --- a/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/SingleUserNotificationService.java +++ b/src/main/java/de/tum/cit/aet/artemis/communication/service/notifications/SingleUserNotificationService.java @@ -23,6 +23,8 @@ import static de.tum.cit.aet.artemis.communication.domain.NotificationType.TUTORIAL_GROUP_REGISTRATION_STUDENT; import static de.tum.cit.aet.artemis.communication.domain.NotificationType.TUTORIAL_GROUP_REGISTRATION_TUTOR; import static de.tum.cit.aet.artemis.communication.domain.NotificationType.TUTORIAL_GROUP_UNASSIGNED; +import static de.tum.cit.aet.artemis.communication.domain.NotificationType.VCS_ACCESS_TOKEN_ADDED; +import static de.tum.cit.aet.artemis.communication.domain.NotificationType.VCS_ACCESS_TOKEN_EXPIRED; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.CONVERSATION_ADD_USER_CHANNEL_TITLE; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.CONVERSATION_ADD_USER_GROUP_CHAT_TITLE; import static de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants.CONVERSATION_CREATE_GROUP_CHAT_TITLE; @@ -150,6 +152,7 @@ private SingleUserNotification createSingleUserNotification(Object notificationS ((NewReplyNotificationSubject) notificationSubject).responsibleUser); case DATA_EXPORT_CREATED, DATA_EXPORT_FAILED -> createNotification((DataExport) notificationSubject, notificationType, typeSpecificInformation); case SSH_KEY_ADDED, SSH_KEY_EXPIRES_SOON, SSH_KEY_HAS_EXPIRED -> createNotification((UserSshPublicKey) notificationSubject, notificationType, typeSpecificInformation); + case VCS_ACCESS_TOKEN_ADDED, VCS_ACCESS_TOKEN_EXPIRED -> createNotification(typeSpecificInformation.getVcsAccessToken(), notificationType, typeSpecificInformation); default -> throw new UnsupportedOperationException("Can not create notification for type : " + notificationType); }; } @@ -282,7 +285,7 @@ public void notifyUserAboutSoonExpiringSshKey(User recipient, UserSshPublicKey k } /** - * Notify user about an upcoming expiry of an SSH key + * Notify user about the expiration of an SSH key * * @param recipient the user to whose account the SSH key was added * @param key the key which was added @@ -291,6 +294,24 @@ public void notifyUserAboutExpiredSshKey(User recipient, UserSshPublicKey key) { notifyRecipientWithNotificationType(key, SSH_KEY_HAS_EXPIRED, recipient, null); } + /** + * Notify user about the addition of a VCS access token + * + * @param recipient the user to whose account the SSH key was added + */ + public void notifyUserAboutNewlyAddedVcsAccessToken(User recipient) { + notifyRecipientWithNotificationType(null, VCS_ACCESS_TOKEN_ADDED, recipient, null); + } + + /** + * Notify user about the expiration of the VCS access token + * + * @param recipient the user to whose account the SSH key was added + */ + public void notifyUserAboutExpiredVcsAccessToken(User recipient) { + notifyRecipientWithNotificationType(null, VCS_ACCESS_TOKEN_EXPIRED, recipient, null); + } + /** * Notify student about possible plagiarism case. * 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 e2a8206a1b78..5846c4c449a2 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 @@ -85,6 +85,8 @@ public interface UserRepository extends ArtemisJpaRepository, JpaSpe Optional findOneByEmailIgnoreCase(String email); + List findByvcsAccessTokenExpiryDateBetween(ZonedDateTime from, ZonedDateTime to); + @EntityGraph(type = LOAD, attributePaths = { "groups" }) Optional findOneWithGroupsByEmailIgnoreCase(String email); diff --git a/src/main/java/de/tum/cit/aet/artemis/core/web/AccountResource.java b/src/main/java/de/tum/cit/aet/artemis/core/web/AccountResource.java index 16c94629047c..8b8989827483 100644 --- a/src/main/java/de/tum/cit/aet/artemis/core/web/AccountResource.java +++ b/src/main/java/de/tum/cit/aet/artemis/core/web/AccountResource.java @@ -26,6 +26,7 @@ import org.springframework.web.bind.annotation.RestController; import org.springframework.web.multipart.MultipartFile; +import de.tum.cit.aet.artemis.communication.service.notifications.SingleUserNotificationService; import de.tum.cit.aet.artemis.core.domain.User; import de.tum.cit.aet.artemis.core.dto.PasswordChangeDTO; import de.tum.cit.aet.artemis.core.dto.UserDTO; @@ -64,15 +65,18 @@ public class AccountResource { private final FileService fileService; + private final SingleUserNotificationService singleUserNotificationService; + private static final float MAX_PROFILE_PICTURE_FILESIZE_IN_MEGABYTES = 0.1f; - public AccountResource(UserRepository userRepository, UserService userService, UserCreationService userCreationService, AccountService accountService, - FileService fileService) { + public AccountResource(UserRepository userRepository, UserService userService, UserCreationService userCreationService, AccountService accountService, FileService fileService, + SingleUserNotificationService singleUserNotificationService) { this.userRepository = userRepository; this.userService = userService; this.userCreationService = userCreationService; this.accountService = accountService; this.fileService = fileService; + this.singleUserNotificationService = singleUserNotificationService; } /** @@ -140,6 +144,7 @@ public ResponseEntity createVcsAccessToken(@RequestParam("expiryDate") userRepository.updateUserVcsAccessToken(user.getId(), LocalVCPersonalAccessTokenManagementService.generateSecureVCSAccessToken(), expiryDate); log.debug("Successfully created a VCS access token for user {}", user.getLogin()); + singleUserNotificationService.notifyUserAboutNewlyAddedVcsAccessToken(user); user = userRepository.getUser(); UserDTO userDTO = new UserDTO(); userDTO.setLogin(user.getLogin()); diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/tokens/UserTokenExpiryNotificationService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/tokens/UserTokenExpiryNotificationService.java new file mode 100644 index 000000000000..f5759b4c368a --- /dev/null +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/tokens/UserTokenExpiryNotificationService.java @@ -0,0 +1,64 @@ +package de.tum.cit.aet.artemis.programming.service.tokens; + +import static de.tum.cit.aet.artemis.core.config.Constants.PROFILE_SCHEDULING; +import static java.time.ZonedDateTime.now; + +import java.time.ZonedDateTime; +import java.util.function.Function; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.context.annotation.Profile; +import org.springframework.scheduling.annotation.Scheduled; +import org.springframework.stereotype.Service; + +import de.tum.cit.aet.artemis.communication.service.notifications.SingleUserNotificationService; +import de.tum.cit.aet.artemis.core.domain.User; +import de.tum.cit.aet.artemis.core.repository.UserRepository; + +@Profile(PROFILE_SCHEDULING) +@Service +public class UserTokenExpiryNotificationService { + + private static final Logger log = LoggerFactory.getLogger(UserTokenExpiryNotificationService.class); + + private final SingleUserNotificationService singleUserNotificationService; + + private final UserRepository userRepository; + + public UserTokenExpiryNotificationService(SingleUserNotificationService singleUserNotificationService, UserRepository userRepository) { + this.singleUserNotificationService = singleUserNotificationService; + this.userRepository = userRepository; + } + + /** + * Schedules VCS access token expiry notifications to users every morning at 6:00:00 am + */ + @Scheduled(cron = "0 * * * * *") + public void sendTokenExpirationNotifications() { + log.info("Sending Token expiration notifications to single user"); + notifyOnExpiredToken(); + } + + /** + * Notifies the user at the day of key expiry, that the key has expired + */ + public void notifyOnExpiredToken() { + notifyUsersForKeyExpiryWindow(now().minusDays(1), now(), (user) -> { + singleUserNotificationService.notifyUserAboutExpiredVcsAccessToken(user); + return null; + }); + } + + /** + * Notifies users whose SSH keys are expiring within the specified date range, with the notification specified by the + * notifyFunction + * + * @param fromDate the start of the expiry date range + * @param toDate the end of the expiry date range + * @param notifyFunction a function to handle user notification + */ + private void notifyUsersForKeyExpiryWindow(ZonedDateTime fromDate, ZonedDateTime toDate, Function notifyFunction) { + userRepository.findByvcsAccessTokenExpiryDateBetween(fromDate, toDate).forEach(notifyFunction::apply); + } +} From ce143013975722dd2e34220d29809e8652f4d7d6 Mon Sep 17 00:00:00 2001 From: entholzer Date: Sat, 21 Dec 2024 12:32:51 +0100 Subject: [PATCH 5/8] changed cron job task back to 6am --- .../service/tokens/UserTokenExpiryNotificationService.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/java/de/tum/cit/aet/artemis/programming/service/tokens/UserTokenExpiryNotificationService.java b/src/main/java/de/tum/cit/aet/artemis/programming/service/tokens/UserTokenExpiryNotificationService.java index f5759b4c368a..20753cf486bf 100644 --- a/src/main/java/de/tum/cit/aet/artemis/programming/service/tokens/UserTokenExpiryNotificationService.java +++ b/src/main/java/de/tum/cit/aet/artemis/programming/service/tokens/UserTokenExpiryNotificationService.java @@ -34,7 +34,7 @@ public UserTokenExpiryNotificationService(SingleUserNotificationService singleUs /** * Schedules VCS access token expiry notifications to users every morning at 6:00:00 am */ - @Scheduled(cron = "0 * * * * *") + @Scheduled(cron = "0 0 6 * * *") public void sendTokenExpirationNotifications() { log.info("Sending Token expiration notifications to single user"); notifyOnExpiredToken(); From 5bf95eafeb8b582493033ee437239838eceb00a5 Mon Sep 17 00:00:00 2001 From: entholzer Date: Sun, 22 Dec 2024 09:10:47 +0100 Subject: [PATCH 6/8] fix notification translation --- src/main/webapp/i18n/de/notification.json | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/src/main/webapp/i18n/de/notification.json b/src/main/webapp/i18n/de/notification.json index d98ca682a349..51f46fc033a0 100644 --- a/src/main/webapp/i18n/de/notification.json +++ b/src/main/webapp/i18n/de/notification.json @@ -71,9 +71,7 @@ "courseArchiveFailed": "Kursarchivierung fehlgeschlagen", "examArchiveStarted": "Klausurarchivierung gestartet", "examArchiveFinished": "Klausurarchivierung beendet", - "examArchiveFailed": "Klausurarchivierung fehlgeschlagen", - "vcsAccessTokenAdded": "Neues VCS Zugriffstoken hinzugefügt", - "vcsAccessTokenExpired": "VCS Zugriffstoken abgelaufen" + "examArchiveFailed": "Klausurarchivierung fehlgeschlagen" }, "text": { "attachmentChange": "Der Anhang \"{{ placeholderValues.1 }}\" von \"{{ placeholderValues.2 }}\" im Kurs \"{{ placeholderValues.0}}\" wurde aktualisiert.", @@ -99,9 +97,7 @@ "examArchiveStarted": "Die Klausur \"{{ placeholderValues.1 }}\" wird archiviert.", "examArchiveFinishedWithoutErrors": "Die Klausur \"{{ placeholderValues.1 }}\" wurde archiviert.", "examArchiveFinishedWithErrors": "Die Klausur \"{{ placeholderValues.1 }}\" wurde archiviert. Einige Aufgaben konnten nicht in das Archiv aufgenommen werden:

{{ placeholderValues.2 }}", - "examArchiveFailed": "Es gab ein Problem beim Archivieren der Klausur \"{{ placeholderValues.1 }}\":

{{ placeholderValues.2 }}", - "vcsAccessTokenAdded": "Du hast erfolgreich ein persönliches Zugriffstoken hinzugefügt.", - "vcsAccessTokenExpired": "Dein persönliches Zugriffstoken ist abgelaufen." + "examArchiveFailed": "Es gab ein Problem beim Archivieren der Klausur \"{{ placeholderValues.1 }}\":

{{ placeholderValues.2 }}" } }, "singleUserNotification": { @@ -130,7 +126,9 @@ "dataExportFailed": "Datenexport fehlgeschlagen", "sshKeyAdded": "Neuer SSH-Schlüssel hinzugefügt", "sshKeyExpiresSoon": "SSH-Schlüssel läuft bald ab", - "sshKeyHasExpired": "SSH-Schlüssel ist abgelaufen" + "sshKeyHasExpired": "SSH-Schlüssel ist abgelaufen", + "vcsAccessTokenAdded": "Neues VCS Zugriffstoken hinzugefügt", + "vcsAccessTokenExpired": "VCS Zugriffstoken abgelaufen" }, "text": { "newReplyForExercisePost": "Auf deinen Beitrag zur Aufgabe \"{{ placeholderValues.8 }}\" im Kurs \"{{ placeholderValues.0 }}\" wurde geantwortet: \"{{ placeholderValues.5 }}\"", @@ -160,7 +158,9 @@ "dataExportFailed": "Dein Datenexport konnte nicht erstellt werden. Bitte versuche es später erneut.", "sshKeyAdded": "Du hast erfolgreich einen SSH-Schlüssel hinzugefügt", "sshKeyExpiresSoon": "Dein SSH-Schlüssel mit dem Label \"{{ placeholderValues.0}}\" läuft am {{ placeholderValues.1 }} ab.", - "sshKeyHasExpired": "Dein SSH-Schlüssel mit dem Label \"{{ placeholderValues.0}}\" ist am {{ placeholderValues.1 }} abgelaufen." + "sshKeyHasExpired": "Dein SSH-Schlüssel mit dem Label \"{{ placeholderValues.0}}\" ist am {{ placeholderValues.1 }} abgelaufen.", + "vcsAccessTokenAdded": "Du hast erfolgreich ein persönliches Zugriffstoken hinzugefügt.", + "vcsAccessTokenExpired": "Dein persönliches Zugriffstoken ist abgelaufen." } }, "tutorialGroupNotification": { From 0997b6a04afc20af192161210ce9e7c655784056 Mon Sep 17 00:00:00 2001 From: entholzer Date: Fri, 3 Jan 2025 14:42:37 +0100 Subject: [PATCH 7/8] fixed and improved tests by introducing test repository --- .../SingleUserNotificationServiceTest.java | 48 +++++++++---------- .../NotificationTestRepository.java | 19 ++++++++ .../core/user/util/UserTestService.java | 5 ++ 3 files changed, 48 insertions(+), 24 deletions(-) create mode 100644 src/test/java/de/tum/cit/aet/artemis/core/test_repository/NotificationTestRepository.java diff --git a/src/test/java/de/tum/cit/aet/artemis/communication/notification/SingleUserNotificationServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/communication/notification/SingleUserNotificationServiceTest.java index 956696bfd53b..3a69cc39816f 100644 --- a/src/test/java/de/tum/cit/aet/artemis/communication/notification/SingleUserNotificationServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/communication/notification/SingleUserNotificationServiceTest.java @@ -89,7 +89,6 @@ import de.tum.cit.aet.artemis.communication.domain.conversation.OneToOneChat; import de.tum.cit.aet.artemis.communication.domain.notification.Notification; import de.tum.cit.aet.artemis.communication.domain.notification.SingleUserNotification; -import de.tum.cit.aet.artemis.communication.repository.NotificationRepository; import de.tum.cit.aet.artemis.communication.repository.NotificationSettingRepository; import de.tum.cit.aet.artemis.communication.service.notifications.SingleUserNotificationService; import de.tum.cit.aet.artemis.core.domain.Course; @@ -97,6 +96,7 @@ import de.tum.cit.aet.artemis.core.domain.DomainObject; import de.tum.cit.aet.artemis.core.domain.User; import de.tum.cit.aet.artemis.core.security.SecurityUtils; +import de.tum.cit.aet.artemis.core.test_repository.NotificationTestRepository; import de.tum.cit.aet.artemis.core.user.util.UserUtilService; import de.tum.cit.aet.artemis.core.util.CourseUtilService; import de.tum.cit.aet.artemis.exercise.domain.Exercise; @@ -127,7 +127,7 @@ class SingleUserNotificationServiceTest extends AbstractSpringIntegrationIndepen private SingleUserNotificationService singleUserNotificationService; @Autowired - private NotificationRepository notificationRepository; + private NotificationTestRepository notificationTestRepository; @Autowired private NotificationSettingRepository notificationSettingRepository; @@ -217,7 +217,7 @@ void setUp() { userTwo = userUtilService.getUserByLogin(TEST_PREFIX + "student2"); userThree = userUtilService.getUserByLogin(TEST_PREFIX + "student3"); - notificationRepository.deleteAllInBatch(); + notificationTestRepository.deleteAllInBatch(); exercise = new TextExercise(); exercise.setCourse(course); @@ -304,7 +304,7 @@ void setUp() { * @param expectedNotificationTitle is the title (NotificationTitleTypeConstants) of the expected notification */ private void verifyRepositoryCallWithCorrectNotification(String expectedNotificationTitle) { - List capturedNotifications = notificationRepository.findAll(); + List capturedNotifications = notificationTestRepository.findAll(); assertThat(capturedNotifications).isNotEmpty(); List relevantNotifications = capturedNotifications.stream().filter(e -> e.getTitle().equals(expectedNotificationTitle)).toList(); assertThat(relevantNotifications).as("Title of the captured notification should be equal to the expected one").hasSize(1); @@ -319,13 +319,13 @@ private void verifyRepositoryCallWithCorrectNotification(String expectedNotifica @Test void testSendNoNotificationOrEmailWhenSettingsAreDeactivated() { notificationSettingRepository.save(new NotificationSetting(user, false, true, true, NOTIFICATION__EXERCISE_NOTIFICATION__NEW_REPLY_FOR_EXERCISE_POST)); - assertThat(notificationRepository.findAll()).as("No notifications should be present prior to the method call").isEmpty(); + assertThat(notificationTestRepository.findAll()).as("No notifications should be present prior to the method call").isEmpty(); SingleUserNotification notification = singleUserNotificationService.createNotificationAboutNewMessageReply(answerPost, answerPost.getAuthor(), answerPost.getPost().getConversation()); singleUserNotificationService.notifyUserAboutNewMessageReply(answerPost, notification, user, userTwo, NEW_REPLY_FOR_EXERCISE_POST); - assertThat(notificationRepository.findAll()).as("The notification should have been saved to the DB").hasSize(1); + assertThat(notificationTestRepository.findAll()).as("The notification should have been saved to the DB").hasSize(1); // no web app notification or email should be sent verify(websocketMessagingService, never()).sendMessage(any(), any()); } @@ -364,7 +364,7 @@ void testNotifyUserAboutAssessedExerciseSubmission() { void testCheckNotificationForAssessmentExerciseSubmission_pastAssessmentDueDate() { exercise = TextExerciseFactory.generateTextExercise(null, null, ZonedDateTime.now().minusMinutes(1), course); singleUserNotificationService.checkNotificationForAssessmentExerciseSubmission(exercise, user, result); - assertThat(notificationRepository.findAll()).as("One new notification should have been created").hasSize(1); + assertThat(notificationTestRepository.findAll()).as("One new notification should have been created").hasSize(1); } /** @@ -374,7 +374,7 @@ void testCheckNotificationForAssessmentExerciseSubmission_pastAssessmentDueDate( void testCheckNotificationForAssessmentExerciseSubmission_futureAssessmentDueDate() { exercise = TextExerciseFactory.generateTextExercise(null, null, ZonedDateTime.now().plusHours(1), course); singleUserNotificationService.checkNotificationForAssessmentExerciseSubmission(exercise, user, result); - assertThat(notificationRepository.findAll()).as("No new notification should have been created").isEmpty(); + assertThat(notificationTestRepository.findAll()).as("No new notification should have been created").isEmpty(); } @Test @@ -397,7 +397,7 @@ void testNotifyUsersAboutAssessedExerciseSubmission() { singleUserNotificationService.notifyUsersAboutAssessedExerciseSubmission(testExercise); - List sentNotifications = notificationRepository.findAll(); + List sentNotifications = notificationTestRepository.findAll(); assertThat(sentNotifications).as("Only one notification should have been created (for the user with a valid participation, submission, and manual result)").hasSize(1); assertThat(sentNotifications.getFirst()).isInstanceOf(SingleUserNotification.class); @@ -428,7 +428,7 @@ void shouldNotifyUserAboutNewlyCreatedSshKeyWithExpirationDate() throws GeneralS userSshPublicKeyService.createSshKeyForUser(user, AuthorizedKeyEntry.parseAuthorizedKeyEntry(keyDTO.publicKey()), keyDTO); - sentNotifications = notificationRepository.findAll(); + sentNotifications = notificationTestRepository.findAllByRecipientId(user.getId()); checkFirstNotification(); } @@ -438,7 +438,7 @@ void shouldNotifyUserAboutNewlyCreatedSshKeyWithNoDate() throws GeneralSecurityE userSshPublicKeyService.createSshKeyForUser(user, AuthorizedKeyEntry.parseAuthorizedKeyEntry(keyDTO.publicKey()), keyDTO); - sentNotifications = notificationRepository.findAll(); + sentNotifications = notificationTestRepository.findAllByRecipientId(user.getId()); checkFirstNotification(); } @@ -449,7 +449,7 @@ void shouldNotifyUserAboutUpcomingSshKeyExpiry() throws GeneralSecurityException userSshPublicKeyExpiryNotificationService.notifyUserOnUpcomingKeyExpiry(); - sentNotifications = notificationRepository.findAll(); + sentNotifications = notificationTestRepository.findAllByRecipientId(user.getId()); assertThat(sentNotifications).hasSize(2); assertThat(((SingleUserNotification) sentNotifications.getFirst()).getRecipient()).isEqualTo(user); assertThat((sentNotifications.get(1)).getText()).isEqualTo(SSH_KEY_EXPIRES_SOON_TEXT); @@ -463,7 +463,7 @@ void shouldNotifyUserAboutExpiredSshKey() throws GeneralSecurityException, IOExc userSshPublicKeyExpiryNotificationService.notifyUserOnExpiredKey(); - sentNotifications = notificationRepository.findAll(); + sentNotifications = notificationTestRepository.findAllByRecipientId(user.getId()); assertThat(sentNotifications).hasSize(2); assertThat(((SingleUserNotification) sentNotifications.getFirst()).getRecipient()).isEqualTo(user); assertThat((sentNotifications.get(1)).getText()).isEqualTo(SSH_KEY_HAS_EXPIRED_TEXT); @@ -477,7 +477,7 @@ void shouldNotNotifyUserAboutUpcomingSshKeyExpiryWhenKeyDoesNotExpireSoon() thro userSshPublicKeyExpiryNotificationService.notifyUserOnUpcomingKeyExpiry(); - sentNotifications = notificationRepository.findAll(); + sentNotifications = notificationTestRepository.findAllByRecipientId(user.getId()); assertThat(sentNotifications).hasSize(1); checkFirstNotification(); } @@ -489,7 +489,7 @@ void shouldNotNotifyUserAboutExpiredSshKeyWhenKeyIsNotExpired() throws GeneralSe userSshPublicKeyExpiryNotificationService.notifyUserOnExpiredKey(); - sentNotifications = notificationRepository.findAll(); + sentNotifications = notificationTestRepository.findAllByRecipientId(user.getId()); assertThat(sentNotifications).hasSize(1); checkFirstNotification(); } @@ -498,7 +498,7 @@ void shouldNotNotifyUserAboutExpiredSshKeyWhenKeyIsNotExpired() throws GeneralSe void scheduleKeyExpiryNotifications() { userSshPublicKeyExpiryNotificationService.sendKeyExpirationNotifications(); - sentNotifications = notificationRepository.findAll(); + sentNotifications = notificationTestRepository.findAllByRecipientId(user.getId()); assertThat(sentNotifications).hasSize(0); } @@ -527,7 +527,7 @@ void tearDown() throws Exception { void shouldNotifyUserAboutNewlyAddedVcsAccessToken() { singleUserNotificationService.notifyUserAboutNewlyAddedVcsAccessToken(user); - sentNotifications = notificationRepository.findAll(); + sentNotifications = notificationTestRepository.findAll(); assertThat(sentNotifications.getFirst()).isInstanceOf(SingleUserNotification.class); assertThat(((SingleUserNotification) sentNotifications.getFirst()).getRecipient()).isEqualTo(user); assertThat((sentNotifications.getFirst()).getText()).isEqualTo(VCS_ACCESS_TOKEN_ADDED_TEXT); @@ -541,7 +541,7 @@ void shouldNotifyUserAboutExpiredVcsAccessToken() { userTokenExpiryNotificationService.sendTokenExpirationNotifications(); - sentNotifications = notificationRepository.findAll(); + sentNotifications = notificationTestRepository.findAll(); assertThat(sentNotifications).hasSize(1); assertThat(sentNotifications.getFirst()).isInstanceOf(SingleUserNotification.class); assertThat(((SingleUserNotification) sentNotifications.getFirst()).getRecipient()).isEqualTo(user); @@ -556,7 +556,7 @@ void shouldNotNotifyUserAboutVcsAccessTokenExpiryWhenTokenIsNotExpired() { userTokenExpiryNotificationService.sendTokenExpirationNotifications(); - sentNotifications = notificationRepository.findAll(); + sentNotifications = notificationTestRepository.findAll(); assertThat(sentNotifications).hasSize(0); } } @@ -600,9 +600,9 @@ void testNotifyUserAboutFinalPlagiarismState() throws MessagingException, IOExce @Test void testConversationNotificationsOneToOneChatCreation() { - var notificationsBefore = (int) notificationRepository.count(); + var notificationsBefore = (int) notificationTestRepository.count(); singleUserNotificationService.notifyClientAboutConversationCreationOrDeletion(oneToOneChat, user, userTwo, CONVERSATION_CREATE_ONE_TO_ONE_CHAT); - List capturedNotifications = notificationRepository.findAll(); + List capturedNotifications = notificationTestRepository.findAll(); assertThat(capturedNotifications).as("Notification should not have been saved").hasSize(notificationsBefore); // notification should be sent verify(websocketMessagingService).sendMessage(eq("/topic/user/" + user.getId() + "/notifications"), (Object) any()); @@ -610,14 +610,14 @@ void testConversationNotificationsOneToOneChatCreation() { @Test void testConversationNotificationsGroupChatCreation() { - int notificationsBefore = (int) notificationRepository.count(); + int notificationsBefore = (int) notificationTestRepository.count(); singleUserNotificationService.notifyClientAboutConversationCreationOrDeletion(groupChat, user, userTwo, CONVERSATION_CREATE_GROUP_CHAT); verify(websocketMessagingService).sendMessage(eq("/topic/user/" + user.getId() + "/notifications"), (Object) any()); singleUserNotificationService.notifyClientAboutConversationCreationOrDeletion(groupChat, userThree, userTwo, CONVERSATION_CREATE_GROUP_CHAT); verify(websocketMessagingService).sendMessage(eq("/topic/user/" + userThree.getId() + "/notifications"), (Object) any()); - List capturedNotifications = notificationRepository.findAll(); + List capturedNotifications = notificationTestRepository.findAll(); assertThat(capturedNotifications).as("Both notifications should have been saved").hasSize(notificationsBefore + 2); capturedNotifications.forEach(capturedNotification -> { assertThat(capturedNotification.getTitle()).as("Title of the captured notification should be equal to the expected one") @@ -661,7 +661,7 @@ void testConversationNotificationsNewMessageReply() { answerPost.getPost().getConversation()); singleUserNotificationService.notifyUserAboutNewMessageReply(answerPost, notification, user, userTwo, CONVERSATION_NEW_REPLY_MESSAGE); verify(websocketMessagingService, never()).sendMessage(eq("/topic/user/" + user.getId() + "/notifications"), (Object) any()); - Notification sentNotification = notificationRepository.findAll().stream().max(Comparator.comparing(DomainObject::getId)).orElseThrow(); + Notification sentNotification = notificationTestRepository.findAll().stream().max(Comparator.comparing(DomainObject::getId)).orElseThrow(); SingleUserNotificationService.NewReplyNotificationSubject notificationSubject = new SingleUserNotificationService.NewReplyNotificationSubject(answerPost, user, userTwo); verify(generalInstantNotificationService, times(1)).sendNotification(sentNotification, user, notificationSubject); diff --git a/src/test/java/de/tum/cit/aet/artemis/core/test_repository/NotificationTestRepository.java b/src/test/java/de/tum/cit/aet/artemis/core/test_repository/NotificationTestRepository.java new file mode 100644 index 000000000000..89354e5fae86 --- /dev/null +++ b/src/test/java/de/tum/cit/aet/artemis/core/test_repository/NotificationTestRepository.java @@ -0,0 +1,19 @@ +package de.tum.cit.aet.artemis.core.test_repository; + +import java.util.List; + +import org.springframework.data.jpa.repository.Query; + +import de.tum.cit.aet.artemis.communication.domain.notification.Notification; +import de.tum.cit.aet.artemis.communication.repository.NotificationRepository; + +public interface NotificationTestRepository extends NotificationRepository { + + @Query(""" + SELECT notification + FROM Notification notification + LEFT JOIN TREAT(notification AS SingleUserNotification).recipient recipient + WHERE recipient.id = :recipientId + """) + List findAllByRecipientId(long recipientId); +} diff --git a/src/test/java/de/tum/cit/aet/artemis/core/user/util/UserTestService.java b/src/test/java/de/tum/cit/aet/artemis/core/user/util/UserTestService.java index 3f4606f09e63..1702f990a1d7 100644 --- a/src/test/java/de/tum/cit/aet/artemis/core/user/util/UserTestService.java +++ b/src/test/java/de/tum/cit/aet/artemis/core/user/util/UserTestService.java @@ -41,6 +41,7 @@ import de.tum.cit.aet.artemis.core.security.Role; import de.tum.cit.aet.artemis.core.service.user.PasswordService; import de.tum.cit.aet.artemis.core.test_repository.CourseTestRepository; +import de.tum.cit.aet.artemis.core.test_repository.NotificationTestRepository; import de.tum.cit.aet.artemis.core.test_repository.UserTestRepository; import de.tum.cit.aet.artemis.core.util.CourseUtilService; import de.tum.cit.aet.artemis.core.util.RequestUtilService; @@ -137,6 +138,9 @@ public class UserTestService { @Autowired private ExerciseTestRepository exerciseTestRepository; + @Autowired + private NotificationTestRepository notificationTestRepository; + public void setup(String testPrefix, MockDelegate mockDelegate) throws Exception { this.TEST_PREFIX = testPrefix; this.mockDelegate = mockDelegate; @@ -156,6 +160,7 @@ public void setup(String testPrefix, MockDelegate mockDelegate) throws Exception } public void tearDown() throws IOException { + notificationTestRepository.deleteAllInBatch(notificationTestRepository.findAllByRecipientId(student.getId())); userTestRepository.deleteAll(userTestRepository.searchAllByLoginOrName(Pageable.unpaged(), TEST_PREFIX)); } From 4cea0f9285cc11ac9cfd1ace9e10544d7080745e Mon Sep 17 00:00:00 2001 From: entholzer Date: Fri, 3 Jan 2025 16:27:09 +0100 Subject: [PATCH 8/8] fixed architecture tests --- .../GroupNotificationServiceTest.java | 16 ++--- .../NotificationResourceIntegrationTest.java | 60 +++++++++---------- .../NotificationScheduleServiceTest.java | 10 ++-- .../ConversationNotificationServiceTest.java | 6 +- .../NotificationTestRepository.java | 4 ++ .../core/user/util/UserTestService.java | 4 +- 6 files changed, 53 insertions(+), 47 deletions(-) diff --git a/src/test/java/de/tum/cit/aet/artemis/communication/notification/GroupNotificationServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/communication/notification/GroupNotificationServiceTest.java index 4ccf0116d463..2f109cb2d32e 100644 --- a/src/test/java/de/tum/cit/aet/artemis/communication/notification/GroupNotificationServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/communication/notification/GroupNotificationServiceTest.java @@ -48,11 +48,11 @@ import de.tum.cit.aet.artemis.communication.domain.Post; import de.tum.cit.aet.artemis.communication.domain.conversation.Channel; import de.tum.cit.aet.artemis.communication.domain.notification.Notification; -import de.tum.cit.aet.artemis.communication.repository.NotificationRepository; import de.tum.cit.aet.artemis.communication.repository.NotificationSettingRepository; 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.test_repository.NotificationTestRepository; import de.tum.cit.aet.artemis.core.test_repository.UserTestRepository; import de.tum.cit.aet.artemis.core.user.util.UserUtilService; import de.tum.cit.aet.artemis.core.util.CourseUtilService; @@ -76,7 +76,7 @@ class GroupNotificationServiceTest extends AbstractSpringIntegrationIndependentT private static final String TEST_PREFIX = "groupnotificationservice"; @Autowired - private NotificationRepository notificationRepository; + private NotificationTestRepository notificationTestRepository; @Autowired private NotificationSettingRepository notificationSettingRepository; @@ -231,12 +231,12 @@ void setUp() { userUtilService.changeUser(TEST_PREFIX + "instructor1"); // store the current notification count to let tests work even if notifications are created in other tests - notificationCountBeforeTest = notificationRepository.findAll().size(); + notificationCountBeforeTest = notificationTestRepository.findAll().size(); } @AfterEach void tearDown() { - notificationRepository.deleteAllInBatch(); + notificationTestRepository.deleteAllInBatch(); } /** @@ -262,10 +262,10 @@ private Notification verifyRepositoryCallWithCorrectNotificationAndReturnNotific */ private Notification verifyRepositoryCallWithCorrectNotificationAndReturnNotificationAtIndex(int numberOfGroupsAndCalls, String expectedNotificationTitle, int index) { await().untilAsserted( - () -> assertThat(notificationRepository.findAll()).as("The number of created notifications should be the same as the number of notified groups/authorities") + () -> assertThat(notificationTestRepository.findAll()).as("The number of created notifications should be the same as the number of notified groups/authorities") .hasSize(numberOfGroupsAndCalls + notificationCountBeforeTest)); - List capturedNotifications = notificationRepository.findAll(); + List capturedNotifications = notificationTestRepository.findAll(); Notification lastCapturedNotification = capturedNotifications.get(capturedNotifications.size() - 1); assertThat(lastCapturedNotification.getTitle()).as("The title of the captured notification should be equal to the expected one").isEqualTo(expectedNotificationTitle); @@ -440,10 +440,10 @@ private void verifyPush(Notification notification, Set users, Object notif */ @Test void testNotifyStudentGroupAboutAttachmentChange_futureReleaseDate() { - var countBefore = notificationRepository.count(); + var countBefore = notificationTestRepository.count(); attachment.setReleaseDate(FUTURE_TIME); groupNotificationService.notifyStudentGroupAboutAttachmentChange(attachment, NOTIFICATION_TEXT); - var countAfter = notificationRepository.count(); + var countAfter = notificationTestRepository.count(); assertThat(countAfter).as("No notification should be created/saved").isEqualTo(countBefore); } diff --git a/src/test/java/de/tum/cit/aet/artemis/communication/notification/NotificationResourceIntegrationTest.java b/src/test/java/de/tum/cit/aet/artemis/communication/notification/NotificationResourceIntegrationTest.java index 07c00a1b9a3a..d01fd96749fd 100644 --- a/src/test/java/de/tum/cit/aet/artemis/communication/notification/NotificationResourceIntegrationTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/communication/notification/NotificationResourceIntegrationTest.java @@ -20,17 +20,17 @@ import de.tum.cit.aet.artemis.communication.domain.notification.NotificationConstants; import de.tum.cit.aet.artemis.communication.domain.notification.SingleUserNotification; import de.tum.cit.aet.artemis.communication.notification.util.NotificationFactory; -import de.tum.cit.aet.artemis.communication.repository.NotificationRepository; import de.tum.cit.aet.artemis.communication.repository.NotificationSettingRepository; 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.test_repository.NotificationTestRepository; import de.tum.cit.aet.artemis.shared.base.AbstractSpringIntegrationIndependentTest; import de.tum.cit.aet.artemis.text.util.TextExerciseUtilService; class NotificationResourceIntegrationTest extends AbstractSpringIntegrationIndependentTest { @Autowired - private NotificationRepository notificationRepository; + private NotificationTestRepository notificationTestRepository; @Autowired private NotificationSettingRepository notificationSettingRepository; @@ -49,7 +49,7 @@ void initTestCase() { userUtilService.addUsers(TEST_PREFIX, 2, 1, 1, 1); course1 = textExerciseUtilService.addCourseWithOneReleasedTextExercise(); course2 = textExerciseUtilService.addCourseWithOneReleasedTextExercise(); - notificationRepository.deleteAll(); + notificationTestRepository.deleteAll(); User student1 = userUtilService.getUserByLogin(TEST_PREFIX + "student1"); student1.setLastNotificationRead(ZonedDateTime.now().minusDays(1)); @@ -58,7 +58,7 @@ void initTestCase() { @AfterEach void tearDown() { - notificationRepository.deleteAll(); + notificationTestRepository.deleteAll(); } @Test @@ -66,9 +66,9 @@ void tearDown() { void testGetNotifications_recipientEvaluation() throws Exception { User recipient = userTestRepository.getUser(); SingleUserNotification notification1 = NotificationFactory.generateSingleUserNotification(ZonedDateTime.now(), recipient); - notificationRepository.save(notification1); + notificationTestRepository.save(notification1); SingleUserNotification notification2 = NotificationFactory.generateSingleUserNotification(ZonedDateTime.now(), userUtilService.getUserByLogin(TEST_PREFIX + "student2")); - notificationRepository.save(notification2); + notificationTestRepository.save(notification2); List notifications = request.getList("/api/notifications", HttpStatus.OK, Notification.class); assertThat(notifications).as("Notification with recipient equal to current user is returned").contains(notification1); @@ -81,11 +81,11 @@ void testGetNotifications_courseEvaluation() throws Exception { // student1 is member of `testgroup` and `tumuser` per default // the studentGroupName of course1 is `tumuser` per default GroupNotification notification1 = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.STUDENT); - notificationRepository.save(notification1); + notificationTestRepository.save(notification1); course2.setStudentGroupName("some-group"); courseRepository.save(course2); GroupNotification notification2 = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course2, GroupNotificationType.STUDENT); - notificationRepository.save(notification2); + notificationTestRepository.save(notification2); List notifications = request.getList("/api/notifications", HttpStatus.OK, Notification.class); assertThat(notifications).as("Notification with course the current user belongs to is returned").contains(notification1); @@ -96,13 +96,13 @@ void testGetNotifications_courseEvaluation() throws Exception { @WithMockUser(username = TEST_PREFIX + "student1", roles = "USER") void testGetNotifications_groupNotificationTypeEvaluation_asStudent() throws Exception { GroupNotification notificationStudent = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.STUDENT); - notificationRepository.save(notificationStudent); + notificationTestRepository.save(notificationStudent); GroupNotification notificationTutor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.TA); - notificationRepository.save(notificationTutor); + notificationTestRepository.save(notificationTutor); GroupNotification notificationEditor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.EDITOR); - notificationRepository.save(notificationEditor); + notificationTestRepository.save(notificationEditor); GroupNotification notificationInstructor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.INSTRUCTOR); - notificationRepository.save(notificationInstructor); + notificationTestRepository.save(notificationInstructor); List notifications = request.getList("/api/notifications", HttpStatus.OK, Notification.class); assertThat(notifications).as("Notification with type student is returned").contains(notificationStudent); @@ -115,13 +115,13 @@ void testGetNotifications_groupNotificationTypeEvaluation_asStudent() throws Exc @WithMockUser(username = TEST_PREFIX + "tutor1", roles = "TA") void testGetNotifications_groupNotificationTypeEvaluation_asTutor() throws Exception { GroupNotification notificationStudent = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.STUDENT); - notificationRepository.save(notificationStudent); + notificationTestRepository.save(notificationStudent); GroupNotification notificationTutor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.TA); - notificationRepository.save(notificationTutor); + notificationTestRepository.save(notificationTutor); GroupNotification notificationEditor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.EDITOR); - notificationRepository.save(notificationEditor); + notificationTestRepository.save(notificationEditor); GroupNotification notificationInstructor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.INSTRUCTOR); - notificationRepository.save(notificationInstructor); + notificationTestRepository.save(notificationInstructor); List notifications = request.getList("/api/notifications", HttpStatus.OK, Notification.class); assertThat(notifications).as("Notification with type student is not returned").doesNotContain(notificationStudent); @@ -134,13 +134,13 @@ void testGetNotifications_groupNotificationTypeEvaluation_asTutor() throws Excep @WithMockUser(username = TEST_PREFIX + "editor1", roles = "EDITOR") void testGetNotifications_groupNotificationTypeEvaluation_asEditor() throws Exception { GroupNotification notificationStudent = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.STUDENT); - notificationRepository.save(notificationStudent); + notificationTestRepository.save(notificationStudent); GroupNotification notificationTutor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.TA); - notificationRepository.save(notificationTutor); + notificationTestRepository.save(notificationTutor); GroupNotification notificationEditor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.EDITOR); - notificationRepository.save(notificationEditor); + notificationTestRepository.save(notificationEditor); GroupNotification notificationInstructor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.INSTRUCTOR); - notificationRepository.save(notificationInstructor); + notificationTestRepository.save(notificationInstructor); List notifications = request.getList("/api/notifications", HttpStatus.OK, Notification.class); assertThat(notifications).as("Notification with type student is not returned").doesNotContain(notificationStudent); @@ -153,13 +153,13 @@ void testGetNotifications_groupNotificationTypeEvaluation_asEditor() throws Exce @WithMockUser(username = TEST_PREFIX + "instructor1", roles = "INSTRUCTOR") void testGetNotifications_groupNotificationTypeEvaluation_asInstructor() throws Exception { GroupNotification notificationStudent = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.STUDENT); - notificationRepository.save(notificationStudent); + notificationTestRepository.save(notificationStudent); GroupNotification notificationTutor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.TA); - notificationRepository.save(notificationTutor); + notificationTestRepository.save(notificationTutor); GroupNotification notificationEditor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.EDITOR); - notificationRepository.save(notificationEditor); + notificationTestRepository.save(notificationEditor); GroupNotification notificationInstructor = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.INSTRUCTOR); - notificationRepository.save(notificationInstructor); + notificationTestRepository.save(notificationInstructor); List notifications = request.getList("/api/notifications", HttpStatus.OK, Notification.class); assertThat(notifications).as("Notification with type student is not returned").doesNotContain(notificationStudent); @@ -184,11 +184,11 @@ void testGetAllNotificationsForCurrentUserFilteredBySettings() throws Exception GroupNotification allowedNotification = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.STUDENT); allowedNotification.setTitle(NotificationConstants.findCorrespondingNotificationTitle(allowedType)); - notificationRepository.save(allowedNotification); + notificationTestRepository.save(allowedNotification); GroupNotification blockedNotification = NotificationFactory.generateGroupNotification(ZonedDateTime.now(), course1, GroupNotificationType.STUDENT); blockedNotification.setTitle(NotificationConstants.findCorrespondingNotificationTitle(blockedType)); - notificationRepository.save(blockedNotification); + notificationTestRepository.save(blockedNotification); List notifications = request.getList("/api/notifications", HttpStatus.OK, Notification.class); @@ -207,10 +207,10 @@ void testGetAllNotificationsForCurrentUser_hideUntilDeactivated() throws Excepti userTestRepository.save(student1); GroupNotification futureNotification = NotificationFactory.generateGroupNotification(timeNow.plusHours(1), course1, GroupNotificationType.STUDENT); - notificationRepository.save(futureNotification); + notificationTestRepository.save(futureNotification); GroupNotification pastNotification = NotificationFactory.generateGroupNotification(timeNow.minusHours(1), course1, GroupNotificationType.STUDENT); - notificationRepository.save(pastNotification); + notificationTestRepository.save(pastNotification); List notifications = request.getList("/api/notifications", HttpStatus.OK, Notification.class); @@ -228,10 +228,10 @@ void testGetAllNotificationsForCurrentUser_hideUntilActivated() throws Exception userTestRepository.save(student1); GroupNotification futureNotification = NotificationFactory.generateGroupNotification(timeNow.plusHours(1), course1, GroupNotificationType.STUDENT); - notificationRepository.save(futureNotification); + notificationTestRepository.save(futureNotification); GroupNotification pastNotification = NotificationFactory.generateGroupNotification(timeNow.minusHours(1), course1, GroupNotificationType.STUDENT); - notificationRepository.save(pastNotification); + notificationTestRepository.save(pastNotification); List notifications = request.getList("/api/notifications", HttpStatus.OK, Notification.class); diff --git a/src/test/java/de/tum/cit/aet/artemis/communication/notification/NotificationScheduleServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/communication/notification/NotificationScheduleServiceTest.java index c49163742ba4..a7506c413281 100644 --- a/src/test/java/de/tum/cit/aet/artemis/communication/notification/NotificationScheduleServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/communication/notification/NotificationScheduleServiceTest.java @@ -22,11 +22,11 @@ import de.tum.cit.aet.artemis.assessment.domain.Result; import de.tum.cit.aet.artemis.assessment.test_repository.ResultTestRepository; import de.tum.cit.aet.artemis.communication.domain.NotificationSetting; -import de.tum.cit.aet.artemis.communication.repository.NotificationRepository; import de.tum.cit.aet.artemis.communication.repository.NotificationSettingRepository; 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.messaging.InstanceMessageReceiveService; +import de.tum.cit.aet.artemis.core.test_repository.NotificationTestRepository; import de.tum.cit.aet.artemis.core.user.util.UserUtilService; import de.tum.cit.aet.artemis.core.util.CourseUtilService; import de.tum.cit.aet.artemis.exercise.domain.Exercise; @@ -43,7 +43,7 @@ class NotificationScheduleServiceTest extends AbstractSpringIntegrationLocalCILo private InstanceMessageReceiveService instanceMessageReceiveService; @Autowired - private NotificationRepository notificationRepository; + private NotificationTestRepository notificationTestRepository; @Autowired private NotificationSettingRepository notificationSettingRepository; @@ -80,7 +80,7 @@ void init() { exercise.setMaxPoints(5.0); exerciseRepository.saveAndFlush(exercise); - sizeBefore = notificationRepository.count(); + sizeBefore = notificationTestRepository.count(); } @Test @@ -91,7 +91,7 @@ void shouldCreateNotificationAndEmailAtReleaseDate() { exerciseRepository.saveAndFlush(exercise); instanceMessageReceiveService.processScheduleExerciseReleasedNotification(exercise.getId()); - await().until(() -> notificationRepository.count() > sizeBefore); + await().until(() -> notificationTestRepository.count() > sizeBefore); verify(groupNotificationService, timeout(TIMEOUT_MS)).notifyAllGroupsAboutReleasedExercise(exercise); verify(mailService, timeout(TIMEOUT_MS).atLeastOnce()).sendNotification(any(), anySet(), any()); } @@ -114,7 +114,7 @@ void shouldCreateNotificationAndEmailAtAssessmentDueDate() { exerciseRepository.saveAndFlush(exercise); instanceMessageReceiveService.processScheduleAssessedExerciseSubmittedNotification(exercise.getId()); - await().until(() -> notificationRepository.count() > sizeBefore); + await().until(() -> notificationTestRepository.count() > sizeBefore); verify(singleUserNotificationService, timeout(TIMEOUT_MS)).notifyUsersAboutAssessedExerciseSubmission(exercise); verify(javaMailSender, timeout(TIMEOUT_MS)).send(any(MimeMessage.class)); } diff --git a/src/test/java/de/tum/cit/aet/artemis/communication/notifications/service/ConversationNotificationServiceTest.java b/src/test/java/de/tum/cit/aet/artemis/communication/notifications/service/ConversationNotificationServiceTest.java index 11911a26af6e..cf646f594ef0 100644 --- a/src/test/java/de/tum/cit/aet/artemis/communication/notifications/service/ConversationNotificationServiceTest.java +++ b/src/test/java/de/tum/cit/aet/artemis/communication/notifications/service/ConversationNotificationServiceTest.java @@ -20,7 +20,6 @@ import de.tum.cit.aet.artemis.communication.domain.notification.ConversationNotification; import de.tum.cit.aet.artemis.communication.domain.notification.Notification; import de.tum.cit.aet.artemis.communication.repository.ConversationMessageRepository; -import de.tum.cit.aet.artemis.communication.repository.NotificationRepository; import de.tum.cit.aet.artemis.communication.repository.conversation.ConversationNotificationRepository; import de.tum.cit.aet.artemis.communication.service.notifications.ConversationNotificationService; import de.tum.cit.aet.artemis.communication.test_repository.ConversationParticipantTestRepository; @@ -28,6 +27,7 @@ import de.tum.cit.aet.artemis.core.domain.Course; import de.tum.cit.aet.artemis.core.domain.DomainObject; import de.tum.cit.aet.artemis.core.domain.User; +import de.tum.cit.aet.artemis.core.test_repository.NotificationTestRepository; import de.tum.cit.aet.artemis.core.test_repository.UserTestRepository; import de.tum.cit.aet.artemis.core.user.util.UserUtilService; import de.tum.cit.aet.artemis.core.util.CourseUtilService; @@ -62,7 +62,7 @@ class ConversationNotificationServiceTest extends AbstractSpringIntegrationIndep private CourseUtilService courseUtilService; @Autowired - private NotificationRepository notificationRepository; + private NotificationTestRepository notificationTestRepository; private OneToOneChat oneToOneChat; @@ -115,7 +115,7 @@ void createNotificationForNewMessageInConversation() { conversationNotificationService.notifyAboutNewMessage(post, notification, Set.of(user2)); verifyRepositoryCallWithCorrectNotification(NEW_MESSAGE_TITLE); - Notification sentNotification = notificationRepository.findAll().stream().max(Comparator.comparing(DomainObject::getId)).orElseThrow(); + Notification sentNotification = notificationTestRepository.findAll().stream().max(Comparator.comparing(DomainObject::getId)).orElseThrow(); verify(generalInstantNotificationService).sendNotification(sentNotification, Set.of(user2), post); diff --git a/src/test/java/de/tum/cit/aet/artemis/core/test_repository/NotificationTestRepository.java b/src/test/java/de/tum/cit/aet/artemis/core/test_repository/NotificationTestRepository.java index 89354e5fae86..5ac06e0cc6c9 100644 --- a/src/test/java/de/tum/cit/aet/artemis/core/test_repository/NotificationTestRepository.java +++ b/src/test/java/de/tum/cit/aet/artemis/core/test_repository/NotificationTestRepository.java @@ -2,11 +2,15 @@ import java.util.List; +import org.springframework.context.annotation.Primary; import org.springframework.data.jpa.repository.Query; +import org.springframework.stereotype.Repository; import de.tum.cit.aet.artemis.communication.domain.notification.Notification; import de.tum.cit.aet.artemis.communication.repository.NotificationRepository; +@Repository +@Primary public interface NotificationTestRepository extends NotificationRepository { @Query(""" diff --git a/src/test/java/de/tum/cit/aet/artemis/core/user/util/UserTestService.java b/src/test/java/de/tum/cit/aet/artemis/core/user/util/UserTestService.java index 1702f990a1d7..67ce0cc894a8 100644 --- a/src/test/java/de/tum/cit/aet/artemis/core/user/util/UserTestService.java +++ b/src/test/java/de/tum/cit/aet/artemis/core/user/util/UserTestService.java @@ -160,7 +160,9 @@ public void setup(String testPrefix, MockDelegate mockDelegate) throws Exception } public void tearDown() throws IOException { - notificationTestRepository.deleteAllInBatch(notificationTestRepository.findAllByRecipientId(student.getId())); + if (student.getId() != null) { + notificationTestRepository.deleteAllInBatch(notificationTestRepository.findAllByRecipientId(student.getId())); + } userTestRepository.deleteAll(userTestRepository.searchAllByLoginOrName(Pageable.unpaged(), TEST_PREFIX)); }