From 4878379313b91e3eccd572aaddfc9c55be0d66ae Mon Sep 17 00:00:00 2001 From: ErykKul Date: Thu, 24 Mar 2022 16:09:43 +0100 Subject: [PATCH 001/192] New feature for muting email and/or in-app notifications by users for given types of notifications --- .../iq/dataverse/UserNotification.java | 37 +++++++++++++++++- .../UserNotificationServiceBean.java | 14 ++++++- .../harvard/iq/dataverse/UserServiceBean.java | 6 ++- .../providers/builtin/DataverseUserPage.java | 39 +++++++++++++++++++ .../users/AuthenticatedUser.java | 39 ++++++++++++++++++- .../iq/dataverse/userdata/UserUtil.java | 18 +++++++++ src/main/java/propertyFiles/Bundle.properties | 29 ++++++++++++++ .../V5.10.0.1__7492-muting-notifications.sql | 5 +++ src/main/webapp/dataverseuser.xhtml | 27 +++++++++++++ 9 files changed, 209 insertions(+), 5 deletions(-) create mode 100644 src/main/resources/db/migration/V5.10.0.1__7492-muting-notifications.sql diff --git a/src/main/java/edu/harvard/iq/dataverse/UserNotification.java b/src/main/java/edu/harvard/iq/dataverse/UserNotification.java index 58152f6673e..72622b11942 100644 --- a/src/main/java/edu/harvard/iq/dataverse/UserNotification.java +++ b/src/main/java/edu/harvard/iq/dataverse/UserNotification.java @@ -1,11 +1,14 @@ package edu.harvard.iq.dataverse; import edu.harvard.iq.dataverse.authorization.users.AuthenticatedUser; +import edu.harvard.iq.dataverse.util.BundleUtil; import edu.harvard.iq.dataverse.util.DateUtil; import java.io.Serializable; import java.sql.Timestamp; import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.List; import javax.persistence.Column; import javax.persistence.Entity; import javax.persistence.Enumerated; @@ -30,7 +33,39 @@ public enum Type { ASSIGNROLE, REVOKEROLE, CREATEDV, CREATEDS, CREATEACC, SUBMITTEDDS, RETURNEDDS, PUBLISHEDDS, REQUESTFILEACCESS, GRANTFILEACCESS, REJECTFILEACCESS, FILESYSTEMIMPORT, CHECKSUMIMPORT, CHECKSUMFAIL, CONFIRMEMAIL, APIGENERATED, INGESTCOMPLETED, INGESTCOMPLETEDWITHERRORS, - PUBLISHFAILED_PIDREG, WORKFLOW_SUCCESS, WORKFLOW_FAILURE, STATUSUPDATED, DATASETCREATED + PUBLISHFAILED_PIDREG, WORKFLOW_SUCCESS, WORKFLOW_FAILURE, STATUSUPDATED, DATASETCREATED; + + public String getDescription() { + return BundleUtil.getStringFromBundle("notification.typeDescription." + this.name()); + } + + public long flagValue() { + return 1 << this.ordinal(); + } + + public static List fromFlag(Long flag) { + final List types = new ArrayList(); + if (flag == null) { + return types; + } + for (final Type t : values()) { + if ((flag & t.flagValue()) > 0) { + types.add(t); + } + } + return types; + } + + public static Long toFlag(final List types) { + if (types == null || types.isEmpty()) { + return null; + } + long flag = 0; + for (final Type t : types) { + flag |= t.flagValue(); + } + return flag; + } }; private static final long serialVersionUID = 1L; diff --git a/src/main/java/edu/harvard/iq/dataverse/UserNotificationServiceBean.java b/src/main/java/edu/harvard/iq/dataverse/UserNotificationServiceBean.java index 071805d3d26..610b2f49e87 100644 --- a/src/main/java/edu/harvard/iq/dataverse/UserNotificationServiceBean.java +++ b/src/main/java/edu/harvard/iq/dataverse/UserNotificationServiceBean.java @@ -110,12 +110,22 @@ public void sendNotification(AuthenticatedUser dataverseUser, Timestamp sendDate userNotification.setObjectId(objectId); userNotification.setRequestor(requestor); - if (mailService.sendNotificationEmail(userNotification, comment, requestor, isHtmlContent)) { + if (!isEmailMuted(userNotification) && mailService.sendNotificationEmail(userNotification, comment, requestor, isHtmlContent)) { logger.fine("email was sent"); userNotification.setEmailed(true); } else { logger.fine("email was not sent"); } - save(userNotification); + if (!isNotificationMuted(userNotification)) { + save(userNotification); + } + } + + public boolean isEmailMuted(UserNotification userNotification) { + return userNotification.getUser().hasEmailMuted(userNotification.getType()); + } + + public boolean isNotificationMuted(UserNotification userNotification) { + return userNotification.getUser().hasNotificationMuted(userNotification.getType()); } } diff --git a/src/main/java/edu/harvard/iq/dataverse/UserServiceBean.java b/src/main/java/edu/harvard/iq/dataverse/UserServiceBean.java index 9ec0527a318..02477c032a1 100644 --- a/src/main/java/edu/harvard/iq/dataverse/UserServiceBean.java +++ b/src/main/java/edu/harvard/iq/dataverse/UserServiceBean.java @@ -144,6 +144,9 @@ private AuthenticatedUser createAuthenticatedUserForView (Object[] dbRowValues, user.setDeactivated((Boolean)(dbRowValues[13])); user.setDeactivatedTime(UserUtil.getTimestampOrNull(dbRowValues[14])); + user.setMutedEmails((Long)(dbRowValues[15])); + user.setMutedNotifications((Long)(dbRowValues[16])); + user.setRoles(roles); return user; } @@ -415,7 +418,8 @@ private List getUserListCore(String searchTerm, qstr += " u.position,"; qstr += " u.createdtime, u.lastlogintime, u.lastapiusetime, "; qstr += " prov.id, prov.factoryalias, "; - qstr += " u.deactivated, u.deactivatedtime "; + qstr += " u.deactivated, u.deactivatedtime, "; + qstr += " u.mutedEmails, u.mutedNotifications "; qstr += " FROM authenticateduser u,"; qstr += " authenticateduserlookup prov_lookup,"; qstr += " authenticationproviderrow prov"; diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java index 177c59c5873..4c5122c2a7d 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java @@ -48,6 +48,7 @@ import java.util.Set; import java.util.logging.Level; import java.util.logging.Logger; +import java.util.stream.Collectors; import javax.ejb.EJB; import javax.faces.application.FacesMessage; import javax.faces.component.UIComponent; @@ -134,6 +135,11 @@ public enum EditMode { private String username; boolean nonLocalLoginEnabled; private List passwordErrors; + + + private List notificationTypeList; + private List mutedEmailList; + private List mutedNotificationList; public String init() { @@ -161,6 +167,11 @@ public String init() { setCurrentUser((AuthenticatedUser) session.getUser()); userAuthProvider = authenticationService.lookupProvider(currentUser); notificationsList = userNotificationService.findByUser(currentUser.getId()); + notificationTypeList = Arrays.asList(UserNotification.Type.values()).stream() + .filter(x -> x.getDescription() != null && !x.getDescription().isEmpty()) + .collect(Collectors.toList()); + mutedEmailList = UserNotification.Type.fromFlag(currentUser.getMutedEmails()); + mutedNotificationList = UserNotification.Type.fromFlag(currentUser.getMutedNotifications()); switch (selectTab) { case "notifications": @@ -368,6 +379,8 @@ public String save() { logger.info("Redirecting"); return permissionsWrapper.notAuthorized() + "faces-redirect=true"; }else { + currentUser.setMutedEmails(UserNotification.Type.toFlag(mutedEmailList)); + currentUser.setMutedNotifications(UserNotification.Type.toFlag(mutedNotificationList)); String emailBeforeUpdate = currentUser.getEmail(); AuthenticatedUser savedUser = authenticationService.updateAuthenticatedUser(currentUser, userDisplayInfo); String emailAfterUpdate = savedUser.getEmail(); @@ -708,4 +721,30 @@ public String getRequestorEmail(UserNotification notification) { if(notification.getRequestor() == null) return BundleUtil.getStringFromBundle("notification.email.info.unavailable");; return notification.getRequestor().getEmail() != null ? notification.getRequestor().getEmail() : BundleUtil.getStringFromBundle("notification.email.info.unavailable"); } + + public List getNotificationTypeList() { + return notificationTypeList; + } + + public void setNotificationTypeList(List notificationTypeList) { + this.notificationTypeList = notificationTypeList; + } + + public List getMutedEmailList() { + return mutedEmailList; + } + + public void setMutedEmailList(List mutedEmailList) { + this.mutedEmailList = mutedEmailList; + } + + public List getMutedNotificationList() { + return mutedNotificationList; + } + + public void setMutedNotificationList(List mutedNotificationList) { + this.mutedNotificationList = mutedNotificationList; + } + + } diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java b/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java index 9d76ce0e47c..ccad45c87ff 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java @@ -4,7 +4,6 @@ import edu.harvard.iq.dataverse.DatasetLock; import edu.harvard.iq.dataverse.UserNotification; import edu.harvard.iq.dataverse.ValidateEmail; -import edu.harvard.iq.dataverse.authorization.AccessRequest; import edu.harvard.iq.dataverse.authorization.AuthenticatedUserDisplayInfo; import edu.harvard.iq.dataverse.authorization.AuthenticatedUserLookup; import edu.harvard.iq.dataverse.authorization.providers.oauth2.OAuth2TokenData; @@ -121,6 +120,12 @@ public class AuthenticatedUser implements User, Serializable { @Column(nullable=true) private Timestamp deactivatedTime; + @Column(nullable=true) + private Long mutedEmails; + + @Column(nullable=true) + private Long mutedNotifications; + /** * @todo Consider storing a hash of *all* potentially interesting Shibboleth * attribute key/value pairs, not just the Identity Provider (IdP). @@ -389,6 +394,8 @@ public JsonObjectBuilder toJson() { authenicatedUserJson.add("deactivated", this.deactivated); authenicatedUserJson.add("deactivatedTime", UserUtil.getTimestampStringOrNull(this.deactivatedTime)); + authenicatedUserJson.add("mutedEmails", UserUtil.getMutedStringOrNull(this.mutedEmails)); + authenicatedUserJson.add("mutedNotifications", UserUtil.getMutedStringOrNull(this.mutedEmails)); return authenicatedUserJson; } @@ -492,4 +499,34 @@ public Cart getCart() { public void setCart(Cart cart) { this.cart = cart; } + + public Long getMutedEmails() { + return mutedEmails; + } + + public void setMutedEmails(Long mutedEmails) { + this.mutedEmails = mutedEmails; + } + + public Long getMutedNotifications() { + return mutedNotifications; + } + + public void setMutedNotifications(Long mutedNotifications) { + this.mutedNotifications = mutedNotifications; + } + + public boolean hasEmailMuted(UserNotification.Type type) { + if (this.mutedEmails == null || type == null) { + return false; + } + return (type.flagValue() & this.mutedEmails) > 0; + } + + public boolean hasNotificationMuted(UserNotification.Type type) { + if (this.mutedNotifications == null || type == null) { + return false; + } + return (type.flagValue() & this.mutedNotifications) > 0; + } } diff --git a/src/main/java/edu/harvard/iq/dataverse/userdata/UserUtil.java b/src/main/java/edu/harvard/iq/dataverse/userdata/UserUtil.java index 1ec17ac5928..d11a5a56400 100644 --- a/src/main/java/edu/harvard/iq/dataverse/userdata/UserUtil.java +++ b/src/main/java/edu/harvard/iq/dataverse/userdata/UserUtil.java @@ -5,7 +5,11 @@ */ package edu.harvard.iq.dataverse.userdata; +import com.beust.jcommander.Strings; +import edu.harvard.iq.dataverse.UserNotification; import java.sql.Timestamp; +import java.util.List; +import java.util.stream.Collectors; /** * @@ -65,5 +69,19 @@ public static Timestamp getTimestampOrNull(Object dbResult){ } return (Timestamp)dbResult; } + + /** + * Convenience method to format UserNotification.Type from flag value + * @param mutedLongFlags + * @return + */ + public static String getMutedStringOrNull(Long mutedLongFlags){ + if (mutedLongFlags == null){ + return null; + } + final List types = UserNotification.Type.fromFlag(mutedLongFlags) + .stream().map(x -> x.name()).collect(Collectors.toList()); + return Strings.join(",", types); + } } diff --git a/src/main/java/propertyFiles/Bundle.properties b/src/main/java/propertyFiles/Bundle.properties index 4c0938ee33f..4dcf2693eee 100644 --- a/src/main/java/propertyFiles/Bundle.properties +++ b/src/main/java/propertyFiles/Bundle.properties @@ -218,6 +218,35 @@ notification.workflowFailed=An external workflow run on {0} in {1} has failed. C notification.workflowSucceeded=An external workflow run on {0} in {1} has succeeded. Check your email and/or view the Dataset page which may have additional details. notification.statusUpdated=The status of dataset {0} has been updated to {1}. +# set this to false to disable muting options: +notification.showMuteOptions=true +notification.mutedEmails=Muted email notification types +notification.mutedNotifications=Muted notification types +notification.typeDescription.ASSIGNROLE=role is assigned +notification.typeDescription.REVOKEROLE=role is revoked +notification.typeDescription.CREATEDV=dataverse is created +notification.typeDescription.CREATEDS=your dataset is created +notification.typeDescription.CREATEACC=account is created +notification.typeDescription.SUBMITTEDDS=submitted for review +notification.typeDescription.RETURNEDDS=returned from review +notification.typeDescription.PUBLISHEDDS=dataset is published +notification.typeDescription.REQUESTFILEACCESS=access to file is requested +notification.typeDescription.GRANTFILEACCESS=access to file is granted +notification.typeDescription.REJECTFILEACCES=access to file is rejected +notification.typeDescription.FILESYSTEMIMPORT=dataset has been successfully uploaded and verified +notification.typeDescription.CHECKSUMIMPORT=dataset had file checksums added via a batch job +notification.typeDescription.CHECKSUMFAIL=checksum validation failed +# it could be counter-productive mute this one (email could not be verified), it is skipped by leaving it empty: +notification.typeDescription.CONFIRMEMAIL= +notification.typeDescription.APIGENERATED=API token is generated +notification.typeDescription.INGESTCOMPLETED=ingest is completed +notification.typeDescription.INGESTCOMPLETEDWITHERRORS=ingest completed with errors +notification.typeDescription.PUBLISHFAILED_PIDREG=publish has failed +notification.typeDescription.WORKFLOW_SUCCESS=external workflow run has succeeded +notification.typeDescription.WORKFLOW_FAILURE=external workflow run has failed +notification.typeDescription.STATUSUPDATED=status of dataset has been updated +notification.typeDescription.DATASETCREATED=dataset was created by user + notification.ingestCompleted=Dataset {1} has one or more tabular files that completed the tabular ingest process and are available in archival formats. notification.ingestCompletedWithErrors=Dataset {1} has one or more tabular files that are available but are not supported for tabular ingest. notification.generic.objectDeleted=The dataverse, dataset, or file for this notification has been deleted. diff --git a/src/main/resources/db/migration/V5.10.0.1__7492-muting-notifications.sql b/src/main/resources/db/migration/V5.10.0.1__7492-muting-notifications.sql new file mode 100644 index 00000000000..4afbd5abe3c --- /dev/null +++ b/src/main/resources/db/migration/V5.10.0.1__7492-muting-notifications.sql @@ -0,0 +1,5 @@ +ALTER TABLE authenticateduser +ADD COLUMN IF NOT EXISTS mutedemails BIGINT; + +ALTER TABLE authenticateduser +ADD COLUMN IF NOT EXISTS mutednotifications BIGINT; diff --git a/src/main/webapp/dataverseuser.xhtml b/src/main/webapp/dataverseuser.xhtml index cb922a0164d..0699e17d89b 100644 --- a/src/main/webapp/dataverseuser.xhtml +++ b/src/main/webapp/dataverseuser.xhtml @@ -377,6 +377,33 @@ +
+
+
+ +
+ + + +
+
+
+ +
+ + + +
+
+ +
From 99b47e03d26d69961cda0401d8412912a9678b18 Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Tue, 29 Mar 2022 11:10:51 +0200 Subject: [PATCH 002/192] added tests to AuthenticatedUserTest.java --- .../users/AuthenticatedUserTest.java | 57 +++++++++++++++++++ 1 file changed, 57 insertions(+) diff --git a/src/test/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUserTest.java b/src/test/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUserTest.java index a756d7cd69e..23e4bb6b91b 100644 --- a/src/test/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUserTest.java +++ b/src/test/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUserTest.java @@ -6,16 +6,20 @@ package edu.harvard.iq.dataverse.authorization.users; import edu.harvard.iq.dataverse.DatasetLock; +import edu.harvard.iq.dataverse.UserNotification; import edu.harvard.iq.dataverse.authorization.AuthenticatedUserDisplayInfo; import edu.harvard.iq.dataverse.authorization.AuthenticatedUserLookup; import edu.harvard.iq.dataverse.mocks.MocksFactory; import java.sql.Timestamp; +import java.util.Arrays; import java.util.Date; import java.util.List; import org.junit.Test; import static org.junit.Assert.*; import org.junit.Before; +import javax.json.JsonObject; + /** * Tested class: AuthenticatedUser.java * @@ -30,6 +34,7 @@ public AuthenticatedUserTest() { public static Timestamp expResult; public static Timestamp loginTime = Timestamp.valueOf("2000-01-01 00:00:00.0"); public static final String IDENTIFIER_PREFIX = "@"; + public static final List mutedTypes = Arrays.asList(UserNotification.Type.ASSIGNROLE, UserNotification.Type.REVOKEROLE); @Before public void setUp() { @@ -320,6 +325,58 @@ public void testHashCode() { int result = instance.hashCode(); assertEquals(expResult, result); } + + @Test + public void testMutingEmails() { + long mutedTypesFlags = UserNotification.Type.toFlag(mutedTypes); + System.out.println("setMutedEmails"); + testUser.setMutedEmails(mutedTypesFlags); + assertEquals(mutedTypesFlags, testUser.getMutedEmails().longValue()); + assertEquals(mutedTypes, UserNotification.Type.fromFlag(testUser.getMutedEmails())); + } + + @Test + public void testMutingNotifications() { + long mutedTypesFlags = UserNotification.Type.toFlag(mutedTypes); + System.out.println("setMutedNotifications"); + testUser.setMutedNotifications(mutedTypesFlags); + assertEquals(mutedTypesFlags, testUser.getMutedNotifications().longValue()); + assertEquals(mutedTypes, UserNotification.Type.fromFlag(testUser.getMutedNotifications())); + } + + @Test + public void testMutingInJson() { + long mutedTypesFlags = UserNotification.Type.toFlag(mutedTypes); + testUser.setMutedEmails(mutedTypesFlags); + testUser.setMutedNotifications(mutedTypesFlags); + System.out.println("toJson"); + JsonObject jObject = testUser.toJson().build(); + assertEquals("ASSIGNROLE,REVOKEROLE", jObject.getString("mutedEmails")); + assertEquals("ASSIGNROLE,REVOKEROLE", jObject.getString("mutedNotifications")); + } + + @Test + public void testHasEmailMuted() { + long mutedTypesFlags = UserNotification.Type.toFlag(mutedTypes); + testUser.setMutedEmails(mutedTypesFlags); + System.out.println("hasEmailMuted"); + assertEquals(true, testUser.hasEmailMuted(UserNotification.Type.ASSIGNROLE)); + assertEquals(true, testUser.hasEmailMuted(UserNotification.Type.REVOKEROLE)); + assertEquals(false, testUser.hasEmailMuted(UserNotification.Type.CREATEDV)); + assertEquals(false, testUser.hasEmailMuted(null)); + } + + @Test + public void testHasNotificationsMutedMuted() { + long mutedTypesFlags = UserNotification.Type.toFlag(mutedTypes); + testUser.setMutedNotifications(mutedTypesFlags); + System.out.println("hasNotificationMuted"); + assertEquals(true, testUser.hasNotificationMuted(UserNotification.Type.ASSIGNROLE)); + assertEquals(true, testUser.hasNotificationMuted(UserNotification.Type.REVOKEROLE)); + assertEquals(false, testUser.hasNotificationMuted(UserNotification.Type.CREATEDV)); + assertEquals(false, testUser.hasNotificationMuted(null)); + } + /** * All commented tests below have only been generated / are not complete for * AuthenticatedUser.java The tests above should all run fine, due to time From 234143569574412cb860c0064c7dd30dec567442 Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Tue, 29 Mar 2022 12:05:39 +0200 Subject: [PATCH 003/192] muting properties are now better documented and moved to more logical position in bundle.properties --- src/main/java/propertyFiles/Bundle.properties | 45 ++++++++++--------- 1 file changed, 25 insertions(+), 20 deletions(-) diff --git a/src/main/java/propertyFiles/Bundle.properties b/src/main/java/propertyFiles/Bundle.properties index 4dcf2693eee..cfb267108a4 100644 --- a/src/main/java/propertyFiles/Bundle.properties +++ b/src/main/java/propertyFiles/Bundle.properties @@ -218,10 +218,33 @@ notification.workflowFailed=An external workflow run on {0} in {1} has failed. C notification.workflowSucceeded=An external workflow run on {0} in {1} has succeeded. Check your email and/or view the Dataset page which may have additional details. notification.statusUpdated=The status of dataset {0} has been updated to {1}. -# set this to false to disable muting options: +notification.ingestCompleted=Dataset {1} has one or more tabular files that completed the tabular ingest process and are available in archival formats. +notification.ingestCompletedWithErrors=Dataset {1} has one or more tabular files that are available but are not supported for tabular ingest. +notification.generic.objectDeleted=The dataverse, dataset, or file for this notification has been deleted. +notification.access.granted.dataverse=You have been granted the {0} role for {1}. +notification.access.granted.dataset=You have been granted the {0} role for {1}. +notification.access.granted.datafile=You have been granted the {0} role for file in {1}. +notification.access.granted.fileDownloader.additionalDataverse={0} You now have access to all published restricted and unrestricted files in this dataverse. +notification.access.granted.fileDownloader.additionalDataset={0} You now have access to all published restricted and unrestricted files in this dataset. +notification.access.revoked.dataverse=You have been removed from a role in {0}. +notification.access.revoked.dataset=You have been removed from a role in {0}. +notification.access.revoked.datafile=You have been removed from a role in {0}. +notification.checksumfail=One or more files in your upload failed checksum validation for dataset {1}. Please re-run the upload script. If the problem persists, please contact support. +notification.ingest.completed=Your Dataset {2} has one or more tabular files that completed the tabular ingest process. These files will be available for download in their original formats and other formats for enhanced archival purposes after you publish the dataset. The archival .tab files are displayed in the file table. Please see the guides for more information about ingest and support for tabular files. +notification.ingest.completedwitherrors=Your Dataset {2} has one or more tabular files that have been uploaded successfully but are not supported for tabular ingest. After you publish the dataset, these files will not have additional archival features. Please see the guides for more information about ingest and support for tabular files.

Files with incomplete ingest:{5} +notification.mail.import.filesystem=Dataset {2} ({0}/dataset.xhtml?persistentId={1}) has been successfully uploaded and verified. +notification.import.filesystem=Dataset {1} has been successfully uploaded and verified. +notification.import.checksum={1}, dataset had file checksums added via a batch job. +removeNotification=Remove Notification + +# Set notification.showMuteOptions to false to disable muting options: notification.showMuteOptions=true + +# These are the labels of the comboboxes where the muted notifications can be selected by the users notification.mutedEmails=Muted email notification types notification.mutedNotifications=Muted notification types + +# Notification types descriptions as presentend to the user. Leave the description empty or leave it out entirely in order to make it not selectable for muting by the user. notification.typeDescription.ASSIGNROLE=role is assigned notification.typeDescription.REVOKEROLE=role is revoked notification.typeDescription.CREATEDV=dataverse is created @@ -236,7 +259,7 @@ notification.typeDescription.REJECTFILEACCES=access to file is rejected notification.typeDescription.FILESYSTEMIMPORT=dataset has been successfully uploaded and verified notification.typeDescription.CHECKSUMIMPORT=dataset had file checksums added via a batch job notification.typeDescription.CHECKSUMFAIL=checksum validation failed -# it could be counter-productive mute this one (email could not be verified), it is skipped by leaving it empty: +# It could be counter-productive to mute this one (email could not be verified), it is skipped (not selectable by the user) by leaving the description empty: notification.typeDescription.CONFIRMEMAIL= notification.typeDescription.APIGENERATED=API token is generated notification.typeDescription.INGESTCOMPLETED=ingest is completed @@ -247,24 +270,6 @@ notification.typeDescription.WORKFLOW_FAILURE=external workflow run has failed notification.typeDescription.STATUSUPDATED=status of dataset has been updated notification.typeDescription.DATASETCREATED=dataset was created by user -notification.ingestCompleted=Dataset {1} has one or more tabular files that completed the tabular ingest process and are available in archival formats. -notification.ingestCompletedWithErrors=Dataset {1} has one or more tabular files that are available but are not supported for tabular ingest. -notification.generic.objectDeleted=The dataverse, dataset, or file for this notification has been deleted. -notification.access.granted.dataverse=You have been granted the {0} role for {1}. -notification.access.granted.dataset=You have been granted the {0} role for {1}. -notification.access.granted.datafile=You have been granted the {0} role for file in {1}. -notification.access.granted.fileDownloader.additionalDataverse={0} You now have access to all published restricted and unrestricted files in this dataverse. -notification.access.granted.fileDownloader.additionalDataset={0} You now have access to all published restricted and unrestricted files in this dataset. -notification.access.revoked.dataverse=You have been removed from a role in {0}. -notification.access.revoked.dataset=You have been removed from a role in {0}. -notification.access.revoked.datafile=You have been removed from a role in {0}. -notification.checksumfail=One or more files in your upload failed checksum validation for dataset {1}. Please re-run the upload script. If the problem persists, please contact support. -notification.ingest.completed=Your Dataset {2} has one or more tabular files that completed the tabular ingest process. These files will be available for download in their original formats and other formats for enhanced archival purposes after you publish the dataset. The archival .tab files are displayed in the file table. Please see the guides for more information about ingest and support for tabular files. -notification.ingest.completedwitherrors=Your Dataset {2} has one or more tabular files that have been uploaded successfully but are not supported for tabular ingest. After you publish the dataset, these files will not have additional archival features. Please see the guides for more information about ingest and support for tabular files.

Files with incomplete ingest:{5} -notification.mail.import.filesystem=Dataset {2} ({0}/dataset.xhtml?persistentId={1}) has been successfully uploaded and verified. -notification.import.filesystem=Dataset {1} has been successfully uploaded and verified. -notification.import.checksum={1}, dataset had file checksums added via a batch job. -removeNotification=Remove Notification groupAndRoles.manageTips=Here is where you can access and manage all the groups you belong to, and the roles you have been assigned. user.message.signup.label=Create Account user.message.signup.tip=Why have a Dataverse account? To create your own dataverse and customize it, add datasets, or request access to restricted files. From 9b79afb90c4186e7b14b1ade6d5cbc93e2b8f84f Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Tue, 29 Mar 2022 15:36:52 +0200 Subject: [PATCH 004/192] added support for muting notification types for the installation where users cannot unmuted these muted types --- .../harvard/iq/dataverse/UserNotification.java | 15 ++++++++++++++- .../iq/dataverse/UserNotificationServiceBean.java | 7 +++++-- .../providers/builtin/DataverseUserPage.java | 3 ++- src/main/java/propertyFiles/Bundle.properties | 3 +++ .../users/AuthenticatedUserTest.java | 11 +++++++++++ 5 files changed, 35 insertions(+), 4 deletions(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/UserNotification.java b/src/main/java/edu/harvard/iq/dataverse/UserNotification.java index 72622b11942..6ab28d6db9a 100644 --- a/src/main/java/edu/harvard/iq/dataverse/UserNotification.java +++ b/src/main/java/edu/harvard/iq/dataverse/UserNotification.java @@ -9,6 +9,10 @@ import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.List; +import java.util.Set; +import java.util.StringTokenizer; +import java.util.Collections; +import java.util.stream.Collectors; import javax.persistence.Column; import javax.persistence.Entity; import javax.persistence.Enumerated; @@ -44,7 +48,7 @@ public long flagValue() { } public static List fromFlag(Long flag) { - final List types = new ArrayList(); + final List types = new ArrayList(); if (flag == null) { return types; } @@ -66,6 +70,15 @@ public static Long toFlag(final List types) { } return flag; } + + public static Set tokenizeToSet(String tokens) { + if (tokens == null || tokens.isEmpty()) { + return Collections.emptySet(); + } + return Collections.list(new StringTokenizer(tokens, ",")).stream() + .map(token -> Type.valueOf(((String) token).trim())) + .collect(Collectors.toSet()); + } }; private static final long serialVersionUID = 1L; diff --git a/src/main/java/edu/harvard/iq/dataverse/UserNotificationServiceBean.java b/src/main/java/edu/harvard/iq/dataverse/UserNotificationServiceBean.java index 610b2f49e87..cf4aeec51c8 100644 --- a/src/main/java/edu/harvard/iq/dataverse/UserNotificationServiceBean.java +++ b/src/main/java/edu/harvard/iq/dataverse/UserNotificationServiceBean.java @@ -8,8 +8,10 @@ import edu.harvard.iq.dataverse.UserNotification.Type; import edu.harvard.iq.dataverse.authorization.users.AuthenticatedUser; +import edu.harvard.iq.dataverse.util.BundleUtil; import java.sql.Timestamp; import java.util.List; +import java.util.Set; import java.util.logging.Logger; import javax.ejb.EJB; import javax.ejb.Stateless; @@ -103,6 +105,7 @@ public void sendNotification(AuthenticatedUser dataverseUser, Timestamp sendDate } public void sendNotification(AuthenticatedUser dataverseUser, Timestamp sendDate, Type type, Long objectId, String comment, AuthenticatedUser requestor, boolean isHtmlContent) { + final Set alwaysMuted = Type.tokenizeToSet(BundleUtil.getStringFromBundle("notification.alwaysMuted")); UserNotification userNotification = new UserNotification(); userNotification.setUser(dataverseUser); userNotification.setSendDate(sendDate); @@ -110,13 +113,13 @@ public void sendNotification(AuthenticatedUser dataverseUser, Timestamp sendDate userNotification.setObjectId(objectId); userNotification.setRequestor(requestor); - if (!isEmailMuted(userNotification) && mailService.sendNotificationEmail(userNotification, comment, requestor, isHtmlContent)) { + if (!alwaysMuted.contains(userNotification.getType()) && !isEmailMuted(userNotification) && mailService.sendNotificationEmail(userNotification, comment, requestor, isHtmlContent)) { logger.fine("email was sent"); userNotification.setEmailed(true); } else { logger.fine("email was not sent"); } - if (!isNotificationMuted(userNotification)) { + if (!alwaysMuted.contains(userNotification.getType()) && !isNotificationMuted(userNotification)) { save(userNotification); } } diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java index 4c5122c2a7d..172607bb988 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java @@ -167,8 +167,9 @@ public String init() { setCurrentUser((AuthenticatedUser) session.getUser()); userAuthProvider = authenticationService.lookupProvider(currentUser); notificationsList = userNotificationService.findByUser(currentUser.getId()); + final Set alwaysMuted = UserNotification.Type.tokenizeToSet(BundleUtil.getStringFromBundle("notification.alwaysMuted")); notificationTypeList = Arrays.asList(UserNotification.Type.values()).stream() - .filter(x -> x.getDescription() != null && !x.getDescription().isEmpty()) + .filter(x -> !alwaysMuted.contains(x) && x.getDescription() != null && !x.getDescription().isEmpty()) .collect(Collectors.toList()); mutedEmailList = UserNotification.Type.fromFlag(currentUser.getMutedEmails()); mutedNotificationList = UserNotification.Type.fromFlag(currentUser.getMutedNotifications()); diff --git a/src/main/java/propertyFiles/Bundle.properties b/src/main/java/propertyFiles/Bundle.properties index cfb267108a4..305f7162ed5 100644 --- a/src/main/java/propertyFiles/Bundle.properties +++ b/src/main/java/propertyFiles/Bundle.properties @@ -240,6 +240,9 @@ removeNotification=Remove Notification # Set notification.showMuteOptions to false to disable muting options: notification.showMuteOptions=true +# List (comma separated) of always muted notifications that cannot be turned on by the users: +# notification.alwaysMuted=ASSIGNROLE, REVOKEROLE + # These are the labels of the comboboxes where the muted notifications can be selected by the users notification.mutedEmails=Muted email notification types notification.mutedNotifications=Muted notification types diff --git a/src/test/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUserTest.java b/src/test/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUserTest.java index 23e4bb6b91b..4afcc9ecbb5 100644 --- a/src/test/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUserTest.java +++ b/src/test/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUserTest.java @@ -14,6 +14,8 @@ import java.util.Arrays; import java.util.Date; import java.util.List; +import java.util.Set; + import org.junit.Test; import static org.junit.Assert.*; import org.junit.Before; @@ -377,6 +379,15 @@ public void testHasNotificationsMutedMuted() { assertEquals(false, testUser.hasNotificationMuted(null)); } + @Test + public void testTypeTokenizer() { + final Set typeSet = UserNotification.Type.tokenizeToSet(" ASSIGNROLE , CREATEDV,REVOKEROLE "); + assertTrue("typeSet contains 3 elements", typeSet.size() == 3); + assertTrue("typeSet contains ASSIGNROLE", typeSet.contains(UserNotification.Type.ASSIGNROLE)); + assertTrue("typeSet contains CREATEDV", typeSet.contains(UserNotification.Type.CREATEDV)); + assertTrue("typeSet contains REVOKEROLE", typeSet.contains(UserNotification.Type.REVOKEROLE)); + } + /** * All commented tests below have only been generated / are not complete for * AuthenticatedUser.java The tests above should all run fine, due to time From 55ddd7039e6a1b28bc53bf75855c42d8dfac2e1a Mon Sep 17 00:00:00 2001 From: Julian Gautier Date: Tue, 29 Mar 2022 10:38:22 -0400 Subject: [PATCH 005/192] Update 5.10-release-notes.md --- doc/release-notes/5.10-release-notes.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/doc/release-notes/5.10-release-notes.md b/doc/release-notes/5.10-release-notes.md index 0da42a7b527..c13ae8a6b78 100644 --- a/doc/release-notes/5.10-release-notes.md +++ b/doc/release-notes/5.10-release-notes.md @@ -140,7 +140,7 @@ or To find datasets with a without a CC0 license and with empty terms: ``` -select CONCAT('doi:', dvo.authority, '/', dvo.identifier), v.alias as dataverse_alias, case when versionstate='RELEASED' then concat(dv.versionnumber, '.', dv.minorversionnumber) else versionstate END as version, dv.id as datasetversion_id, t.id as termsofuseandaccess_id, t.termsofuse, t.confidentialitydeclaration, t.specialpermissions, t.restrictions, t.citationrequirements, t.depositorrequirements, t.conditions, t.disclaimer from dvobject dvo, termsofuseandaccess t, datasetversion dv, dataverse v where dv.dataset_id=dvo.id and dv.termsofuseandaccess_id=t.id and dvo.owner_id=v.id and t.license='NONE' and t.termsofuse is null; +select CONCAT('doi:', dvo.authority, '/', dvo.identifier), v.alias as dataverse_alias, case when versionstate='RELEASED' then concat(dv.versionnumber, '.', dv.minorversionnumber) else versionstate END as version, dv.id as datasetversion_id, t.id as termsofuseandaccess_id, t.termsofuse, t.confidentialitydeclaration, t.specialpermissions, t.restrictions, t.citationrequirements, t.depositorrequirements, t.conditions, t.disclaimer from dvobject dvo, termsofuseandaccess t, datasetversion dv, dataverse v where dv.dataset_id=dvo.id and dv.termsofuseandaccess_id=t.id and dvo.owner_id=v.id and (t.license='NONE' or t.license is null) and t.termsofuse is null; ``` As before, there are a couple options. From abe1af23a84ca0e4dfb9e004c133827ebb2a5276 Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Wed, 30 Mar 2022 18:21:10 +0200 Subject: [PATCH 006/192] implementation of in review requested changes --- .../harvard/iq/dataverse/SettingsWrapper.java | 51 +++++++++++++++---- .../iq/dataverse/UserNotification.java | 8 ++- .../providers/builtin/DataverseUserPage.java | 35 ++++++------- .../users/AuthenticatedUser.java | 46 ++++++++++++----- .../settings/SettingsServiceBean.java | 14 ++++- .../iq/dataverse/userdata/UserUtil.java | 12 ++--- src/main/java/propertyFiles/Bundle.properties | 6 --- src/main/webapp/dataverseuser.xhtml | 2 +- .../users/AuthenticatedUserTest.java | 42 ++++++++------- 9 files changed, 141 insertions(+), 75 deletions(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/SettingsWrapper.java b/src/main/java/edu/harvard/iq/dataverse/SettingsWrapper.java index 7c4569e1dc5..6a2b70b6334 100644 --- a/src/main/java/edu/harvard/iq/dataverse/SettingsWrapper.java +++ b/src/main/java/edu/harvard/iq/dataverse/SettingsWrapper.java @@ -13,18 +13,18 @@ import edu.harvard.iq.dataverse.util.MailUtil; import edu.harvard.iq.dataverse.util.StringUtil; import edu.harvard.iq.dataverse.util.SystemConfig; +import edu.harvard.iq.dataverse.UserNotification.Type; import java.time.LocalDate; import java.time.format.DateTimeFormatter; -import java.io.StringReader; import java.util.ArrayList; import java.util.Arrays; import java.util.HashMap; import java.util.LinkedHashMap; import java.util.List; -import java.util.Locale; import java.util.Map; import java.util.logging.Logger; +import java.util.Set; import javax.ejb.EJB; import javax.faces.application.FacesMessage; @@ -34,15 +34,8 @@ import javax.faces.validator.ValidatorException; import javax.faces.view.ViewScoped; import javax.inject.Named; -import javax.json.Json; -import javax.json.JsonArray; import javax.json.JsonObject; -import javax.json.JsonReader; -import javax.json.JsonValue; import javax.mail.internet.InternetAddress; -import org.json.JSONArray; -import org.json.JSONException; -import org.json.JSONObject; /** * @@ -111,6 +104,10 @@ public class SettingsWrapper implements java.io.Serializable { private Boolean customLicenseAllowed = null; + private Set alwaysMuted = null; + + private Set neverMuted = null; + public String get(String settingKey) { if (settingsMap == null) { initSettingsMap(); @@ -185,6 +182,42 @@ private void initSettingsMap() { } } + private void initAlwyasMuted() { + alwaysMuted = UserNotification.Type.tokenizeToSet(getValueForKey(Key.AlwaysMuted)); + } + + private void initNeverMuted() { + neverMuted = UserNotification.Type.tokenizeToSet(getValueForKey(Key.NeverMuted)); + } + + private Set getAlwaysMutedSet() { + if (alwaysMuted == null) { + initAlwyasMuted(); + } + return alwaysMuted; + } + + private Set getNeverMutedSet() { + if (neverMuted == null) { + initNeverMuted(); + } + return neverMuted; + } + + public boolean isAlwaysMuted(Type type) { + return getAlwaysMutedSet().contains(type); + } + + public boolean isNeverMuted(Type type) { + return getNeverMutedSet().contains(type); + } + + public String getShowMuteOptions() { + final boolean safeDefaultIfKeyNotFound = false; + final boolean doShow = isTrueForKey(Key.ShowMuteOptions, safeDefaultIfKeyNotFound); + return doShow ? "TRUE" : "FALSE"; + } + public String getGuidesBaseUrl() { if (guidesBaseUrl == null) { diff --git a/src/main/java/edu/harvard/iq/dataverse/UserNotification.java b/src/main/java/edu/harvard/iq/dataverse/UserNotification.java index 6ab28d6db9a..2a6e60be567 100644 --- a/src/main/java/edu/harvard/iq/dataverse/UserNotification.java +++ b/src/main/java/edu/harvard/iq/dataverse/UserNotification.java @@ -10,6 +10,7 @@ import java.util.ArrayList; import java.util.List; import java.util.Set; +import java.util.Collection; import java.util.StringTokenizer; import java.util.Collections; import java.util.stream.Collectors; @@ -42,6 +43,11 @@ public enum Type { public String getDescription() { return BundleUtil.getStringFromBundle("notification.typeDescription." + this.name()); } + + public boolean hasDescription() { + final String description = getDescription(); + return description != null && !description.isEmpty(); + } public long flagValue() { return 1 << this.ordinal(); @@ -60,7 +66,7 @@ public static List fromFlag(Long flag) { return types; } - public static Long toFlag(final List types) { + public static Long toFlag(final Collection types) { if (types == null || types.isEmpty()) { return null; } diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java index 172607bb988..880ef1e4ded 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java @@ -17,6 +17,7 @@ import edu.harvard.iq.dataverse.SettingsWrapper; import edu.harvard.iq.dataverse.UserNameValidator; import edu.harvard.iq.dataverse.UserNotification; +import edu.harvard.iq.dataverse.UserNotification.Type; import edu.harvard.iq.dataverse.UserNotificationServiceBean; import edu.harvard.iq.dataverse.UserServiceBean; import edu.harvard.iq.dataverse.authorization.AuthUtil; @@ -46,6 +47,7 @@ import java.util.List; import java.util.Objects; import java.util.Set; +import java.util.EnumSet; import java.util.logging.Level; import java.util.logging.Logger; import java.util.stream.Collectors; @@ -137,9 +139,9 @@ public enum EditMode { private List passwordErrors; - private List notificationTypeList; - private List mutedEmailList; - private List mutedNotificationList; + private List notificationTypeList; + private List mutedEmailList; + private List mutedNotificationList; public String init() { @@ -167,12 +169,11 @@ public String init() { setCurrentUser((AuthenticatedUser) session.getUser()); userAuthProvider = authenticationService.lookupProvider(currentUser); notificationsList = userNotificationService.findByUser(currentUser.getId()); - final Set alwaysMuted = UserNotification.Type.tokenizeToSet(BundleUtil.getStringFromBundle("notification.alwaysMuted")); - notificationTypeList = Arrays.asList(UserNotification.Type.values()).stream() - .filter(x -> !alwaysMuted.contains(x) && x.getDescription() != null && !x.getDescription().isEmpty()) + notificationTypeList = Arrays.asList(Type.values()).stream() + .filter(x -> !settingsWrapper.isAlwaysMuted(x) && !settingsWrapper.isAlwaysMuted(x) && x.hasDescription()) .collect(Collectors.toList()); - mutedEmailList = UserNotification.Type.fromFlag(currentUser.getMutedEmails()); - mutedNotificationList = UserNotification.Type.fromFlag(currentUser.getMutedNotifications()); + mutedEmailList = new ArrayList<>(currentUser.getMutedEmailsSet()); + mutedNotificationList = new ArrayList<>(currentUser.getMutedNotificationsSet()); switch (selectTab) { case "notifications": @@ -346,7 +347,7 @@ public String save() { */ userNotificationService.sendNotification(au, new Timestamp(new Date().getTime()), - UserNotification.Type.CREATEACC, null); + Type.CREATEACC, null); // go back to where user came from @@ -380,8 +381,8 @@ public String save() { logger.info("Redirecting"); return permissionsWrapper.notAuthorized() + "faces-redirect=true"; }else { - currentUser.setMutedEmails(UserNotification.Type.toFlag(mutedEmailList)); - currentUser.setMutedNotifications(UserNotification.Type.toFlag(mutedNotificationList)); + currentUser.setMutedEmailsSet(EnumSet.copyOf(mutedEmailList)); + currentUser.setMutedNotificationsSet(EnumSet.copyOf(mutedNotificationList)); String emailBeforeUpdate = currentUser.getEmail(); AuthenticatedUser savedUser = authenticationService.updateAuthenticatedUser(currentUser, userDisplayInfo); String emailAfterUpdate = savedUser.getEmail(); @@ -723,27 +724,27 @@ public String getRequestorEmail(UserNotification notification) { return notification.getRequestor().getEmail() != null ? notification.getRequestor().getEmail() : BundleUtil.getStringFromBundle("notification.email.info.unavailable"); } - public List getNotificationTypeList() { + public List getNotificationTypeList() { return notificationTypeList; } - public void setNotificationTypeList(List notificationTypeList) { + public void setNotificationTypeList(List notificationTypeList) { this.notificationTypeList = notificationTypeList; } - public List getMutedEmailList() { + public List getMutedEmailList() { return mutedEmailList; } - public void setMutedEmailList(List mutedEmailList) { + public void setMutedEmailList(List mutedEmailList) { this.mutedEmailList = mutedEmailList; } - public List getMutedNotificationList() { + public List getMutedNotificationList() { return mutedNotificationList; } - public void setMutedNotificationList(List mutedNotificationList) { + public void setMutedNotificationList(List mutedNotificationList) { this.mutedNotificationList = mutedNotificationList; } diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java b/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java index ccad45c87ff..0f31134062f 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java @@ -3,6 +3,7 @@ import edu.harvard.iq.dataverse.Cart; import edu.harvard.iq.dataverse.DatasetLock; import edu.harvard.iq.dataverse.UserNotification; +import edu.harvard.iq.dataverse.UserNotification.Type; import edu.harvard.iq.dataverse.ValidateEmail; import edu.harvard.iq.dataverse.authorization.AuthenticatedUserDisplayInfo; import edu.harvard.iq.dataverse.authorization.AuthenticatedUserLookup; @@ -14,8 +15,11 @@ import edu.harvard.iq.dataverse.util.json.NullSafeJsonBuilder; import java.io.Serializable; import java.sql.Timestamp; +import java.util.EnumSet; import java.util.List; import java.util.Objects; +import java.util.Set; + import javax.json.Json; import javax.json.JsonObjectBuilder; import javax.persistence.CascadeType; @@ -125,6 +129,12 @@ public class AuthenticatedUser implements User, Serializable { @Column(nullable=true) private Long mutedNotifications; + + @Transient + private Set mutedEmailsSet; + + @Transient + private Set mutedNotificationsSet; /** * @todo Consider storing a hash of *all* potentially interesting Shibboleth @@ -394,8 +404,8 @@ public JsonObjectBuilder toJson() { authenicatedUserJson.add("deactivated", this.deactivated); authenicatedUserJson.add("deactivatedTime", UserUtil.getTimestampStringOrNull(this.deactivatedTime)); - authenicatedUserJson.add("mutedEmails", UserUtil.getMutedStringOrNull(this.mutedEmails)); - authenicatedUserJson.add("mutedNotifications", UserUtil.getMutedStringOrNull(this.mutedEmails)); + authenicatedUserJson.add("mutedEmails", UserUtil.getMutedStringOrNull(this.mutedEmailsSet)); + authenicatedUserJson.add("mutedNotifications", UserUtil.getMutedStringOrNull(this.mutedEmailsSet)); return authenicatedUserJson; } @@ -500,33 +510,45 @@ public void setCart(Cart cart) { this.cart = cart; } - public Long getMutedEmails() { - return mutedEmails; + public Set getMutedEmailsSet() { + return mutedEmailsSet; } public void setMutedEmails(Long mutedEmails) { this.mutedEmails = mutedEmails; + this.mutedEmailsSet = EnumSet.copyOf(Type.fromFlag(mutedEmails)); } - public Long getMutedNotifications() { - return mutedNotifications; + public void setMutedEmailsSet(Set mutedEmails) { + this.mutedEmailsSet = mutedEmails; + this.mutedEmails = Type.toFlag(mutedEmails); + } + + public Set getMutedNotificationsSet() { + return mutedNotificationsSet; } public void setMutedNotifications(Long mutedNotifications) { this.mutedNotifications = mutedNotifications; + this.mutedNotificationsSet = EnumSet.copyOf(Type.fromFlag(mutedNotifications)); + } + + public void setMutedNotificationsSet(Set mutedNotifications) { + this.mutedNotificationsSet = mutedNotifications; + this.mutedNotifications = Type.toFlag(mutedNotifications); } - public boolean hasEmailMuted(UserNotification.Type type) { - if (this.mutedEmails == null || type == null) { + public boolean hasEmailMuted(Type type) { + if (this.mutedEmailsSet == null || type == null) { return false; } - return (type.flagValue() & this.mutedEmails) > 0; + return this.mutedEmailsSet.contains(type); } - public boolean hasNotificationMuted(UserNotification.Type type) { - if (this.mutedNotifications == null || type == null) { + public boolean hasNotificationMuted(Type type) { + if (this.mutedNotificationsSet == null || type == null) { return false; } - return (type.flagValue() & this.mutedNotifications) > 0; + return this.mutedNotificationsSet.contains(type); } } diff --git a/src/main/java/edu/harvard/iq/dataverse/settings/SettingsServiceBean.java b/src/main/java/edu/harvard/iq/dataverse/settings/SettingsServiceBean.java index e13ea806dc7..1c1d6bd41a5 100644 --- a/src/main/java/edu/harvard/iq/dataverse/settings/SettingsServiceBean.java +++ b/src/main/java/edu/harvard/iq/dataverse/settings/SettingsServiceBean.java @@ -505,7 +505,19 @@ Whether Harvesting (OAI) service is enabled /* * Include "Custom Terms" as an item in the license drop-down or not. */ - AllowCustomTermsOfUse + AllowCustomTermsOfUse, + /* + * Allow users to mute notifications or not. + */ + ShowMuteOptions, + /* + * List (comma separated, e.g., "ASSIGNROLE, REVOKEROLE") of always muted notifications that cannot be turned on by the users. + */ + AlwaysMuted, + /* + * List (comma separated, e.g., "ASSIGNROLE, REVOKEROLE") of never muted notifications that cannot be turned off by the users. + */ + NeverMuted ; @Override diff --git a/src/main/java/edu/harvard/iq/dataverse/userdata/UserUtil.java b/src/main/java/edu/harvard/iq/dataverse/userdata/UserUtil.java index d11a5a56400..8b655d9bd8f 100644 --- a/src/main/java/edu/harvard/iq/dataverse/userdata/UserUtil.java +++ b/src/main/java/edu/harvard/iq/dataverse/userdata/UserUtil.java @@ -5,10 +5,10 @@ */ package edu.harvard.iq.dataverse.userdata; -import com.beust.jcommander.Strings; -import edu.harvard.iq.dataverse.UserNotification; +import edu.harvard.iq.dataverse.UserNotification.Type; import java.sql.Timestamp; import java.util.List; +import java.util.Set; import java.util.stream.Collectors; /** @@ -75,13 +75,13 @@ public static Timestamp getTimestampOrNull(Object dbResult){ * @param mutedLongFlags * @return */ - public static String getMutedStringOrNull(Long mutedLongFlags){ - if (mutedLongFlags == null){ + public static String getMutedStringOrNull(Set muted){ + if (muted == null){ return null; } - final List types = UserNotification.Type.fromFlag(mutedLongFlags) + final List types = muted .stream().map(x -> x.name()).collect(Collectors.toList()); - return Strings.join(",", types); + return String.join(",", types); } } diff --git a/src/main/java/propertyFiles/Bundle.properties b/src/main/java/propertyFiles/Bundle.properties index 305f7162ed5..d9955f2e791 100644 --- a/src/main/java/propertyFiles/Bundle.properties +++ b/src/main/java/propertyFiles/Bundle.properties @@ -237,12 +237,6 @@ notification.import.filesystem=Dataset {1}, dataset had file checksums added via a batch job. removeNotification=Remove Notification -# Set notification.showMuteOptions to false to disable muting options: -notification.showMuteOptions=true - -# List (comma separated) of always muted notifications that cannot be turned on by the users: -# notification.alwaysMuted=ASSIGNROLE, REVOKEROLE - # These are the labels of the comboboxes where the muted notifications can be selected by the users notification.mutedEmails=Muted email notification types notification.mutedNotifications=Muted notification types diff --git a/src/main/webapp/dataverseuser.xhtml b/src/main/webapp/dataverseuser.xhtml index 0699e17d89b..9ad475c7079 100644 --- a/src/main/webapp/dataverseuser.xhtml +++ b/src/main/webapp/dataverseuser.xhtml @@ -377,7 +377,7 @@
-
+

- - - + + +
@@ -395,10 +394,9 @@ #{bundle['notification.mutedNotifications']}
- - - + + +
Date: Wed, 13 Apr 2022 16:32:13 -0400 Subject: [PATCH 012/192] remove thread config code --- .../edu/harvard/iq/dataverse/util/bagit/BagGenerator.java | 5 ----- 1 file changed, 5 deletions(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/util/bagit/BagGenerator.java b/src/main/java/edu/harvard/iq/dataverse/util/bagit/BagGenerator.java index 08a46f523c2..3b0b9f27ea4 100644 --- a/src/main/java/edu/harvard/iq/dataverse/util/bagit/BagGenerator.java +++ b/src/main/java/edu/harvard/iq/dataverse/util/bagit/BagGenerator.java @@ -116,7 +116,6 @@ public class BagGenerator { private boolean usetemp = false; private int numConnections = 8; - public static final String BAG_GENERATOR_THREADS = ":BagGeneratorThreads"; private OREMap oremap; @@ -1107,8 +1106,4 @@ public void setAuthenticationKey(String tokenString) { apiKey = tokenString; } - public void setNumConnections(int numConnections) { - this.numConnections = numConnections; - } - } \ No newline at end of file From 2d981e3d2326c29316b445c5503657346259c03c Mon Sep 17 00:00:00 2001 From: qqmyers Date: Wed, 13 Apr 2022 18:35:30 -0400 Subject: [PATCH 013/192] duracloud and google thread mgmt fixes --- .../impl/DuraCloudSubmitToArchiveCommand.java | 141 ++++++++++++------ .../GoogleCloudSubmitToArchiveCommand.java | 71 +++++---- 2 files changed, 138 insertions(+), 74 deletions(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/DuraCloudSubmitToArchiveCommand.java b/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/DuraCloudSubmitToArchiveCommand.java index 468e99f24c1..b3b303d7407 100644 --- a/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/DuraCloudSubmitToArchiveCommand.java +++ b/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/DuraCloudSubmitToArchiveCommand.java @@ -41,19 +41,38 @@ public class DuraCloudSubmitToArchiveCommand extends AbstractSubmitToArchiveComm private static final String DURACLOUD_PORT = ":DuraCloudPort"; private static final String DURACLOUD_HOST = ":DuraCloudHost"; private static final String DURACLOUD_CONTEXT = ":DuraCloudContext"; - + private static final int DEFAULT_THREADS = 2; + + boolean success = false; + int bagThreads = DEFAULT_THREADS; public DuraCloudSubmitToArchiveCommand(DataverseRequest aRequest, DatasetVersion version) { super(aRequest, version); } @Override - public WorkflowStepResult performArchiveSubmission(DatasetVersion dv, ApiToken token, Map requestedSettings) { + public WorkflowStepResult performArchiveSubmission(DatasetVersion dv, ApiToken token, + Map requestedSettings) { - String port = requestedSettings.get(DURACLOUD_PORT) != null ? requestedSettings.get(DURACLOUD_PORT) : DEFAULT_PORT; - String dpnContext = requestedSettings.get(DURACLOUD_CONTEXT) != null ? requestedSettings.get(DURACLOUD_CONTEXT) : DEFAULT_CONTEXT; + String port = requestedSettings.get(DURACLOUD_PORT) != null ? requestedSettings.get(DURACLOUD_PORT) + : DEFAULT_PORT; + String dpnContext = requestedSettings.get(DURACLOUD_CONTEXT) != null ? requestedSettings.get(DURACLOUD_CONTEXT) + : DEFAULT_CONTEXT; String host = requestedSettings.get(DURACLOUD_HOST); + + if (requestedSettings.get(BagGenerator.BAG_GENERATOR_THREADS) != null) { + try { + bagThreads=Integer.valueOf(requestedSettings.get(BagGenerator.BAG_GENERATOR_THREADS)); + } catch (NumberFormatException nfe) { + logger.warning("Can't parse the value of setting " + BagGenerator.BAG_GENERATOR_THREADS + " as an integer - using default:" + DEFAULT_THREADS); + } + } + if (host != null) { Dataset dataset = dv.getDataset(); + // ToDo - change after HDC 3A changes to status reporting + // This will make the archivalCopyLocation non-null after a failure which should + // stop retries + dv.setArchivalCopyLocation("Attempted"); if (dataset.getLockFor(Reason.finalizePublication) == null && dataset.getLockFor(Reason.FileValidationFailed) == null) { // Use Duracloud client classes to login @@ -61,9 +80,24 @@ public WorkflowStepResult performArchiveSubmission(DatasetVersion dv, ApiToken t Credential credential = new Credential(System.getProperty("duracloud.username"), System.getProperty("duracloud.password")); storeManager.login(credential); - - String spaceName = dataset.getGlobalId().asString().replace(':', '-').replace('/', '-') - .replace('.', '-').toLowerCase(); + /* + * Aliases can contain upper case characters which are not allowed in space + * names. Similarly, aliases can contain '_' which isn't allowed in a space + * name. The line below replaces any upper case chars with lowercase and + * replaces any '_' with '.-' . The '-' after the dot assures we don't break the + * rule that + * "The last period in a aspace may not immediately be followed by a number". + * (Although we could check, it seems better to just add '.-' all the time.As + * written the replaceAll will also change any chars not valid in a spaceName to + * '.' which would avoid code breaking if the alias constraints change. That + * said, this line may map more than one alias to the same spaceName, e.g. + * "test" and "Test" aliases both map to the "test" space name. This does not + * break anything but does potentially put bags from more than one collection in + * the same space. + */ + String spaceName = dataset.getOwner().getAlias().toLowerCase().replaceAll("[^a-z0-9-]", ".dcsafe"); + String baseFileName = dataset.getGlobalId().asString().replace(':', '-').replace('/', '-') + .replace('.', '-').toLowerCase() + "_v" + dv.getFriendlyVersionNumber(); ContentStore store; try { @@ -75,87 +109,109 @@ public WorkflowStepResult performArchiveSubmission(DatasetVersion dv, ApiToken t */ store = storeManager.getPrimaryContentStore(); // Create space to copy archival files to - store.createSpace(spaceName); + if (!store.spaceExists(spaceName)) { + store.createSpace(spaceName); + } DataCitation dc = new DataCitation(dv); Map metadata = dc.getDataCiteMetadata(); String dataciteXml = DOIDataCiteRegisterService.getMetadataFromDvObject( dv.getDataset().getGlobalId().asString(), metadata, dv.getDataset()); MessageDigest messageDigest = MessageDigest.getInstance("MD5"); - try (PipedInputStream dataciteIn = new PipedInputStream(); DigestInputStream digestInputStream = new DigestInputStream(dataciteIn, messageDigest)) { + try (PipedInputStream dataciteIn = new PipedInputStream(); + DigestInputStream digestInputStream = new DigestInputStream(dataciteIn, messageDigest)) { // Add datacite.xml file - new Thread(new Runnable() { + Thread dcThread = new Thread(new Runnable() { public void run() { try (PipedOutputStream dataciteOut = new PipedOutputStream(dataciteIn)) { dataciteOut.write(dataciteXml.getBytes(Charset.forName("utf-8"))); dataciteOut.close(); + success=true; } catch (Exception e) { logger.severe("Error creating datacite.xml: " + e.getMessage()); // TODO Auto-generated catch block e.printStackTrace(); - throw new RuntimeException("Error creating datacite.xml: " + e.getMessage()); } } - }).start(); - //Have seen Pipe Closed errors for other archivers when used as a workflow without this delay loop - int i=0; - while(digestInputStream.available()<=0 && i<100) { + }); + dcThread.start(); + // Have seen Pipe Closed errors for other archivers when used as a workflow + // without this delay loop + int i = 0; + while (digestInputStream.available() <= 0 && i < 100) { Thread.sleep(10); i++; } - String checksum = store.addContent(spaceName, "datacite.xml", digestInputStream, -1l, null, null, - null); + String checksum = store.addContent(spaceName, baseFileName + "_datacite.xml", digestInputStream, + -1l, null, null, null); logger.fine("Content: datacite.xml added with checksum: " + checksum); + dcThread.join(); String localchecksum = Hex.encodeHexString(digestInputStream.getMessageDigest().digest()); - if (!checksum.equals(localchecksum)) { - logger.severe(checksum + " not equal to " + localchecksum); + if (!success || !checksum.equals(localchecksum)) { + logger.severe("Failure on " + baseFileName); + logger.severe(success ? checksum + " not equal to " + localchecksum : "failed to transfer to DuraCloud"); + try { + store.deleteContent(spaceName, baseFileName + "_datacite.xml"); + } catch (ContentStoreException cse) { + logger.warning(cse.getMessage()); + } return new Failure("Error in transferring DataCite.xml file to DuraCloud", "DuraCloud Submission Failure: incomplete metadata transfer"); } // Store BagIt file - String fileName = spaceName + "v" + dv.getFriendlyVersionNumber() + ".zip"; + success = false; + String fileName = baseFileName + ".zip"; // Add BagIt ZIP file // Although DuraCloud uses SHA-256 internally, it's API uses MD5 to verify the // transfer + messageDigest = MessageDigest.getInstance("MD5"); - try (PipedInputStream in = new PipedInputStream(); DigestInputStream digestInputStream2 = new DigestInputStream(in, messageDigest)) { - new Thread(new Runnable() { + try (PipedInputStream in = new PipedInputStream(); + DigestInputStream digestInputStream2 = new DigestInputStream(in, messageDigest)) { + Thread bagThread = new Thread(new Runnable() { public void run() { - try (PipedOutputStream out = new PipedOutputStream(in)){ + try (PipedOutputStream out = new PipedOutputStream(in)) { // Generate bag BagGenerator bagger = new BagGenerator(new OREMap(dv, false), dataciteXml); + bagger.setNumConnections(bagThreads); bagger.setAuthenticationKey(token.getTokenString()); bagger.generateBag(out); + success = true; } catch (Exception e) { logger.severe("Error creating bag: " + e.getMessage()); // TODO Auto-generated catch block e.printStackTrace(); - throw new RuntimeException("Error creating bag: " + e.getMessage()); } } - }).start(); - i=0; - while(digestInputStream.available()<=0 && i<100) { + }); + bagThread.start(); + i = 0; + while (digestInputStream.available() <= 0 && i < 100) { Thread.sleep(10); i++; } - checksum = store.addContent(spaceName, fileName, digestInputStream2, -1l, null, null, - null); - logger.fine("Content: " + fileName + " added with checksum: " + checksum); - localchecksum = Hex.encodeHexString(digestInputStream2.getMessageDigest().digest()); - if (!checksum.equals(localchecksum)) { - logger.severe(checksum + " not equal to " + localchecksum); + checksum = store.addContent(spaceName, fileName, digestInputStream2, -1l, null, null, null); + bagThread.join(); + if (success) { + logger.fine("Content: " + fileName + " added with checksum: " + checksum); + localchecksum = Hex.encodeHexString(digestInputStream2.getMessageDigest().digest()); + } + if (!success || !checksum.equals(localchecksum)) { + logger.severe("Failure on " + fileName); + logger.severe(success ? checksum + " not equal to " + localchecksum : "failed to transfer to DuraCloud"); + try { + store.deleteContent(spaceName, fileName); + store.deleteContent(spaceName, baseFileName + "_datacite.xml"); + } catch (ContentStoreException cse) { + logger.warning(cse.getMessage()); + } return new Failure("Error in transferring Zip file to DuraCloud", "DuraCloud Submission Failure: incomplete archive transfer"); } - } catch (RuntimeException rte) { - logger.severe(rte.getMessage()); - return new Failure("Error in generating Bag", - "DuraCloud Submission Failure: archive file not created"); } logger.fine("DuraCloud Submission step: Content Transferred"); @@ -179,10 +235,6 @@ public void run() { e.printStackTrace(); return new Failure("Error in transferring file to DuraCloud", "DuraCloud Submission Failure: archive file not transferred"); - } catch (RuntimeException rte) { - logger.severe(rte.getMessage()); - return new Failure("Error in generating datacite.xml file", - "DuraCloud Submission Failure: metadata file not created"); } catch (InterruptedException e) { logger.warning(e.getLocalizedMessage()); e.printStackTrace(); @@ -194,12 +246,13 @@ public void run() { if (!(1 == dv.getVersion()) || !(0 == dv.getMinorVersionNumber())) { mesg = mesg + ": Prior Version archiving not yet complete?"; } - return new Failure("Unable to create DuraCloud space with name: " + spaceName, mesg); + return new Failure("Unable to create DuraCloud space with name: " + baseFileName, mesg); } catch (NoSuchAlgorithmException e) { logger.severe("MD5 MessageDigest not available!"); } } else { - logger.warning("DuraCloud Submision Workflow aborted: Dataset locked for finalizePublication, or because file validation failed"); + logger.warning( + "DuraCloud Submision Workflow aborted: Dataset locked for finalizePublication, or because file validation failed"); return new Failure("Dataset locked"); } return WorkflowStepResult.OK; @@ -207,5 +260,5 @@ public void run() { return new Failure("DuraCloud Submission not configured - no \":DuraCloudHost\"."); } } - + } diff --git a/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/GoogleCloudSubmitToArchiveCommand.java b/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/GoogleCloudSubmitToArchiveCommand.java index cb729a9807a..6ea7afcc734 100644 --- a/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/GoogleCloudSubmitToArchiveCommand.java +++ b/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/GoogleCloudSubmitToArchiveCommand.java @@ -33,6 +33,7 @@ import com.google.cloud.storage.Blob; import com.google.cloud.storage.Bucket; import com.google.cloud.storage.Storage; +import com.google.cloud.storage.StorageException; import com.google.cloud.storage.StorageOptions; @RequiredPermissions(Permission.PublishDataset) @@ -42,6 +43,8 @@ public class GoogleCloudSubmitToArchiveCommand extends AbstractSubmitToArchiveCo private static final String GOOGLECLOUD_BUCKET = ":GoogleCloudBucket"; private static final String GOOGLECLOUD_PROJECT = ":GoogleCloudProject"; + boolean success = false; + public GoogleCloudSubmitToArchiveCommand(DataverseRequest aRequest, DatasetVersion version) { super(aRequest, version); } @@ -55,7 +58,7 @@ public WorkflowStepResult performArchiveSubmission(DatasetVersion dv, ApiToken t if (bucketName != null && projectName != null) { Storage storage; try { - FileInputStream fis = new FileInputStream(System.getProperty("dataverse.files.directory") + System.getProperty("file.separator")+ "googlecloudkey.json"); + FileInputStream fis = new FileInputStream(System.getProperty("dataverse.files.directory") + System.getProperty("file.separator") + "googlecloudkey.json"); storage = StorageOptions.newBuilder() .setCredentials(ServiceAccountCredentials.fromStream(fis)) .setProjectId(projectName) @@ -73,42 +76,51 @@ public WorkflowStepResult performArchiveSubmission(DatasetVersion dv, ApiToken t Map metadata = dc.getDataCiteMetadata(); String dataciteXml = DOIDataCiteRegisterService.getMetadataFromDvObject( dv.getDataset().getGlobalId().asString(), metadata, dv.getDataset()); - String blobIdString = null; MessageDigest messageDigest = MessageDigest.getInstance("MD5"); try (PipedInputStream dataciteIn = new PipedInputStream(); DigestInputStream digestInputStream = new DigestInputStream(dataciteIn, messageDigest)) { // Add datacite.xml file - new Thread(new Runnable() { + Thread dcThread = new Thread(new Runnable() { public void run() { try (PipedOutputStream dataciteOut = new PipedOutputStream(dataciteIn)) { dataciteOut.write(dataciteXml.getBytes(Charset.forName("utf-8"))); dataciteOut.close(); + success = true; } catch (Exception e) { logger.severe("Error creating datacite.xml: " + e.getMessage()); // TODO Auto-generated catch block e.printStackTrace(); - throw new RuntimeException("Error creating datacite.xml: " + e.getMessage()); + // throw new RuntimeException("Error creating datacite.xml: " + e.getMessage()); } } - }).start(); - //Have seen broken pipe in PostPublishDataset workflow without this delay - int i=0; - while(digestInputStream.available()<=0 && i<100) { + }); + dcThread.start(); + // Have seen broken pipe in PostPublishDataset workflow without this delay + int i = 0; + while (digestInputStream.available() <= 0 && i < 100) { Thread.sleep(10); i++; } - Blob dcXml = bucket.create(spaceName + "/datacite.v" + dv.getFriendlyVersionNumber()+".xml", digestInputStream, "text/xml", Bucket.BlobWriteOption.doesNotExist()); + Blob dcXml = bucket.create(spaceName + "/datacite.v" + dv.getFriendlyVersionNumber() + ".xml", digestInputStream, "text/xml", Bucket.BlobWriteOption.doesNotExist()); + + dcThread.join(); String checksum = dcXml.getMd5ToHexString(); logger.fine("Content: datacite.xml added with checksum: " + checksum); String localchecksum = Hex.encodeHexString(digestInputStream.getMessageDigest().digest()); - if (!checksum.equals(localchecksum)) { - logger.severe(checksum + " not equal to " + localchecksum); + if (!success || !checksum.equals(localchecksum)) { + logger.severe(success ? checksum + " not equal to " + localchecksum : "datacite.xml transfer did not succeed"); + try { + dcXml.delete(Blob.BlobSourceOption.generationMatch()); + } catch (StorageException se) { + logger.warning(se.getMessage()); + } return new Failure("Error in transferring DataCite.xml file to GoogleCloud", "GoogleCloud Submission Failure: incomplete metadata transfer"); } // Store BagIt file + success = false; String fileName = spaceName + ".v" + dv.getFriendlyVersionNumber() + ".zip"; // Add BagIt ZIP file @@ -123,13 +135,14 @@ public void run() { BagGenerator bagger = new BagGenerator(new OREMap(dv, false), dataciteXml); bagger.setAuthenticationKey(token.getTokenString()); bagger.generateBag(out); + success=true; } catch (Exception e) { logger.severe("Error creating bag: " + e.getMessage()); // TODO Auto-generated catch block e.printStackTrace(); try { digestInputStream2.close(); - } catch(Exception ex) { + } catch (Exception ex) { logger.warning(ex.getLocalizedMessage()); } throw new RuntimeException("Error creating bag: " + e.getMessage()); @@ -165,48 +178,46 @@ public void run() { * increased, and/or a change in how archives are sent to google (e.g. as * multiple blobs that get aggregated) would be required. */ - i=0; - while(digestInputStream2.available()<=90000 && i<2000 && writeThread.isAlive()) { + i = 0; + while (digestInputStream2.available() <= 90000 && i < 2000 && writeThread.isAlive()) { Thread.sleep(1000); logger.fine("avail: " + digestInputStream2.available() + " : " + writeThread.getState().toString()); i++; } logger.fine("Bag: transfer started, i=" + i + ", avail = " + digestInputStream2.available()); - if(i==2000) { + if (i == 2000) { throw new IOException("Stream not available"); } Blob bag = bucket.create(spaceName + "/" + fileName, digestInputStream2, "application/zip", Bucket.BlobWriteOption.doesNotExist()); - if(bag.getSize()==0) { + if (bag.getSize() == 0) { throw new IOException("Empty Bag"); } - blobIdString = bag.getBlobId().getBucket() + "/" + bag.getBlobId().getName(); + writeThread.join(); + checksum = bag.getMd5ToHexString(); logger.fine("Bag: " + fileName + " added with checksum: " + checksum); localchecksum = Hex.encodeHexString(digestInputStream2.getMessageDigest().digest()); - if (!checksum.equals(localchecksum)) { - logger.severe(checksum + " not equal to " + localchecksum); + if (!success || !checksum.equals(localchecksum)) { + logger.severe(success ? checksum + " not equal to " + localchecksum : "bag transfer did not succeed"); + try { + bag.delete(Blob.BlobSourceOption.generationMatch()); + } catch (StorageException se) { + logger.warning(se.getMessage()); + } return new Failure("Error in transferring Zip file to GoogleCloud", "GoogleCloud Submission Failure: incomplete archive transfer"); } - } catch (RuntimeException rte) { - logger.severe("Error creating Bag during GoogleCloud archiving: " + rte.getMessage()); - return new Failure("Error in generating Bag", - "GoogleCloud Submission Failure: archive file not created"); } logger.fine("GoogleCloud Submission step: Content Transferred"); // Document the location of dataset archival copy location (actually the URL - // where you can - // view it as an admin) + // where you can view it as an admin) + // Changed to point at bucket where the zip and datacite.xml are visible StringBuffer sb = new StringBuffer("https://console.cloud.google.com/storage/browser/"); - sb.append(blobIdString); + sb.append(bucketName + "/" + spaceName); dv.setArchivalCopyLocation(sb.toString()); - } catch (RuntimeException rte) { - logger.severe("Error creating datacite xml file during GoogleCloud Archiving: " + rte.getMessage()); - return new Failure("Error in generating datacite.xml file", - "GoogleCloud Submission Failure: metadata file not created"); } } else { logger.warning("GoogleCloud Submision Workflow aborted: Dataset locked for pidRegister"); From cf673d1c12d75b6fdd9ac06396cac808ae0460ee Mon Sep 17 00:00:00 2001 From: qqmyers Date: Wed, 13 Apr 2022 18:52:00 -0400 Subject: [PATCH 014/192] remove bag thread support for now --- .../command/impl/DuraCloudSubmitToArchiveCommand.java | 11 ----------- 1 file changed, 11 deletions(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/DuraCloudSubmitToArchiveCommand.java b/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/DuraCloudSubmitToArchiveCommand.java index b3b303d7407..de63eeca754 100644 --- a/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/DuraCloudSubmitToArchiveCommand.java +++ b/src/main/java/edu/harvard/iq/dataverse/engine/command/impl/DuraCloudSubmitToArchiveCommand.java @@ -41,10 +41,8 @@ public class DuraCloudSubmitToArchiveCommand extends AbstractSubmitToArchiveComm private static final String DURACLOUD_PORT = ":DuraCloudPort"; private static final String DURACLOUD_HOST = ":DuraCloudHost"; private static final String DURACLOUD_CONTEXT = ":DuraCloudContext"; - private static final int DEFAULT_THREADS = 2; boolean success = false; - int bagThreads = DEFAULT_THREADS; public DuraCloudSubmitToArchiveCommand(DataverseRequest aRequest, DatasetVersion version) { super(aRequest, version); } @@ -59,14 +57,6 @@ public WorkflowStepResult performArchiveSubmission(DatasetVersion dv, ApiToken t : DEFAULT_CONTEXT; String host = requestedSettings.get(DURACLOUD_HOST); - if (requestedSettings.get(BagGenerator.BAG_GENERATOR_THREADS) != null) { - try { - bagThreads=Integer.valueOf(requestedSettings.get(BagGenerator.BAG_GENERATOR_THREADS)); - } catch (NumberFormatException nfe) { - logger.warning("Can't parse the value of setting " + BagGenerator.BAG_GENERATOR_THREADS + " as an integer - using default:" + DEFAULT_THREADS); - } - } - if (host != null) { Dataset dataset = dv.getDataset(); // ToDo - change after HDC 3A changes to status reporting @@ -177,7 +167,6 @@ public void run() { try (PipedOutputStream out = new PipedOutputStream(in)) { // Generate bag BagGenerator bagger = new BagGenerator(new OREMap(dv, false), dataciteXml); - bagger.setNumConnections(bagThreads); bagger.setAuthenticationKey(token.getTokenString()); bagger.generateBag(out); success = true; From 3bc644fa2822b8c9cd770d335b42b547f28d4267 Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Tue, 19 Apr 2022 14:13:07 +0200 Subject: [PATCH 015/192] removed unused import --- .../authorization/providers/builtin/DataverseUserPage.java | 1 - 1 file changed, 1 deletion(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java index 5dcdc919d4c..850f14d6ce3 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java @@ -64,7 +64,6 @@ import org.apache.commons.lang3.StringUtils; import org.hibernate.validator.constraints.NotBlank; import org.primefaces.event.TabChangeEvent; -import javax.faces.event.AjaxBehaviorEvent; /** * From 70e01d93513e2f1621016fbd59657062e5f1d4c4 Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Tue, 19 Apr 2022 14:14:12 +0200 Subject: [PATCH 016/192] resolved merge conflict --- .../iq/dataverse/authorization/users/AuthenticatedUser.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java b/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java index 8d4ab1432fb..c250556f5dc 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java @@ -4,7 +4,7 @@ import edu.harvard.iq.dataverse.DatasetLock; import edu.harvard.iq.dataverse.UserNotification; import edu.harvard.iq.dataverse.UserNotification.Type; -import edu.harvard.iq.dataverse.ValidateEmail; +import edu.harvard.iq.dataverse.validation.ValidateEmail; import edu.harvard.iq.dataverse.authorization.AuthenticatedUserDisplayInfo; import edu.harvard.iq.dataverse.authorization.AuthenticatedUserLookup; import edu.harvard.iq.dataverse.authorization.providers.oauth2.OAuth2TokenData; From 4fd7e53bd1bf51e4f15dfbaa77763433392137f5 Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Tue, 19 Apr 2022 14:21:31 +0200 Subject: [PATCH 017/192] renames of the notification variables in AuthenticatedUser --- .../providers/builtin/DataverseUserPage.java | 32 +++++++++---------- src/main/webapp/dataverseuser.xhtml | 4 +-- 2 files changed, 18 insertions(+), 18 deletions(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java index 850f14d6ce3..c158011739c 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java @@ -141,8 +141,8 @@ public enum EditMode { private List notificationTypeList; - private Set mutedEmailList; - private Set mutedNotificationList; + private Set mutedEmails; + private Set mutedNotifications; private Set disabledNotifications; public String init() { @@ -174,10 +174,10 @@ public String init() { notificationTypeList = Arrays.asList(Type.values()).stream() .filter(x -> !Type.CONFIRMEMAIL.equals(x) && x.hasDescription()) .collect(Collectors.toList()); - mutedEmailList = new HashSet<>(currentUser.getMutedEmails()); - mutedEmailList.addAll(settingsWrapper.getAlwaysMutedSet()); - mutedNotificationList = new HashSet<>(currentUser.getMutedNotifications()); - mutedNotificationList.addAll(settingsWrapper.getAlwaysMutedSet()); + mutedEmails = new HashSet<>(currentUser.getMutedEmails()); + mutedEmails.addAll(settingsWrapper.getAlwaysMutedSet()); + mutedNotifications = new HashSet<>(currentUser.getMutedNotifications()); + mutedNotifications.addAll(settingsWrapper.getAlwaysMutedSet()); disabledNotifications = new HashSet<>(settingsWrapper.getAlwaysMutedSet()); disabledNotifications.addAll(settingsWrapper.getNeverMutedSet()); @@ -387,8 +387,8 @@ public String save() { logger.info("Redirecting"); return permissionsWrapper.notAuthorized() + "faces-redirect=true"; }else { - currentUser.setMutedEmails(mutedEmailList); - currentUser.setMutedNotifications(mutedNotificationList); + currentUser.setMutedEmails(mutedEmails); + currentUser.setMutedNotifications(mutedNotifications); String emailBeforeUpdate = currentUser.getEmail(); AuthenticatedUser savedUser = authenticationService.updateAuthenticatedUser(currentUser, userDisplayInfo); String emailAfterUpdate = savedUser.getEmail(); @@ -738,20 +738,20 @@ public void setNotificationTypeList(List notificationTypeList) { this.notificationTypeList = notificationTypeList; } - public Set getMutedEmailList() { - return mutedEmailList; + public Set getMutedEmails() { + return mutedEmails; } - public void setMutedEmailList(Set mutedEmailList) { - this.mutedEmailList = mutedEmailList; + public void setMutedEmails(Set mutedEmails) { + this.mutedEmails = mutedEmails; } - public Set getMutedNotificationList() { - return mutedNotificationList; + public Set getMutedNotifications() { + return mutedNotifications; } - public void setMutedNotificationList(Set mutedNotificationList) { - this.mutedNotificationList = mutedNotificationList; + public void setMutedNotifications(Set mutedNotifications) { + this.mutedNotifications = mutedNotifications; } public boolean isDisabled(Type t) { diff --git a/src/main/webapp/dataverseuser.xhtml b/src/main/webapp/dataverseuser.xhtml index 875c93190a1..91e852d5ccf 100644 --- a/src/main/webapp/dataverseuser.xhtml +++ b/src/main/webapp/dataverseuser.xhtml @@ -384,7 +384,7 @@ #{bundle['notification.mutedEmails']}
- +
@@ -394,7 +394,7 @@ #{bundle['notification.mutedNotifications']}
- +
From 8d2e3ce517f279115c8941ae5edc204afac42329 Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Tue, 19 Apr 2022 14:28:01 +0200 Subject: [PATCH 018/192] resolved merge conflict --- .../iq/dataverse/authorization/users/AuthenticatedUser.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java b/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java index c250556f5dc..1706ed846f8 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/users/AuthenticatedUser.java @@ -2,8 +2,8 @@ import edu.harvard.iq.dataverse.Cart; import edu.harvard.iq.dataverse.DatasetLock; -import edu.harvard.iq.dataverse.UserNotification; import edu.harvard.iq.dataverse.UserNotification.Type; +import edu.harvard.iq.dataverse.UserNotification; import edu.harvard.iq.dataverse.validation.ValidateEmail; import edu.harvard.iq.dataverse.authorization.AuthenticatedUserDisplayInfo; import edu.harvard.iq.dataverse.authorization.AuthenticatedUserLookup; From cf13a0b8fd1698afa3d9e2909e76edc30d730192 Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Mon, 25 Apr 2022 16:14:20 +0200 Subject: [PATCH 019/192] muting options are now collapsible and the logic is inverterd: you select notifications that you want to receive, not to muted --- .../providers/builtin/DataverseUserPage.java | 16 ++--- src/main/java/propertyFiles/Bundle.properties | 7 ++- src/main/webapp/dataverseuser.xhtml | 63 +++++++++++-------- 3 files changed, 50 insertions(+), 36 deletions(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java index fdf2ad546c9..ae01c22207b 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java @@ -738,20 +738,20 @@ public void setNotificationTypeList(List notificationTypeList) { this.notificationTypeList = notificationTypeList; } - public Set getMutedEmails() { - return mutedEmails; + public Set getToReceiveEmails() { + return notificationTypeList.stream().filter(x -> !mutedEmails.contains(x)).collect(Collectors.toSet()); } - public void setMutedEmails(Set mutedEmails) { - this.mutedEmails = mutedEmails; + public void setToReceiveEmails(Set toReceiveEmails) { + this.mutedEmails = notificationTypeList.stream().filter(x -> !toReceiveEmails.contains(x)).collect(Collectors.toSet()); } - public Set getMutedNotifications() { - return mutedNotifications; + public Set getToReceiveNotifications() { + return notificationTypeList.stream().filter(x -> !mutedNotifications.contains(x)).collect(Collectors.toSet()); } - public void setMutedNotifications(Set mutedNotifications) { - this.mutedNotifications = mutedNotifications; + public void setToReceiveNotifications(Set toReceiveNotifications) { + this.mutedNotifications = notificationTypeList.stream().filter(x -> !toReceiveNotifications.contains(x)).collect(Collectors.toSet()); } public boolean isDisabled(Type t) { diff --git a/src/main/java/propertyFiles/Bundle.properties b/src/main/java/propertyFiles/Bundle.properties index 567a63a248d..72d6fda8e64 100644 --- a/src/main/java/propertyFiles/Bundle.properties +++ b/src/main/java/propertyFiles/Bundle.properties @@ -237,9 +237,10 @@ notification.import.filesystem=Dataset {1}, dataset had file checksums added via a batch job. removeNotification=Remove Notification -# These are the labels of the comboboxes where the muted notifications can be selected by the users -notification.mutedEmails=Muted email notification types -notification.mutedNotifications=Muted notification types +# These are the labels of the options where the muted notifications can be selected by the users +notification.muteOptions=Notifications settings +notification.mutedEmails=Select the email notifications you wish to receive: +notification.mutedNotifications=Select the in-app notifications you wish to receive: # Notification types descriptions as presentend to the user. Leave the description empty or leave it out entirely in order to make it not selectable for muting by the user. notification.typeDescription.ASSIGNROLE=Role is assigned diff --git a/src/main/webapp/dataverseuser.xhtml b/src/main/webapp/dataverseuser.xhtml index 91e852d5ccf..78553a83c5d 100644 --- a/src/main/webapp/dataverseuser.xhtml +++ b/src/main/webapp/dataverseuser.xhtml @@ -59,6 +59,44 @@ +
+ +
+
+
+ +
+ + + +
+
+
+ +
+ + + +
+
+ +
+
+
+
@@ -377,31 +415,6 @@
-
-
-
- -
- - - -
-
-
- -
- - - -
-
- -
From e5e80d88d7f2f5dbf82008a4135401bb53b6e14d Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Mon, 25 Apr 2022 17:03:09 +0200 Subject: [PATCH 020/192] bugfix in inverted muting logic --- .../providers/builtin/DataverseUserPage.java | 18 ++++++++++++------ src/main/java/propertyFiles/Bundle.properties | 2 +- 2 files changed, 13 insertions(+), 7 deletions(-) diff --git a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java index ae01c22207b..93f84527298 100644 --- a/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java +++ b/src/main/java/edu/harvard/iq/dataverse/authorization/providers/builtin/DataverseUserPage.java @@ -175,9 +175,7 @@ public String init() { .filter(x -> !Type.CONFIRMEMAIL.equals(x) && x.hasDescription()) .collect(Collectors.toList()); mutedEmails = new HashSet<>(currentUser.getMutedEmails()); - mutedEmails.addAll(settingsWrapper.getAlwaysMutedSet()); mutedNotifications = new HashSet<>(currentUser.getMutedNotifications()); - mutedNotifications.addAll(settingsWrapper.getAlwaysMutedSet()); disabledNotifications = new HashSet<>(settingsWrapper.getAlwaysMutedSet()); disabledNotifications.addAll(settingsWrapper.getNeverMutedSet()); @@ -739,19 +737,27 @@ public void setNotificationTypeList(List notificationTypeList) { } public Set getToReceiveEmails() { - return notificationTypeList.stream().filter(x -> !mutedEmails.contains(x)).collect(Collectors.toSet()); + return notificationTypeList.stream().filter( + x -> isDisabled(x) ? !settingsWrapper.isAlwaysMuted(x) && settingsWrapper.isNeverMuted(x) : !mutedEmails.contains(x) + ).collect(Collectors.toSet()); } public void setToReceiveEmails(Set toReceiveEmails) { - this.mutedEmails = notificationTypeList.stream().filter(x -> !toReceiveEmails.contains(x)).collect(Collectors.toSet()); + this.mutedEmails = notificationTypeList.stream().filter( + x -> !isDisabled(x) && !toReceiveEmails.contains(x) + ).collect(Collectors.toSet()); } public Set getToReceiveNotifications() { - return notificationTypeList.stream().filter(x -> !mutedNotifications.contains(x)).collect(Collectors.toSet()); + return notificationTypeList.stream().filter( + x -> isDisabled(x) ? !settingsWrapper.isAlwaysMuted(x) && settingsWrapper.isNeverMuted(x) : !mutedNotifications.contains(x) + ).collect(Collectors.toSet()); } public void setToReceiveNotifications(Set toReceiveNotifications) { - this.mutedNotifications = notificationTypeList.stream().filter(x -> !toReceiveNotifications.contains(x)).collect(Collectors.toSet()); + this.mutedNotifications = notificationTypeList.stream().filter( + x -> !isDisabled(x) && !toReceiveNotifications.contains(x) + ).collect(Collectors.toSet()); } public boolean isDisabled(Type t) { diff --git a/src/main/java/propertyFiles/Bundle.properties b/src/main/java/propertyFiles/Bundle.properties index 72d6fda8e64..eba5e7e71e7 100644 --- a/src/main/java/propertyFiles/Bundle.properties +++ b/src/main/java/propertyFiles/Bundle.properties @@ -238,7 +238,7 @@ notification.import.checksum= + action="#{DataverseUserPage.save}" onclick="PF('muteCollapse').collapse('hide')"/>
From f2b26fe875ba4d8159136ca20f5058551e926913 Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Tue, 26 Apr 2022 12:23:04 +0200 Subject: [PATCH 022/192] when saving notification settings the settings panel collapses; added update property --- src/main/webapp/dataverseuser.xhtml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/webapp/dataverseuser.xhtml b/src/main/webapp/dataverseuser.xhtml index d00553fbee7..1a10e04735a 100644 --- a/src/main/webapp/dataverseuser.xhtml +++ b/src/main/webapp/dataverseuser.xhtml @@ -92,7 +92,7 @@ + action="#{DataverseUserPage.save}" onclick="PF('muteCollapse').collapse('hide')" update="@form"/> From a83cfffaf04d22aa890198e4ddf2a27315e540ec Mon Sep 17 00:00:00 2001 From: Eryk Kulikowski Date: Wed, 27 Apr 2022 10:52:34 +0200 Subject: [PATCH 023/192] fixed inconsistencies when switching tabs and improved accessibility --- src/main/webapp/dataverseuser.xhtml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/main/webapp/dataverseuser.xhtml b/src/main/webapp/dataverseuser.xhtml index 1a10e04735a..97d00f7c326 100644 --- a/src/main/webapp/dataverseuser.xhtml +++ b/src/main/webapp/dataverseuser.xhtml @@ -53,14 +53,14 @@ - +
-