From 2a17ba889f0c1483e5b2ffba516084f1051df747 Mon Sep 17 00:00:00 2001 From: Andrea Bollini Date: Sat, 12 Oct 2024 19:39:05 +0200 Subject: [PATCH 1/8] DSC-1970 reduce noice in the log for client abort --- .../exception/DSpaceApiExceptionControllerAdvice.java | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/dspace-server-webapp/src/main/java/org/dspace/app/rest/exception/DSpaceApiExceptionControllerAdvice.java b/dspace-server-webapp/src/main/java/org/dspace/app/rest/exception/DSpaceApiExceptionControllerAdvice.java index 54c7815d83e4..ded7684b00a4 100644 --- a/dspace-server-webapp/src/main/java/org/dspace/app/rest/exception/DSpaceApiExceptionControllerAdvice.java +++ b/dspace-server-webapp/src/main/java/org/dspace/app/rest/exception/DSpaceApiExceptionControllerAdvice.java @@ -19,6 +19,7 @@ import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; +import org.apache.catalina.connector.ClientAbortException; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; import org.dspace.app.exception.ResourceAlreadyExistsException; @@ -268,6 +269,14 @@ protected ResponseEntity handleTypeMismatch(TypeMismatchException ex, Ht return super.handleTypeMismatch(ex, headers, HttpStatus.BAD_REQUEST, request); } + @ExceptionHandler(ClientAbortException.class) + public void clientAbortExceptionHandler(HttpServletRequest request, ClientAbortException e) { + // This usually means the browser closed or disconnected or + // something. We can't do anything. To avoid excessive stack traces + // in log, just print a simple message + log.warn("ClientAbortException"); + } + @ExceptionHandler(Exception.class) protected void handleGenericException(HttpServletRequest request, HttpServletResponse response, Exception ex) throws IOException { From ed9963894ff5eda1a412cfe22b98dd9b2f786925 Mon Sep 17 00:00:00 2001 From: Andrea Bollini Date: Sat, 12 Oct 2024 19:40:19 +0200 Subject: [PATCH 2/8] DSC-1970 reduce noice in the log due to beanutils excessive WARN --- dspace/config/log4j2.xml | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/dspace/config/log4j2.xml b/dspace/config/log4j2.xml index b2d1c33d20ae..6b8feefdffa5 100644 --- a/dspace/config/log4j2.xml +++ b/dspace/config/log4j2.xml @@ -188,6 +188,13 @@ + + + From 3b975c887b9f5c2d30da3b208a8efb16a5ee8c90 Mon Sep 17 00:00:00 2001 From: Andrea Bollini Date: Mon, 14 Oct 2024 13:19:25 +0200 Subject: [PATCH 3/8] DSC-1970 expose additional config to reduce log noise --- dspace/config/dspace.cfg | 2 ++ 1 file changed, 2 insertions(+) diff --git a/dspace/config/dspace.cfg b/dspace/config/dspace.cfg index fbdeb2bc863f..6ea5bad89788 100644 --- a/dspace/config/dspace.cfg +++ b/dspace/config/dspace.cfg @@ -1801,6 +1801,8 @@ context-menu-entry.requestcorrection.enabled = true context-menu-entry.statistics.enabled = true context-menu-entry.subscriptions.enabled = true context-menu-entry.itemversion.enabled = true +context-menu-entry.orcidview.enabled = true +context-menu-entry.fullitem.enabled = true ################################################ ################ CrossWalk configurations ########### From 602cfcf9d64d8672e9a55f3dd7cd8d9fea9b27d6 Mon Sep 17 00:00:00 2001 From: Andrea Bollini Date: Tue, 15 Oct 2024 15:45:35 +0200 Subject: [PATCH 4/8] DSC-1970 avoid noice from hibernate for lookup of deleted items --- dspace/config/log4j2.xml | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/dspace/config/log4j2.xml b/dspace/config/log4j2.xml index 6b8feefdffa5..b867d0313d88 100644 --- a/dspace/config/log4j2.xml +++ b/dspace/config/log4j2.xml @@ -194,7 +194,12 @@ --> - + + From 99b914f17265199d8b7f4f7f861f9fccc4d093b9 Mon Sep 17 00:00:00 2001 From: Andrea Bollini Date: Sun, 13 Oct 2024 13:19:53 +0200 Subject: [PATCH 5/8] DSC-1968 avoid NPE, assure db connection is released --- .../enhancer/impl/RelatedEntityItemEnhancer.java | 7 +++++-- .../content/enhancer/script/ItemEnhancerScript.java | 13 +++++-------- .../enhancer/RelatedItemEnhancerUpdatePoller.java | 3 +-- 3 files changed, 11 insertions(+), 12 deletions(-) diff --git a/dspace-api/src/main/java/org/dspace/content/enhancer/impl/RelatedEntityItemEnhancer.java b/dspace-api/src/main/java/org/dspace/content/enhancer/impl/RelatedEntityItemEnhancer.java index 870dfbddfa8e..4d3bd44fb927 100644 --- a/dspace-api/src/main/java/org/dspace/content/enhancer/impl/RelatedEntityItemEnhancer.java +++ b/dspace-api/src/main/java/org/dspace/content/enhancer/impl/RelatedEntityItemEnhancer.java @@ -15,6 +15,7 @@ import java.util.List; import java.util.Map; import java.util.Map.Entry; +import java.util.Objects; import java.util.Optional; import java.util.Set; import java.util.UUID; @@ -134,9 +135,11 @@ private boolean equivalent(Map> currMetadataValues, } private boolean equivalent(List metadataValue, List metadataValueDTO) { - if (metadataValue.size() != metadataValueDTO.size()) { + if ((Objects.isNull(metadataValue) && !Objects.isNull(metadataValueDTO)) || + (!Objects.isNull(metadataValue) && Objects.isNull(metadataValueDTO)) || + metadataValue.size() != metadataValueDTO.size()) { return false; - } else { + } else if (!Objects.isNull(metadataValue) && !Objects.isNull(metadataValueDTO)) { for (int i = 0; i < metadataValue.size(); i++) { if (!equivalent(metadataValue.get(i), metadataValueDTO.get(i))) { return false; diff --git a/dspace-api/src/main/java/org/dspace/content/enhancer/script/ItemEnhancerScript.java b/dspace-api/src/main/java/org/dspace/content/enhancer/script/ItemEnhancerScript.java index c965edd6766d..968acb3b477c 100644 --- a/dspace-api/src/main/java/org/dspace/content/enhancer/script/ItemEnhancerScript.java +++ b/dspace-api/src/main/java/org/dspace/content/enhancer/script/ItemEnhancerScript.java @@ -73,9 +73,9 @@ private void enhanceItems(Context context) { try { int total = itemService.countArchivedItems(context); for (int offset = 0; offset < total; offset += PAGE_SIZE) { + // commit and session clear within the enhance method + // do one by one to reduce the risk of conflict with the enhance thread findItemsToEnhance(offset).forEachRemaining(this::enhanceItem); - context.commit(); - context.clear(); } } catch (SQLException e) { throw new SQLRuntimeException(e); @@ -91,13 +91,10 @@ private Iterator findItemsToEnhance(int offset) { } private void enhanceItem(Item item) { - itemEnhancerService.enhance(context, item, force); - uncacheItem(item); - } - - private void uncacheItem(Item item) { try { - context.uncacheEntity(item); + itemEnhancerService.enhance(context, item, force); + context.commit(); + context.clear(); } catch (SQLException e) { throw new SQLRuntimeException(e); } diff --git a/dspace-server-webapp/src/main/java/org/dspace/app/rest/enhancer/RelatedItemEnhancerUpdatePoller.java b/dspace-server-webapp/src/main/java/org/dspace/app/rest/enhancer/RelatedItemEnhancerUpdatePoller.java index 1d341b61cce9..16de405de629 100644 --- a/dspace-server-webapp/src/main/java/org/dspace/app/rest/enhancer/RelatedItemEnhancerUpdatePoller.java +++ b/dspace-server-webapp/src/main/java/org/dspace/app/rest/enhancer/RelatedItemEnhancerUpdatePoller.java @@ -33,9 +33,8 @@ public class RelatedItemEnhancerUpdatePoller { @Scheduled(fixedDelayString = "${related-item-enhancer-poller.delay}") public void pollItemToUpdateAndProcess() { - try { + try (Context context = new Context();) { log.debug("item enhancer poller executed"); - Context context = new Context(); context.setDispatcher(RelatedItemEnhancerUpdatePoller.class.getSimpleName()); context.turnOffAuthorisationSystem(); UUID extractedUuid; From 30682d20496b61f18852073d459cadf3c6b89127 Mon Sep 17 00:00:00 2001 From: Andrea Bollini Date: Tue, 15 Oct 2024 11:06:43 +0200 Subject: [PATCH 6/8] DSC-1969 improve performance and allow to use the preventMetadataSecurity projection also for inprogress submission --- .../security/MetadataSecurityServiceImpl.java | 78 +++++++++---------- dspace/config/modules/public-metadata.cfg | 16 +++- 2 files changed, 53 insertions(+), 41 deletions(-) diff --git a/dspace-api/src/main/java/org/dspace/content/security/MetadataSecurityServiceImpl.java b/dspace-api/src/main/java/org/dspace/content/security/MetadataSecurityServiceImpl.java index 603ddb228f25..e190f107b5c8 100644 --- a/dspace-api/src/main/java/org/dspace/content/security/MetadataSecurityServiceImpl.java +++ b/dspace-api/src/main/java/org/dspace/content/security/MetadataSecurityServiceImpl.java @@ -20,7 +20,7 @@ import javax.annotation.PostConstruct; import javax.annotation.Resource; -import org.apache.commons.collections4.CollectionUtils; +import org.apache.commons.lang3.StringUtils; import org.dspace.app.util.DCInputSet; import org.dspace.app.util.DCInputsReader; import org.dspace.app.util.DCInputsReaderException; @@ -123,11 +123,10 @@ public boolean checkMetadataFieldVisibility(Context context, Item item, Metadata return isMetadataFieldVisible(context, boxes, item, metadataField, false); } - private List getPermissionFilteredMetadata(Context context, Item item, List metadataValues, boolean preventBoxSecurityCheck) { - if (item.isWithdrawn() && isNotAdmin(context)) { + if (item.isWithdrawn() && isNotAdmin(context, item)) { return new ArrayList(); } @@ -177,42 +176,17 @@ private boolean isMetadataValueVisible(Context context, List boxe private boolean isMetadataFieldVisible(Context context, List boxes, Item item, MetadataField metadataField, boolean preventBoxSecurityCheck) { - if (CollectionUtils.isNotEmpty(boxes)) { - return isMetadataFieldVisibleByBoxes(context, boxes, item, metadataField, preventBoxSecurityCheck); - } - return isNotAdmin(context) ? isMetadataFieldVisibleFor(context, item, metadataField) : true; - } - - private boolean isMetadataFieldVisibleFor(Context context, Item item, MetadataField metadataField) { - return canEditItem(context, item) || isNotHidden(context, metadataField); - } - - private boolean isMetadataValueReturnAllowed(Context context, Item item, MetadataValue metadataValue) { - Integer securityLevel = metadataValue.getSecurityLevel(); - if (securityLevel == null) { - return true; - } - - MetadataSecurityEvaluation metadataSecurityEvaluation = getMetadataSecurityEvaluator(securityLevel); - try { - return metadataSecurityEvaluation.allowMetadataFieldReturn(context, item, metadataValue.getMetadataField()); - } catch (SQLException e) { - throw new SQLRuntimeException(e); - } - } - - private boolean isMetadataFieldVisibleByBoxes(Context context, List boxes, Item item, - MetadataField metadataField, boolean preventBoxSecurityCheck) { if (isPublicMetadataField(metadataField, boxes, preventBoxSecurityCheck)) { return true; } + // stop any further costly check if the converter request the fastest strategy if (preventBoxSecurityCheck) { return false; } - EPerson currentUser = context.getCurrentUser(); + EPerson currentUser = context != null ? context.getCurrentUser() : null; List notPublicBoxes = getNotPublicBoxes(metadataField, boxes); if (Objects.nonNull(currentUser)) { @@ -232,13 +206,43 @@ private boolean isMetadataFieldVisibleByBoxes(Context context, List boxes, boolean preventBoxSecurityCheck) { - List publicFields = preventBoxSecurityCheck ? getPublicMetadataFromConfig() : getPublicMetadata(boxes); + List publicFields = preventBoxSecurityCheck || boxes.isEmpty() ? + getPublicMetadataFromConfig() : getPublicMetadata(boxes); return publicFields.stream() - .anyMatch(publicField -> publicField.equals(metadataField.toString('.'))); + .anyMatch(publicField -> metadataMatch(metadataField, publicField)); + } + + private boolean metadataMatch(MetadataField metadataField, String publicField) { + if (publicField.contains(".*")) { + final String exactMatch = publicField.replace(".*", ""); + StringBuffer qualifiedMatch = new StringBuffer(exactMatch).append("."); + return exactMatch.equals(metadataField.toString('.')) || + StringUtils.startsWith(metadataField.toString('.'), qualifiedMatch.toString()); + } else { + return publicField.equals(metadataField.toString('.')); + } } private List getPublicMetadataFromConfig() { @@ -291,13 +295,9 @@ private List dcInputsSet(final String sd) { } } - private boolean isAdmin(Context context) { - return !isNotAdmin(context); - } - - private boolean isNotAdmin(Context context) { + private boolean isNotAdmin(Context context, Item item) { try { - return context == null || !authorizeService.isAdmin(context); + return context == null || !authorizeService.isAdmin(context, item); } catch (SQLException e) { throw new SQLRuntimeException(e); } @@ -311,7 +311,7 @@ private List getFromSubmission(Context context, List Date: Wed, 16 Oct 2024 16:29:05 +0200 Subject: [PATCH 7/8] DSC-1969 avoid NPE when patching the whole /metadata path --- .../content/security/MetadataSecurityServiceImpl.java | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/dspace-api/src/main/java/org/dspace/content/security/MetadataSecurityServiceImpl.java b/dspace-api/src/main/java/org/dspace/content/security/MetadataSecurityServiceImpl.java index e190f107b5c8..02af6dd20d8a 100644 --- a/dspace-api/src/main/java/org/dspace/content/security/MetadataSecurityServiceImpl.java +++ b/dspace-api/src/main/java/org/dspace/content/security/MetadataSecurityServiceImpl.java @@ -235,6 +235,9 @@ private boolean isPublicMetadataField(MetadataField metadataField, List dcInputSets, MetadataFiel private boolean isNotHidden(Context context, MetadataField metadataField) { try { - return !metadataExposureService.isHidden(context, metadataField.getMetadataSchema().getName(), - metadataField.getElement(), metadataField.getQualifier()); + return metadataField != null && + !metadataExposureService.isHidden(context, metadataField.getMetadataSchema().getName(), + metadataField.getElement(), metadataField.getQualifier()); } catch (SQLException e) { throw new SQLRuntimeException(e); } From 6d5042515f2448f352f5c0cdd45427ff195619dc Mon Sep 17 00:00:00 2001 From: Andrea Bollini Date: Sat, 12 Oct 2024 18:16:21 +0200 Subject: [PATCH 8/8] DSC-1960 avoid exception if the query is empty --- .../dspace/app/customurl/service/CustomUrlServiceImpl.java | 5 ++++- .../test/java/org/dspace/app/rest/ItemRestRepositoryIT.java | 4 ++++ 2 files changed, 8 insertions(+), 1 deletion(-) diff --git a/dspace-api/src/main/java/org/dspace/app/customurl/service/CustomUrlServiceImpl.java b/dspace-api/src/main/java/org/dspace/app/customurl/service/CustomUrlServiceImpl.java index cceaa005f86f..b6939aa11dda 100644 --- a/dspace-api/src/main/java/org/dspace/app/customurl/service/CustomUrlServiceImpl.java +++ b/dspace-api/src/main/java/org/dspace/app/customurl/service/CustomUrlServiceImpl.java @@ -17,6 +17,7 @@ import java.util.stream.Stream; import org.apache.commons.collections.CollectionUtils; +import org.apache.commons.lang3.StringUtils; import org.dspace.app.customurl.CustomUrlService; import org.dspace.content.Item; import org.dspace.content.MetadataValue; @@ -129,7 +130,9 @@ public void deleteOldCustomUrlByIndex(Context context, Item item, int index) { @Override @SuppressWarnings("rawtypes") public Optional findItemByCustomUrl(Context context, String customUrl) { - + if (StringUtils.isBlank(customUrl)) { + return Optional.empty(); + } DiscoverQuery discoverQuery = new DiscoverQuery(); discoverQuery.addDSpaceObjectFilter(IndexableItem.TYPE); discoverQuery.addFilterQueries("customurl:" + searchService.escapeQueryChars(customUrl)); diff --git a/dspace-server-webapp/src/test/java/org/dspace/app/rest/ItemRestRepositoryIT.java b/dspace-server-webapp/src/test/java/org/dspace/app/rest/ItemRestRepositoryIT.java index b5584f295125..edbb58332f4c 100644 --- a/dspace-server-webapp/src/test/java/org/dspace/app/rest/ItemRestRepositoryIT.java +++ b/dspace-server-webapp/src/test/java/org/dspace/app/rest/ItemRestRepositoryIT.java @@ -5287,6 +5287,10 @@ public void testSearchItemByCustomUrlWithoutResult() throws Exception { .param("q", "http://example.com/sample")) .andExpect(status().isNoContent()); + getClient(token).perform(get("/api/core/items/search/findByCustomURL") + .param("q", "")) + .andExpect(status().isNoContent()); + } @Test