/** * StringNavigator.java * (C) 2016 by reger24; https://github.com/reger24 * * This is a part of YaCy, a peer-to-peer based web search engine * * LICENSE * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. * If not, see . */ package net.yacy.search.navigator; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.Comparator; import java.util.Iterator; import java.util.List; import java.util.Map; import net.yacy.cora.sorting.ConcurrentScoreMap; import net.yacy.cora.sorting.ReversibleScoreMap; import net.yacy.kelondro.data.meta.URIMetadataNode; import net.yacy.search.query.QueryModifier; import net.yacy.search.schema.CollectionSchema; /** * Search navigator for simple string entries based on ScoreMap to count and * order the result list by counted occurence */ public class StringNavigator extends ConcurrentScoreMap implements Navigator { public String title; protected final CollectionSchema field; /** * The sort properties to apply when iterating over keys with the * {@link #navigatorKeys()} function */ private final NavigatorSort sort; /** * Constructor applying a descending sort by counts as defaut. * @param title the navigator title * @param field the indexed field to count */ public StringNavigator(final String title, final CollectionSchema field) { this(title, field, NavigatorSort.COUNT_DESC); } /** * @param title the navigator title * @param field the indexed field to count * @param sort the sort properties to apply when iterating over keys with the * {@link #navigatorKeys()} function */ public StringNavigator(final String title, final CollectionSchema field, final NavigatorSort sort) { this.title = title; this.field = field; if(sort == null) { this.sort = NavigatorSort.COUNT_DESC; } else { this.sort = sort; } } @Override public String getDisplayName() { return title; } @Override public String getElementDisplayName(final String e) { return e; } @Override public String getQueryModifier(final String key) { String mod; if (field != null) { switch (field) { case author_sxt: if (key.indexOf(' ') > 0) { // may contain spaces mod = "author:(" + key + ")"; } else { mod = "author:" + key; } break; case keywords: mod = "keyword:" + key; break; case url_protocol_s: mod = "/" + key; break; case url_file_ext_s: mod = "filetype:" + key; break; case collection_sxt: mod = "collection:" + key; break; case host_s: mod = "site:" + key; break; case language_s: mod = "/language/" + key; break; default: mod = key; } } else { mod = ""; } return mod; } @Override public void incDocList(List docs) { if (field != null) { for (URIMetadataNode doc : docs) { incDoc(doc); } } } @Override public void incFacet(Map> facets) { if (field != null && facets != null && !facets.isEmpty()) { ReversibleScoreMap fcts = facets.get(field.getSolrFieldName()); if (fcts != null) { this.inc(fcts); } } } /** * Increase the score for the key value contained in the defined field in * the doc. * @param doc URIMetadataNode with field for the key content */ @Override public void incDoc(URIMetadataNode doc) { if (field != null) { Object val = doc.getFieldValue(field.getSolrFieldName()); if (val != null) { if (val instanceof Collection) { Collection ll = (Collection) val; for (Object obj : ll) { if(obj instanceof String) { final String s = (String)obj; if (!s.isEmpty()) { this.inc(s); } } } } else { this.inc((String) val); } } } } @Override public boolean modifieractive(final QueryModifier modifier, final String name) { return modifier.toString().contains(getQueryModifier(name)); } @Override public String getIndexFieldName() { if (this.field != null) { return this.field.getSolrFieldName(); } return ""; } @Override public Iterator navigatorKeys() { if(this.sort.getSortType() == NavigatorSortType.LABEL) { final ArrayList keys = new ArrayList<>(this.map.keySet()); Comparator keyComparator = Comparator.comparing(this::getElementDisplayName); if(this.sort.getSortDir() == NavigatorSortDirection.DESC) { keyComparator = keyComparator.reversed(); } Collections.sort(keys, keyComparator); return keys.iterator(); } return keys(this.sort.getSortDir() == NavigatorSortDirection.ASC); } @Override public NavigatorSort getSort() { return this.sort; } }