Skip to content

Commit

Permalink
RSS feed from search results (REST) (DSpace#9732)
Browse files Browse the repository at this point in the history
* 116466: Port fixes for rss on search to 7.6

* 116466: Add new tests for search filters and configuration

---------

Co-authored-by: Nathan Buckingham <[email protected]>
  • Loading branch information
ConfusionOrb221 and Nathan Buckingham authored Dec 19, 2024
1 parent a554053 commit 8ffc23b
Show file tree
Hide file tree
Showing 3 changed files with 153 additions and 59 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,9 @@
*/
package org.dspace.app.rest;

import static org.dspace.app.rest.utils.HttpHeadersInitializer.CONTENT_DISPOSITION;
import static org.dspace.app.rest.utils.HttpHeadersInitializer.CONTENT_DISPOSITION_INLINE;

import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
Expand All @@ -19,8 +22,11 @@
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import org.apache.commons.lang3.StringUtils;
import org.apache.logging.log4j.Logger;
import org.dspace.app.rest.parameter.SearchFilter;
import org.dspace.app.rest.utils.ContextUtil;
import org.dspace.app.rest.utils.RestDiscoverQueryBuilder;
import org.dspace.app.rest.utils.ScopeResolver;
import org.dspace.app.util.factory.UtilServiceFactory;
import org.dspace.app.util.service.OpenSearchService;
Expand All @@ -46,6 +52,9 @@
import org.dspace.discovery.configuration.DiscoverySortFieldConfiguration;
import org.dspace.discovery.indexobject.IndexableItem;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.GetMapping;
Expand Down Expand Up @@ -83,22 +92,28 @@ public class OpenSearchController {
@Autowired
private ScopeResolver scopeResolver;

@Autowired
private RestDiscoverQueryBuilder restDiscoverQueryBuilder;

/**
* This method provides the OpenSearch query on the path /search
* It will pass the result as a OpenSearchDocument directly to the client
*/
@GetMapping("/search")
public void search(HttpServletRequest request,
HttpServletResponse response,
@RequestParam(name = "query", required = false) String query,
@RequestParam(name = "start", required = false) Integer start,
@RequestParam(name = "rpp", required = false) Integer count,
@RequestParam(name = "format", required = false) String format,
@RequestParam(name = "sort", required = false) String sort,
@RequestParam(name = "sort_direction", required = false) String sortDirection,
@RequestParam(name = "scope", required = false) String dsoObject,
Model model) throws IOException, ServletException {
HttpServletResponse response,
@RequestParam(name = "query", required = false) String query,
@RequestParam(name = "start", required = false) Integer start,
@RequestParam(name = "rpp", required = false) Integer count,
@RequestParam(name = "format", required = false) String format,
@RequestParam(name = "sort", required = false) String sort,
@RequestParam(name = "sort_direction", required = false) String sortDirection,
@RequestParam(name = "scope", required = false) String dsoObject,
@RequestParam(name = "configuration", required = false) String configuration,
List<SearchFilter> searchFilters,
Model model) throws IOException, ServletException {
context = ContextUtil.obtainContext(request);

if (start == null) {
start = 0;
}
Expand Down Expand Up @@ -130,82 +145,103 @@ public void search(HttpServletRequest request,
// then the rest - we are processing the query
IndexableObject container = null;

// support pagination parameters
DiscoverQuery queryArgs = new DiscoverQuery();
if (query == null) {
query = "";
} else {
queryArgs.setQuery(query);
DiscoverQuery queryArgs;

DiscoveryConfiguration discoveryConfiguration = null;
if (StringUtils.isNotBlank(configuration)) {
discoveryConfiguration = searchConfigurationService.getDiscoveryConfiguration(configuration);
}
queryArgs.setStart(start);
queryArgs.setMaxResults(count);
queryArgs.setDSpaceObjectFilter(IndexableItem.TYPE);
if (discoveryConfiguration == null) {
discoveryConfiguration = searchConfigurationService.getDiscoveryConfiguration("default");
}
// If we have search filters, use RestDiscoverQueryBuilder.
if (searchFilters != null && searchFilters.size() > 0) {
IndexableObject scope = scopeResolver.resolveScope(context, dsoObject);
Sort pageSort = sort == null || sortDirection == null
? Sort.unsorted()
: Sort.by(new Sort.Order(Sort.Direction.fromString(sortDirection), sort));
// TODO count can't be < 1 so I put an arbitrary number
Pageable page = PageRequest.of(start, count > 0 ? count : 10, pageSort);
queryArgs = restDiscoverQueryBuilder.buildQuery(context, scope,
discoveryConfiguration, query, searchFilters, IndexableItem.TYPE, page);
queryArgs.setFacetMinCount(-1);
} else { // Else, use the older behavior.
// support pagination parameters
queryArgs = new DiscoverQuery();
if (query == null) {
query = "";
} else {
queryArgs.setQuery(query);
}
queryArgs.setStart(start);
queryArgs.setMaxResults(count);
queryArgs.setDSpaceObjectFilter(IndexableItem.TYPE);

if (sort != null) {
DiscoveryConfiguration discoveryConfiguration =
searchConfigurationService.getDiscoveryConfiguration("");
if (discoveryConfiguration != null) {
DiscoverySortConfiguration searchSortConfiguration = discoveryConfiguration
.getSearchSortConfiguration();
if (searchSortConfiguration != null) {
DiscoverySortFieldConfiguration sortFieldConfiguration = searchSortConfiguration
.getSortFieldConfiguration(sort);
if (sortFieldConfiguration != null) {
String sortField = searchService
.toSortFieldIndex(sortFieldConfiguration.getMetadataField(),
sortFieldConfiguration.getType());
if (sort != null) {
if (discoveryConfiguration != null) {
DiscoverySortConfiguration searchSortConfiguration = discoveryConfiguration
.getSearchSortConfiguration();
if (searchSortConfiguration != null) {
DiscoverySortFieldConfiguration sortFieldConfiguration = searchSortConfiguration
.getSortFieldConfiguration(sort);
if (sortFieldConfiguration != null) {
String sortField = searchService
.toSortFieldIndex(sortFieldConfiguration.getMetadataField(),
sortFieldConfiguration.getType());

if (sortDirection != null && sortDirection.equals("DESC")) {
queryArgs.setSortField(sortField, SORT_ORDER.desc);
if (sortDirection != null && sortDirection.equals("DESC")) {
queryArgs.setSortField(sortField, SORT_ORDER.desc);
} else {
queryArgs.setSortField(sortField, SORT_ORDER.asc);
}
} else {
queryArgs.setSortField(sortField, SORT_ORDER.asc);
throw new IllegalArgumentException(sort + " is not a valid sort field");
}
} else {
throw new IllegalArgumentException(sort + " is not a valid sort field");
}
}
} else {
// this is the default sort so we want to switch this to date accessioned
queryArgs.setSortField("dc.date.accessioned_dt", SORT_ORDER.desc);
}
} else {
// this is the default sort so we want to switch this to date accessioned
queryArgs.setSortField("dc.date.accessioned_dt", SORT_ORDER.desc);
}

if (dsoObject != null) {
container = scopeResolver.resolveScope(context, dsoObject);
DiscoveryConfiguration discoveryConfiguration = searchConfigurationService
.getDiscoveryConfiguration(context, container);
queryArgs.setDiscoveryConfigurationName(discoveryConfiguration.getId());
queryArgs.addFilterQueries(discoveryConfiguration.getDefaultFilterQueries()
.toArray(
new String[discoveryConfiguration.getDefaultFilterQueries()
.size()]));
if (dsoObject != null) {
container = scopeResolver.resolveScope(context, dsoObject);
discoveryConfiguration = searchConfigurationService
.getDiscoveryConfigurationByNameOrIndexableObject(context, "site", container);
queryArgs.setDiscoveryConfigurationName(discoveryConfiguration.getId());
queryArgs.addFilterQueries(discoveryConfiguration.getDefaultFilterQueries()
.toArray(
new String[discoveryConfiguration.getDefaultFilterQueries()
.size()]));
}
}

// Perform the search
DiscoverResult qResults = null;
try {
qResults = SearchUtils.getSearchService().search(context,
container, queryArgs);
container, queryArgs);
} catch (SearchServiceException e) {
log.error(LogHelper.getHeader(context, "opensearch", "query="
+ queryArgs.getQuery()
+ ",error=" + e.getMessage()), e);
+ queryArgs.getQuery()
+ ",error=" + e.getMessage()), e);
throw new RuntimeException(e.getMessage(), e);
}

// Log
log.info("opensearch done, query=\"" + query + "\",results="
+ qResults.getTotalSearchResults());
+ qResults.getTotalSearchResults());

List<IndexableObject> dsoResults = qResults.getIndexableObjects();
Document resultsDoc = openSearchService.getResultsDoc(context, format, query,
(int) qResults.getTotalSearchResults(), qResults.getStart(),
qResults.getMaxResults(), container, dsoResults);
(int) qResults.getTotalSearchResults(), qResults.getStart(),
qResults.getMaxResults(), container, dsoResults);
try {
Transformer xf = TransformerFactory.newInstance().newTransformer();
response.setContentType(openSearchService.getContentType(format));
response.addHeader(CONTENT_DISPOSITION, CONTENT_DISPOSITION_INLINE);
xf.transform(new DOMSource(resultsDoc),
new StreamResult(response.getWriter()));
new StreamResult(response.getWriter()));
} catch (TransformerException e) {
log.error(e);
throw new ServletException(e.toString());
Expand All @@ -226,7 +262,7 @@ public void search(HttpServletRequest request,
*/
@GetMapping("/service")
public void service(HttpServletRequest request,
HttpServletResponse response) throws IOException {
HttpServletResponse response) throws IOException {
log.debug("Show OpenSearch Service document");
if (openSearchService == null) {
openSearchService = UtilServiceFactory.getInstance().getOpenSearchService();
Expand All @@ -235,7 +271,7 @@ public void service(HttpServletRequest request,
String svcDescrip = openSearchService.getDescription(null);
log.debug("opensearchdescription is " + svcDescrip);
response.setContentType(openSearchService
.getContentType("opensearchdescription"));
.getContentType("opensearchdescription"));
response.setContentLength(svcDescrip.length());
response.getWriter().write(svcDescrip);
} else {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ public class HttpHeadersInitializer {
MULTIPART_BOUNDARY;
public static final String CONTENT_DISPOSITION_INLINE = "inline";
public static final String CONTENT_DISPOSITION_ATTACHMENT = "attachment";
public static final String CONTENT_DISPOSITION = "Content-Disposition";
private static final String IF_NONE_MATCH = "If-None-Match";
private static final String IF_MODIFIED_SINCE = "If-Modified-Since";
private static final String ETAG = "ETag";
Expand All @@ -52,7 +53,6 @@ public class HttpHeadersInitializer {
private static final String APPLICATION_OCTET_STREAM = "application/octet-stream";
private static final String IMAGE = "image";
private static final String ACCEPT = "Accept";
private static final String CONTENT_DISPOSITION = "Content-Disposition";
private static final String CONTENT_DISPOSITION_FORMAT = "%s;filename=\"%s\"";
private static final String CACHE_CONTROL = "Cache-Control";

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -316,4 +316,62 @@ public void scopeNotCommunityOrCollectionUUIDTest() throws Exception {
.andExpect(status().isOk())
.andExpect(xpath("feed/totalResults").string("1"));
}

@Test
public void configurationFilterAuthorTest() throws Exception {
context.turnOffAuthorisationSystem();
parentCommunity = CommunityBuilder.createCommunity(context)
.withName("Parent Community")
.build();
Collection collection1 = CollectionBuilder.createCollection(context, parentCommunity).withName("Collection 1")
.build();

Item publicItem1 = ItemBuilder.createItem(context, collection1)
.withTitle("Boars at Yellowstone")
.withIssueDate("2017-10-17")
.withAuthor("Ballini, Andreas").withAuthor("Moriarti, Susan")
.build();
Item publicItem2 = ItemBuilder.createItem(context, collection1)
.withTitle("Test Filter Item 2")
.withIssueDate("2020-10-20")
.withAuthor("Test Author")
.build();

getClient().perform(get("/opensearch/search")
.param("query", "*")
.param("configuration", "defaultConfiguration")
.param("f.author", "Test Author,equals"))
.andExpect(status().isOk())
.andExpect(xpath("feed/totalResults").string("1"));

}

@Test
public void configurationFilterEntityTypeTest() throws Exception {
context.turnOffAuthorisationSystem();
parentCommunity = CommunityBuilder.createCommunity(context)
.withName("Parent Community")
.build();
Collection collection1 = CollectionBuilder.createCollection(context, parentCommunity).withName("Collection 1")
.build();

Item publicItem1 = ItemBuilder.createItem(context, collection1)
.withTitle("Boars at Yellowstone")
.withIssueDate("2017-10-17")
.withAuthor("Ballini, Andreas").withAuthor("Moriarti, Susan")
.build();
Item publicItem2 = ItemBuilder.createItem(context, collection1)
.withTitle("Test Filter Item 2")
.withIssueDate("2020-10-20")
.withMetadata("dspace", "entity", "type", "Publication")
.build();

getClient().perform(get("/opensearch/search")
.param("query", "*")
.param("configuration", "defaultConfiguration")
.param("f.entityType", "Publication,equals"))
.andExpect(status().isOk())
.andExpect(xpath("feed/totalResults").string("1"));

}
}

0 comments on commit 8ffc23b

Please sign in to comment.