diff --git a/dspace-api/src/main/java/org/dspace/discovery/SolrServiceImpl.java b/dspace-api/src/main/java/org/dspace/discovery/SolrServiceImpl.java index 813787bff105..4930b9cee165 100644 --- a/dspace-api/src/main/java/org/dspace/discovery/SolrServiceImpl.java +++ b/dspace-api/src/main/java/org/dspace/discovery/SolrServiceImpl.java @@ -109,6 +109,13 @@ public class SolrServiceImpl implements SearchService, IndexingService { // facet by indexing "each word to end of value' partial value public static final String SOLR_FIELD_SUFFIX_FACET_PREFIXES = "_prefix"; + // Suffix of the solr field used to index the facet/filter so that the facet search can search all word in a + // facet. + private static final String SOLR_FACET_FIELD_ALL_VALUES_SUFFIX = "_filter"; + + // List of all facets which will return facet value with splitter. + private ArrayList allValuesFacetList = new ArrayList<>(); + @Autowired protected ContentServiceFactory contentServiceFactory; @Autowired @@ -1400,12 +1407,22 @@ protected String transformFacetField(DiscoverFacetField facetFieldConfig, String } } else if (facetFieldConfig.getType().equals(DiscoveryConfigurationParameters.TYPE_HIERARCHICAL)) { if (removePostfix) { + // If the current field is configured to show all values instead of only the top level values + if (this.getFacetsToShowAllValues().contains( + StringUtils.substringBeforeLast(field, SOLR_FACET_FIELD_ALL_VALUES_SUFFIX))) { + return StringUtils.substringBeforeLast(field, SOLR_FACET_FIELD_ALL_VALUES_SUFFIX); + } return StringUtils.substringBeforeLast(field, "_tax_"); } else { + // If the current field is configured to show all values instead of only the top level values + if (this.getFacetsToShowAllValues().contains(field)) { + return field + SOLR_FACET_FIELD_ALL_VALUES_SUFFIX; + } //Only display top level filters ! return field + "_tax_0_filter"; } } else if (facetFieldConfig.getType().equals(DiscoveryConfigurationParameters.TYPE_AUTHORITY)) { + if (removePostfix) { return field.substring(0, field.lastIndexOf("_acid")); } else { @@ -1603,4 +1620,15 @@ public String calculateExtremeValue(Context context, String valueField, return null; } + /** + * Return or load the configuration property `discovery.solr.facets.allvalues` as a list. + */ + private ArrayList getFacetsToShowAllValues() { + if (CollectionUtils.isEmpty(allValuesFacetList)) { + String[] allValuesFacetArray = configurationService.getArrayProperty("discovery.solr.facets.allvalues"); + Collections.addAll(allValuesFacetList, allValuesFacetArray); + } + return allValuesFacetList; + } + } diff --git a/dspace-server-webapp/src/test/data/dspaceFolder/config/spring/api/test-discovery.xml b/dspace-server-webapp/src/test/data/dspaceFolder/config/spring/api/test-discovery.xml index 4a91ef051e88..68726b343086 100644 --- a/dspace-server-webapp/src/test/data/dspaceFolder/config/spring/api/test-discovery.xml +++ b/dspace-server-webapp/src/test/data/dspaceFolder/config/spring/api/test-discovery.xml @@ -34,6 +34,7 @@ + @@ -144,6 +145,166 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + (search.resourcetype:Item) OR search.resourcetype:Collection OR search.resourcetype:Community + -withdrawn:true AND -discoverable:false + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + dc.title + dc.contributor.author + dc.creator + dc.contributor.other + dc.subject + + + + + + + + + + + + + + @@ -1114,5 +1275,18 @@ + + + + + dc.subject.* + + + + + + + + diff --git a/dspace-server-webapp/src/test/java/org/dspace/app/rest/ClarinDiscoveryRestControllerIT.java b/dspace-server-webapp/src/test/java/org/dspace/app/rest/ClarinDiscoveryRestControllerIT.java index fed5d27da68f..79b0289c114c 100644 --- a/dspace-server-webapp/src/test/java/org/dspace/app/rest/ClarinDiscoveryRestControllerIT.java +++ b/dspace-server-webapp/src/test/java/org/dspace/app/rest/ClarinDiscoveryRestControllerIT.java @@ -1001,7 +1001,8 @@ public void discoverSearchTest() throws Exception { SearchFilterMatcher.clarinLicenseRightsFilter(), SearchFilterMatcher.clarinItemsLanguageFilter(), SearchFilterMatcher.clarinItemsCommunityFilter(), - SearchFilterMatcher.clarinItemsTypeFilter() + SearchFilterMatcher.clarinItemsTypeFilter(), + SearchFilterMatcher.clarinSubjectFirstValueFilter() ))) //These sortOptions need to be present as it's the default in the configuration .andExpect(jsonPath("$.sortOptions", contains( @@ -5947,4 +5948,112 @@ public void discoverFacetsTestWithDsoTypeTest() throws Exception { .andExpect(jsonPath("$._embedded.values").value(Matchers.hasSize(1))); } + + @Test + public void showFacetValuesWithSplitterInSearchPage() throws Exception { + context.turnOffAuthorisationSystem(); + + parentCommunity = CommunityBuilder.createCommunity(context) + .withName("Parent Community").build(); + + Collection col1 = CollectionBuilder.createCollection(context, parentCommunity) + .withName("Collection 1").build(); + + ItemBuilder.createItem(context, col1) + .withTitle("Public item 1") + .withIssueDate("2017-10-17") + .withAuthor("Smith, Donald") + .withSubject("People") + .build(); + + ItemBuilder.createItem(context, col1) + .withTitle("Public item 2") + .withIssueDate("2020-02-13") + .withAuthor("Doe, Jane") + .withSubject("People::Jane") + .build(); + + ItemBuilder.createItem(context, col1) + .withTitle("Public item 2") + .withIssueDate("2020-02-13") + .withAuthor("Doe, Jane") + .withSubject("People::Jane") + .build(); + + context.restoreAuthSystemState(); + + getClient().perform(get("/api/discover/facets/subject")) + .andExpect(status().isOk()) + .andExpect(jsonPath("$.type", is("discover"))) + .andExpect(jsonPath("$.name", is("subject"))) + .andExpect(jsonPath("$.facetType", is("hierarchical"))) + .andExpect(jsonPath("$.scope", is(emptyOrNullString()))) + .andExpect(jsonPath("$._links.self.href", + containsString("api/discover/facets/subject"))) + .andExpect(jsonPath("$._embedded.values[0].label", is("People"))) + .andExpect(jsonPath("$._embedded.values[0].count", is(3))) + .andExpect(jsonPath("$._embedded.values[1].label", is("People::Jane"))) + .andExpect(jsonPath("$._embedded.values[1].count", is(2))) + .andExpect(jsonPath("$", JsonPathMatchers.hasNoJsonPath("_embedded.values[2].label"))) + .andExpect(jsonPath("$", JsonPathMatchers.hasNoJsonPath("_embedded.values[2].count"))) + .andExpect(jsonPath("$._embedded.values").value(Matchers.hasSize(2))); + } + + @Test + public void doNotShowFacetValuesWithSplitterInHomePage() throws Exception { + context.turnOffAuthorisationSystem(); + + parentCommunity = CommunityBuilder.createCommunity(context) + .withName("Parent Community").build(); + + Collection col1 = CollectionBuilder.createCollection(context, parentCommunity) + .withName("Collection 1").build(); + + ItemBuilder.createItem(context, col1) + .withTitle("Public item 1") + .withIssueDate("2017-10-17") + .withAuthor("Smith, Donald") + .withSubject("People") + .build(); + + ItemBuilder.createItem(context, col1) + .withTitle("Public item 2") + .withIssueDate("2020-02-13") + .withAuthor("Doe, Jane") + .withSubject("People::Jane") + .build(); + + ItemBuilder.createItem(context, col1) + .withTitle("Public item 2") + .withIssueDate("2020-02-13") + .withAuthor("Doe, Jane") + .withSubject("People::Jane") + .build(); + + ItemBuilder.createItem(context, col1) + .withTitle("Public item 2") + .withIssueDate("2020-02-13") + .withAuthor("Doe, Jane") + .withSubject("Another subject") + .build(); + + context.restoreAuthSystemState(); + + getClient().perform(get("/api/discover/facets/subjectFirstValue") + .param("configuration", "homepage")) + .andExpect(status().isOk()) + .andExpect(jsonPath("$.type", is("discover"))) + .andExpect(jsonPath("$.name", is("subjectFirstValue"))) + .andExpect(jsonPath("$.facetType", is("hierarchical"))) + .andExpect(jsonPath("$.scope", is(emptyOrNullString()))) + .andExpect(jsonPath("$._links.self.href", + containsString("api/discover/facets/subjectFirstValue"))) + .andExpect(jsonPath("$._embedded.values[0].label", is("People"))) + .andExpect(jsonPath("$._embedded.values[0].count", is(3))) + .andExpect(jsonPath("$._embedded.values[1].label", is("Another subject"))) + .andExpect(jsonPath("$._embedded.values[1].count", is(1))) + .andExpect(jsonPath("$", JsonPathMatchers.hasNoJsonPath("_embedded.values[2].label"))) + .andExpect(jsonPath("$", JsonPathMatchers.hasNoJsonPath("_embedded.values[2].count"))) + .andExpect(jsonPath("$._embedded.values").value(Matchers.hasSize(2))); + } } diff --git a/dspace-server-webapp/src/test/java/org/dspace/app/rest/DiscoveryVersioningIT.java b/dspace-server-webapp/src/test/java/org/dspace/app/rest/DiscoveryVersioningIT.java index 083b27d0e547..53e8dd77e3c3 100644 --- a/dspace-server-webapp/src/test/java/org/dspace/app/rest/DiscoveryVersioningIT.java +++ b/dspace-server-webapp/src/test/java/org/dspace/app/rest/DiscoveryVersioningIT.java @@ -68,6 +68,7 @@ import org.junit.After; import org.junit.Assert; import org.junit.Before; +import org.junit.Ignore; import org.junit.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.test.web.servlet.request.MockHttpServletRequestBuilder; @@ -290,6 +291,8 @@ protected Collection createCollection(String name, String entityType) { return collection; } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_default_expectLatestVersionsOnly() throws Exception { final String configuration = null; @@ -367,6 +370,8 @@ public void test_discoveryXml_defaultRelationships_allVersions() throws Exceptio )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_site_expectLatestVersionsOnly() throws Exception { final String configuration = "site"; @@ -405,6 +410,8 @@ public void test_discoveryXml_site_expectLatestVersionsOnly() throws Exception { )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_workspace_expectLatestVersionsOnly() throws Exception { final String configuration = "workspace"; @@ -444,9 +451,9 @@ public void test_discoveryXml_workspace_expectLatestVersionsOnly() throws Except } // NOTE: no test for discovery.xml configuration "workflow", because it by definition excludes items - // NOTE: no test for discovery.xml configuration "workflowAdmin", because it by definition excludes items - + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_undiscoverable_expectLatestVersionsOnly() throws Exception { final String configuration = "undiscoverable"; @@ -488,6 +495,8 @@ public void test_discoveryXml_undiscoverable_expectLatestVersionsOnly() throws E )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_administrativeView_expectLatestVersionsOnly() throws Exception { final String configuration = "administrativeView"; @@ -522,6 +531,8 @@ public void test_discoveryXml_administrativeView_expectLatestVersionsOnly() thro )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_publication_expectLatestVersionsOnly() throws Exception { final String configuration = "publication"; @@ -593,6 +604,8 @@ public void test_discoveryXml_publicationRelationships_allVersions() throws Exce )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_person_expectLatestVersionsOnly() throws Exception { final String configuration = "person"; @@ -664,6 +677,8 @@ public void test_discoveryXml_personRelationships_allVersions() throws Exception )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_orgunit_expectLatestVersionsOnly() throws Exception { final String configuration = "orgunit"; @@ -735,6 +750,8 @@ public void test_discoveryXml_orgunitRelationships_allVersions() throws Exceptio )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_journalissue_expectLatestVersionsOnly() throws Exception { final String configuration = "journalissue"; @@ -806,6 +823,8 @@ public void test_discoveryXml_journalissueRelationships_allVersions() throws Exc )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_journalvolume_expectLatestVersionsOnly() throws Exception { final String configuration = "journalvolume"; @@ -878,6 +897,8 @@ public void test_discoveryXml_journalvolumeRelationships_allVersions() throws Ex )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_journal_expectLatestVersionsOnly() throws Exception { final String configuration = "journal"; @@ -950,6 +971,8 @@ public void test_discoveryXml_journalRelationships_allVersions() throws Exceptio )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_project_expectLatestVersionsOnly() throws Exception { final String configuration = "project"; @@ -1021,6 +1044,8 @@ public void test_discoveryXml_projectRelationships_allVersions() throws Exceptio )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_personOrOrgunit_expectLatestVersionsOnly() throws Exception { final String configuration = "personOrOrgunit"; @@ -1056,6 +1081,8 @@ public void test_discoveryXml_personOrOrgunit_expectLatestVersionsOnly() throws )); } + // CLARIN - All items are indexed by all versions + @Ignore @Test public void test_discoveryXml_openAIREFundingAgency_expectLatestVersionsOnly() throws Exception { final String configuration = "openAIREFundingAgency"; diff --git a/dspace-server-webapp/src/test/java/org/dspace/app/rest/matcher/SearchFilterMatcher.java b/dspace-server-webapp/src/test/java/org/dspace/app/rest/matcher/SearchFilterMatcher.java index b60f26940ff8..47bf950c98ed 100644 --- a/dspace-server-webapp/src/test/java/org/dspace/app/rest/matcher/SearchFilterMatcher.java +++ b/dspace-server-webapp/src/test/java/org/dspace/app/rest/matcher/SearchFilterMatcher.java @@ -206,4 +206,14 @@ public static Matcher publisherFilter() { ); } + + public static Matcher clarinSubjectFirstValueFilter() { + return allOf( + hasJsonPath("$.filter", is("subjectFirstValue")), + hasJsonPath("$.hasFacets", is(false)), + hasJsonPath("$.type", is("hierarchical")), + hasJsonPath("$.openByDefault", is(false)), + checkOperators() + ); + } } diff --git a/dspace/config/clarin-dspace.cfg b/dspace/config/clarin-dspace.cfg index a0be8953a26e..b1a589afd3a9 100644 --- a/dspace/config/clarin-dspace.cfg +++ b/dspace/config/clarin-dspace.cfg @@ -263,5 +263,8 @@ item-page.show-handle-and-doi = false #### Searching properties #### -# Filter sort options for the search +# Filter sort options for the search - sorting options in the search page sort.options.filtered = dc.date.accessioned +# Show splitters and all values for the these hierarchical facets +# `subject` - hierarchical facet defined in the `discovery.xml` +discovery.solr.facets.allvalues = subject diff --git a/dspace/config/spring/api/discovery.xml b/dspace/config/spring/api/discovery.xml index da2a9894789c..ab12f23e8dee 100644 --- a/dspace/config/spring/api/discovery.xml +++ b/dspace/config/spring/api/discovery.xml @@ -201,6 +201,7 @@ + @@ -223,7 +224,7 @@ - (search.resourcetype:Item AND latestVersion:true) OR search.resourcetype:Collection OR search.resourcetype:Community + (search.resourcetype:Item) OR search.resourcetype:Collection OR search.resourcetype:Community -withdrawn:true AND -discoverable:false @@ -328,7 +329,7 @@ - + @@ -345,6 +346,7 @@ + @@ -382,7 +384,7 @@ - (search.resourcetype:Item AND latestVersion:true) OR search.resourcetype:Collection OR search.resourcetype:Community + (search.resourcetype:Item) OR search.resourcetype:Collection OR search.resourcetype:Community -withdrawn:true AND -discoverable:false @@ -675,7 +677,7 @@ - search.resourcetype:Item AND latestVersion:true + search.resourcetype:Item withdrawn:true OR discoverable:false @@ -820,7 +822,7 @@ - search.resourcetype:Item AND latestVersion:true + search.resourcetype:Item @@ -951,7 +953,7 @@ - (search.resourcetype:Item AND latestVersion:true) OR search.resourcetype:WorkspaceItem OR search.resourcetype:XmlWorkflowItem + (search.resourcetype:Item) OR search.resourcetype:WorkspaceItem OR search.resourcetype:XmlWorkflowItem @@ -1345,7 +1347,7 @@ - search.resourcetype:Item AND latestVersion:true AND entityType_keyword:Publication + search.resourcetype:Item AND entityType_keyword:Publication -withdrawn:true AND -discoverable:false @@ -1480,7 +1482,7 @@ - search.resourcetype:Item AND latestVersion:true AND entityType_keyword:Person + search.resourcetype:Item AND entityType_keyword:Person -withdrawn:true AND -discoverable:false @@ -1597,7 +1599,7 @@ - search.resourcetype:Item AND latestVersion:true AND entityType_keyword:Project + search.resourcetype:Item AND entityType_keyword:Project -withdrawn:true AND -discoverable:false @@ -1715,7 +1717,7 @@ - search.resourcetype:Item AND latestVersion:true AND entityType_keyword:OrgUnit + search.resourcetype:Item AND entityType_keyword:OrgUnit -withdrawn:true AND -discoverable:false @@ -1840,7 +1842,7 @@ - search.resourcetype:Item AND latestVersion:true AND entityType_keyword:JournalIssue + search.resourcetype:Item AND entityType_keyword:JournalIssue -withdrawn:true AND -discoverable:false @@ -1960,7 +1962,7 @@ - search.resourcetype:Item AND latestVersion:true AND entityType_keyword:JournalVolume + search.resourcetype:Item AND entityType_keyword:JournalVolume -withdrawn:true AND -discoverable:false @@ -2079,7 +2081,7 @@ - search.resourcetype:Item AND latestVersion:true AND entityType_keyword:Journal + search.resourcetype:Item AND entityType_keyword:Journal -withdrawn:true AND -discoverable:false @@ -2213,7 +2215,7 @@ - search.resourcetype:Item AND latestVersion:true AND (entityType_keyword:OrgUnit OR entityType_keyword:Person) + search.resourcetype:Item AND (entityType_keyword:OrgUnit OR entityType_keyword:Person) -withdrawn:true AND -discoverable:false @@ -2269,7 +2271,7 @@ - search.resourcetype:Item AND latestVersion:true AND entityType_keyword:OrgUnit AND dc.type:FundingOrganization + search.resourcetype:Item AND entityType_keyword:OrgUnit AND dc.type:FundingOrganization -withdrawn:true AND -discoverable:false @@ -2590,6 +2592,28 @@ + + + + + + + + + + dc.subject.* + + + + + + + diff --git a/scripts/envs/__basic.example.bat b/scripts/envs/__basic.example.bat index 731c7d4d95f6..bc50cb505244 100644 --- a/scripts/envs/__basic.example.bat +++ b/scripts/envs/__basic.example.bat @@ -3,4 +3,5 @@ set dspace_source=C:\workspace\DSpace\ set tomcat=C:\apache-tomcat-9.0.64\ set dspace_application=C:\dspace\ set m2_source=%USERPROFILE%\.m2 +set dspace_solr=C:\workspace\solr set dspace_source=C:\workspace\DSpace \ No newline at end of file diff --git a/scripts/fast-build/config-update.bat b/scripts/fast-build/config-update.bat index 800fc97205f2..6f80ae31ce7d 100644 --- a/scripts/fast-build/config-update.bat +++ b/scripts/fast-build/config-update.bat @@ -6,3 +6,5 @@ rem copy all config files xcopy /e /h /i /q /y %dspace_source%\dspace\config\ %dspace_application%\config\ cd %dspace_source%\scripts\fast-build\ + +call update-solr-configsets.bat diff --git a/scripts/fast-build/update-solr-configsets.bat b/scripts/fast-build/update-solr-configsets.bat new file mode 100644 index 000000000000..e5ca174ff29f --- /dev/null +++ b/scripts/fast-build/update-solr-configsets.bat @@ -0,0 +1,5 @@ +call ..\envs\__basic.bat + +rm -rf %dspace_solr%server\solr\configsets\authority %dspace_solr%server\solr\configsets\oai dspace_solr%server\solr\configsets\search dspace_solr%server\solr\configsets\statistics +xcopy /e /h /i /q /y %dspace_application%solr\ %dspace_solr%server\solr\configsets\ +