blob: 87755cf44adf6aa1f8a97f89c53c77bcf904bc73 [file] [log] [blame]
package de.ids_mannheim.korap;
import com.fasterxml.jackson.annotation.*;
import com.fasterxml.jackson.annotation.JsonInclude.Include;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.fasterxml.jackson.databind.node.ArrayNode;
import de.ids_mannheim.korap.index.PositionsToOffset;
import de.ids_mannheim.korap.index.SearchContext;
import de.ids_mannheim.korap.response.KorapResponse;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.ArrayList;
import java.util.List;
/*
TODO: Reuse the KorapSearch code for data serialization!
*/
/**
* Response class for search results.
*
* TODO: Synopsis
*
* @author diewald
* @see KorapResponse
*/
@JsonInclude(Include.NON_NULL)
@JsonIgnoreProperties(ignoreUnknown = true)
public class KorapResult extends KorapResponse {
ObjectMapper mapper = new ObjectMapper();
@JsonIgnore
public static final short ITEMS_PER_PAGE = 25;
public static final short ITEMS_PER_PAGE_MAX = 100;
private int startIndex = 0;
private String query;
private List<KorapMatch> matches;
private SearchContext context;
private short
itemsPerPage = ITEMS_PER_PAGE,
itemsPerResource = 0;
private JsonNode request;
// Logger
// This is KorapMatch instead of KorapResult!
private final static Logger log = LoggerFactory.getLogger(KorapMatch.class);
/**
* Construct a new KorapResult object.
*/
public KorapResult() {
mapper.enable(SerializationFeature.INDENT_OUTPUT);
};
/**
* Construct a new KorapResult object.
*
* @param query Query representation as a string.
* @param startIndex Offset position in match array.
* @param itemsPerPage Number of matches per page.
* @param context Requested {@link SearchContext}
*/
public KorapResult(String query,
int startIndex,
short itemsPerPage,
SearchContext context) {
mapper.enable(SerializationFeature.INDENT_OUTPUT);
// mapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS);
// mapper.disable(SerializationFeature.WRITE_NULL_MAP_VALUES);
this.matches = new ArrayList<>(itemsPerPage);
this.query = query;
this.startIndex = startIndex;
this.itemsPerPage =
(itemsPerPage > ITEMS_PER_PAGE_MAX || itemsPerPage < 1) ?
ITEMS_PER_PAGE : itemsPerPage;
this.context = context;
};
/**
* Add a new match to the result set.
*
* @param match A {@link KorapMatch} to add.
*/
public void add (KorapMatch km) {
this.matches.add(km);
};
/**
* Get the number of items (documents) shown per page.
*
* @return Number of items shown per page.
*/
public short getItemsPerPage () {
return this.itemsPerPage;
};
/**
* Set the number of items (documents) shown per page.
*
* @param count Number of items shown per page.
* @return {@link KorapResult} object for chaining.
*/
public KorapResult setItemsPerPage (short count) {
this.itemsPerPage = count;
return this;
};
/**
* Get serialized query as a {@link JsonNode}.
*
* @return {@link JsonNode} representation of the query object.
*/
public JsonNode getRequest () {
return this.request;
};
/**
* Set serialized query as a {@link JsonNode}.
*
* @param request {@link JsonNode} representation of the query object.
* @return {@link KorapResult} object for chaining.
*/
public KorapResult setRequest (JsonNode request) {
this.request = request;
return this;
};
/**
* Get the number of items shown per resource (document).
* Defaults to <tt>0</tt>, which is infinite.
*
* @return The number of items shown per resource.
*/
public short getItemsPerResource () {
return this.itemsPerResource;
};
/**
* Set the number of items (matches) shown per resource (text).
* Defaults to <tt>0</tt>, which is infinite.
*
* @param value The number of items shown per resource.
* @return {@link KorapResult} object for chaining.
*/
public KorapResult setItemsPerResource (short value) {
this.itemsPerResource = value;
return this;
};
/**
* Set the number of items (matches) shown per resource (document).
* Defaults to <tt>0</tt>, which is infinite.
*
* @param value The number of items shown per resource.
* @return {@link KorapResult} object for chaining.
*/
public KorapResult setItemsPerResource (int value) {
this.itemsPerResource = (short) value;
return this;
};
/**
* Get the string representation of the search query.
*
* @return The string representation of the search query.
*/
public String getQuery () {
return this.query;
};
/**
* Get a certain {@link KorapMatch} by index.
*
* @param index The numerical index of the match,
* starts with <tt>0</tt>.
* @return The {@link KorapMatch} object.
*/
@JsonIgnore
public KorapMatch getMatch (int index) {
return this.matches.get(index);
};
/**
* Get the list of {@link KorapMatch} matches.
*
* @return The list of {@link KorapMatch} objects.
*/
public List<KorapMatch> getMatches() {
return this.matches;
};
/**
* Get the number of the first match in the result set
* (<i>aka</i> the offset). Starts with <tt>0</tt>.
*
* @return The index number of the first match in the result set.
*/
public int getStartIndex () {
return startIndex;
};
/**
* Get the context parameters of the search by means of a
* {@link SearchContext} object.
*
* @return The {@link SearchContext} object.
*/
public SearchContext getContext () {
return this.context;
};
/**
* Set the context parameters of the search by means of a
* {@link SearchContext} object.
*
* @param context The {@link SearchContext} object providing
* search context parameters.
* @return {@link KorapResult} object for chaining.
*/
public KorapResult setContext (SearchContext context) {
this.context = context;
return this;
};
/**
* Serialize the result set as a {@link JsonNode}.
*
* @return {@link JsonNode} representation of the search results.
*/
public JsonNode toJsonNode () {
ObjectNode json = (ObjectNode) mapper.valueToTree(super.toJsonNode());
// Relevant context setting
if (this.context != null)
json.put("context", this.getContext().toJsonNode());
// ItemsPerPage
json.put("itemsPerPage", this.itemsPerPage);
// Relevant itemsPerResource setting
if (this.itemsPerResource > 0)
json.put("itemsPerResource", this.itemsPerResource);
json.put("startIndex", this.startIndex);
// Add matches
if (this.matches != null)
json.putPOJO("matches", this.getMatches());
// TODO: <test>
if (this.request != null)
json.put("request", this.request);
if (this.query != null)
json.put("query", this.query);
// </test>
return json;
};
// For Collocation Analysis API
@Deprecated
public String toTokenListJsonString () {
ObjectNode json = (ObjectNode) mapper.valueToTree(this);
ArrayNode array = json.putArray("matches");
// Add matches as token lists
for (KorapMatch km : this.getMatches())
array.add(km.toTokenList());
try {
return mapper.writeValueAsString(json);
}
catch (Exception e) {
log.warn(e.getLocalizedMessage());
};
return "{}";
};
};