blob: 63ae3678c6907b6dbf251e897f5424d82af86873 [file] [log] [blame]
package de.ids_mannheim.korap.query.serialize;
import com.fasterxml.jackson.core.JsonFactory;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.collect.ArrayListMultimap;
import com.google.common.collect.Multimap;
import com.google.common.collect.Multiset;
import java.io.IOException;
import java.util.*;
/**
* @author hanl
* @date 06/12/2013
*/
public class CollectionQuery {
private static ObjectMapper serialzer = new ObjectMapper();
private JsonFactory factory;
private CollectionTypes types;
private List<Map> rq;
private Multimap<String, String> mfilter;
private Multimap<String, String> mextension;
public enum RELATION {
AND, OR
}
public CollectionQuery() {
this.rq = new ArrayList<>();
this.mfilter = ArrayListMultimap.create();
this.mextension = ArrayListMultimap.create();
this.factory = serialzer.getFactory();
this.types = new CollectionTypes();
}
public CollectionQuery addResource(String query) {
try {
List v = serialzer.readValue(query, LinkedList.class);
this.rq.addAll(v);
} catch (IOException e) {
throw new IllegalArgumentException("Conversion went wrong!");
}
return this;
}
public CollectionQuery addResources(List<String> queries) {
for (String query : queries)
addResource(query);
return this;
}
public CollectionQuery addMetaFilter(String key, String value) {
this.mfilter.put(key, value);
return this;
}
public CollectionQuery addMetaFilter(String queries, RELATION rel) {
this.mfilter.putAll(resRel(queries, rel));
return this;
}
public CollectionQuery addMetaExtend(String key, String value) {
this.mextension.put(key, value);
return this;
}
public CollectionQuery addMetaExtend(String queries, RELATION rel) {
this.mextension.putAll(resRel(queries, rel));
return this;
}
private List<Map> createFilter(RELATION rel) {
String relation = rel == RELATION.AND ? "and" : "or";
List<Map> mfil = new ArrayList();
boolean multypes = this.mfilter.keySet().size() > 1;
String def_key = null;
if (!multypes) {
Multiset<String> keys = this.mfilter.keys();
def_key = keys.toArray(new String[keys.size()])[0];
}
List value = this.createValue(this.mfilter);
if (mfilter.values().size() == 1)
Collections.addAll(mfil, types.createMetaFilter((Map) value.get(0)));
else {
Map group;
if (!multypes)
group = types.createGroup(relation, def_key, value);
else
group = types.createGroup(relation, null, value);
Collections.addAll(mfil, types.createMetaFilter(group));
}
return mfil;
}
private List<Map> createExtender(RELATION rel) {
String relation = rel == RELATION.AND ? "and" : "or";
List<Map> mex = new ArrayList();
boolean multypes = this.mextension.keys().size() > 1;
String def_key = null;
if (!multypes)
def_key = this.mextension.keys().toArray(new String[0])[0];
List value = this.createValue(this.mextension);
// todo: missing: - takes only one resource, but resources can be chained!
if (this.mextension.values().size() == 1)
Collections.addAll(mex, types.createMetaExtend((Map) value.get(0)));
else {
Map group;
if (!multypes)
group = types.createGroup(relation, def_key, value);
else
group = types.createGroup(relation, null, value);
Collections.addAll(mex, types.createMetaExtend(group));
}
return mex;
}
private List<Map> join(RELATION filter, RELATION extension) {
List<Map> cursor = new ArrayList<>(this.rq);
if (!this.mfilter.isEmpty())
cursor.addAll(this.createFilter(filter));
if (!this.mextension.isEmpty())
cursor.addAll(this.createExtender(extension));
return cursor;
}
private List createValue(Multimap<String, String> map) {
List value = new ArrayList<>();
String[] dates = new String[3];
for (String key : map.keySet()) {
if (key.equals("pubDate")) {
dates = processDates((List<String>) map.get(key));
continue;
}
if (map.get(key).size() == 1) {
Map term = types.createTerm(key, null,
map.get(key).toArray(new String[0])[0], null);
value.add(term);
} else {
boolean multypes = map.keySet().size() > 1;
List g = new ArrayList();
for (String v : map.get(key))
g.add(types.createTerm(null, v, null));
if (multypes) {
Map group = types.createGroup("and", key, g);
value.add(group);
} else
value.addAll(g);
}
}
int idx = 3;
if (dates[0] != null && dates[0].equals("r")) {
Map term1 = types.createTerm(null, dates[1], "korap:date");
Map term2 = types.createTerm(null, dates[2], "korap:date");
Map group = types.createGroup("between", "pubDate", Arrays.asList(term1, term2));
value.add(group);
} else if (dates[1] != null) {
Map term1 = types.createTerm(null, dates[1], "korap:date");
Map group = types.createGroup("since", "pubDate", Arrays.asList(term1));
value.add(group);
} else if (dates[2] != null) {
Map term1 = types.createTerm(null, dates[2], "korap:date");
Map group = types.createGroup("until", "pubDate", Arrays.asList(term1));
value.add(group);
}
for (int i = idx; i < dates.length; i++) {
if (dates[i] != null) {
Map term1 = types.createTerm(dates[i], "korap:date");
Map group = types.createGroup("exact", "pubDate", Arrays.asList(term1));
value.add(group);
}
}
return value;
}
private String[] processDates(List<String> dates) {
if (dates.isEmpty())
return new String[3];
String[] el = new String[dates.size() + 3];
int idx = 3;
for (String value : dates) {
if (value.contains("<")) {
String[] sp = value.split("<");
el[1] = sp[1];
} else if (value.contains(">")) {
String[] sp = value.split(">");
el[2] = sp[1];
} else {
el[idx] = value;
idx++;
}
}
if (el[1] != null && el[2] != null)
el[0] = "r";
return el;
}
public List<Map> raw(RELATION filter, RELATION extension) {
return join(filter, extension);
}
public String toCollections(RELATION filter, RELATION extension) {
Map meta = new LinkedHashMap();
meta.put("collections", join(filter, extension));
try {
return serialzer.writeValueAsString(meta);
} catch (JsonProcessingException e) {
e.printStackTrace();
return "";
}
}
public String toCollections() {
return toCollections(RELATION.AND, RELATION.AND);
}
/**
* returns all references to parents and meta query as string representation
*
* @return
*/
public JsonNode buildNode(RELATION filter, RELATION extension) {
return serialzer.valueToTree(join(filter, extension));
}
public String buildString(RELATION filter, RELATION extension) {
try {
return serialzer.writeValueAsString(join(filter, extension));
} catch (JsonProcessingException e) {
e.printStackTrace();
return "";
}
}
/**
* resolves all queries as equal (hierarchy) AND relations
*
* @param queries
* @return
*/
private Multimap<String, String> resRel(String queries, RELATION rel) {
Multimap<String, String> qmap = ArrayListMultimap.create();
String[] spl = queries.split(rel.toString());
for (String query : spl) {
String[] q = query.split(":");
String attr = q[0];
String val = q[1];
qmap.put(attr, val);
}
return qmap;
}
public void clear() {
this.rq.clear();
this.mfilter.clear();
this.mextension.clear();
}
}