package org.ff4j.elastic;
/*
* #%L
* ff4j-store-elastic
* %%
* Copyright (C) 2013 - 2016 FF4J
* %%
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
* #L%
*/
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.elasticsearch.index.query.BoolQueryBuilder;
import org.elasticsearch.index.query.QueryBuilder;
import org.elasticsearch.index.query.QueryBuilders;
import org.elasticsearch.index.query.RangeQueryBuilder;
import org.elasticsearch.search.builder.SearchSourceBuilder;
import org.ff4j.audit.Event;
import org.ff4j.audit.EventConstants;
import org.ff4j.audit.EventQueryDefinition;
import org.ff4j.core.Feature;
import org.ff4j.property.Property;
import io.searchbox.client.JestResult;
import io.searchbox.core.Delete;
import io.searchbox.core.Index;
import io.searchbox.core.Search;
import io.searchbox.core.SearchResult;
import io.searchbox.core.SearchResult.Hit;
import io.searchbox.core.Update;
import io.searchbox.indices.DeleteIndex;
import io.searchbox.indices.Flush;
/**
* Helper to create Jest queries.
*
* @author Cedrick LUNVEN (@clunven)
* @author Andre BLASZCZYK (andre.blaszczyk@gmail.com)
*/
public class ElasticQueryBuilder {
/** Connection. */
private final ElasticConnection connection;
/**
* Initialization of the builder with a dedicated connection.
*
* @param conn
* current elastic collection.
*/
public ElasticQueryBuilder(ElasticConnection conn) {
this.connection = conn;
}
public Flush queryFlushIndex() {
return new Flush.Builder().addIndex(connection.getIndexName()).build();
}
/**
* Syntaxic sugar to have query on feature.
*
* @param uid
* target feature uid
* @return query for JEST
*/
public Search queryGetFeatureById(String uid) {
SearchSourceBuilder source = new SearchSourceBuilder();
source.query(QueryBuilders.matchQuery("uid", uid));
return new Search.Builder(source.toString()).addIndex(connection.getIndexName())
.addType(ElasticConstants.TYPE_FEATURE).build();
}
public Search getGroupByGroupName(String groupName) {
SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder();
searchSourceBuilder.query(QueryBuilders.matchQuery("group", groupName));
return new Search.Builder(searchSourceBuilder.toString()) //
.addIndex(connection.getIndexName()) //
.addType(ElasticConstants.TYPE_FEATURE).build();
}
public Index queryCreateFeature(Feature fp) {
return new Index.Builder(fp).index(connection.getIndexName()).type(ElasticConstants.TYPE_FEATURE).refresh(true)
.build();
}
public Search queryReadAllFeatures() {
return new Search.Builder(new SearchSourceBuilder().toString()).addIndex(connection.getIndexName())
.addType(ElasticConstants.TYPE_FEATURE).build();
}
public Delete queryDeleteFeature(String uid) {
return new Delete.Builder(uid).index(connection.getIndexName()).type(ElasticConstants.TYPE_FEATURE)
.id(getFeatureTechId(uid)).refresh(true).build();
}
public Index queryUpdateFeature(Feature fp) {
return new Index.Builder(fp).index(connection.getIndexName()).type(ElasticConstants.TYPE_FEATURE)
.id(getFeatureTechId(fp.getUid())).refresh(true).build();
}
public String getFeatureTechId(String uid) {
SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder();
searchSourceBuilder.query(QueryBuilders.matchQuery("uid", uid));
Search search = new Search.Builder(searchSourceBuilder.toString()) //
.addIndex(connection.getIndexName()) //
.addType(ElasticConstants.TYPE_FEATURE) //
.build();
// feature existence must have been checked before (technical function)
@SuppressWarnings("rawtypes")
List<Hit<Map, Void>> items = connection.search(search).getHits(Map.class);
if (null != items && !items.isEmpty()) {
return connection.search(search).getHits(Map.class).get(0).source.get(JestResult.ES_METADATA_ID).toString();
}
return null;
}
@SuppressWarnings({ "rawtypes" })
public Set<String> getFeatureTechIdByGroup(String groupName) {
SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder();
searchSourceBuilder.query(QueryBuilders.matchQuery("group", groupName));
Search search = new Search.Builder(searchSourceBuilder.toString()) //
.addIndex(connection.getIndexName()) //
.addType(ElasticConstants.TYPE_FEATURE) //
.build();
SearchResult result = connection.search(search, true);
Set<String> metadatas = new HashSet<String>();
if (null != result && result.isSucceeded()) {
List<Hit<Map, Void>> features = result.getHits(Map.class);
for (Hit<Map, Void> hit : features) {
metadatas.add(hit.source.get(JestResult.ES_METADATA_ID).toString());
}
}
return metadatas;
}
/**
* Update status of feature.
*
* @param uid
* feature id
* @param enable
* enabler
*/
public Update updateStatus(String uid, boolean enable) {
String partialDoc = "{ \"doc\" : { \"enable\" : " + enable + " } }";
return new Update.Builder(partialDoc) //
.index(connection.getIndexName()) //
.type(ElasticConstants.TYPE_FEATURE) //
.id(getFeatureTechId(uid)) //
.refresh(true) //
.build();
}
public Update queryUpdateStatusWithTechId(String _id, boolean enable) {
String partialDoc = "{ \"doc\" : { \"enable\" : " + enable + " } }";
return new Update.Builder(partialDoc) //
.index(connection.getIndexName()) //
.type(ElasticConstants.TYPE_FEATURE) //
.id(_id) //
.build();
}
public Update queryEnableWithTechId(String _id) {
return queryUpdateStatusWithTechId(_id, true);
}
public Update queryDisableWithTechId(String _id) {
return queryUpdateStatusWithTechId(_id, false);
}
public Update queryEnable(String uid) {
return updateStatus(uid, true);
}
public Update queryDisable(String uid) {
return updateStatus(uid, false);
}
public DeleteIndex queryClear() {
return new DeleteIndex.Builder(connection.getIndexName()).build();
}
// "Property" methods
public Search queryReadAllProperties() {
return new Search.Builder(new SearchSourceBuilder().toString()).addIndex(connection.getIndexName())
.addType(ElasticConstants.TYPE_PROPERTY).build();
}
public Search queryPropertyByName(String name) {
SearchSourceBuilder source = new SearchSourceBuilder();
source.query(QueryBuilders.matchQuery("name", name));
return new Search.Builder(source.toString()).addIndex(connection.getIndexName())
.addType(ElasticConstants.TYPE_PROPERTY).build();
}
public Index queryCreateProperty(Property<?> property) {
return new Index.Builder(property).index(connection.getIndexName()).type(ElasticConstants.TYPE_PROPERTY)
.refresh(true).build();
}
public Delete queryDeletePropertyByName(String name) {
return new Delete.Builder(name).index(connection.getIndexName()).type(ElasticConstants.TYPE_PROPERTY)
.id(getPropertyTechIdByName(name)).refresh(true).build();
}
@SuppressWarnings("rawtypes")
public String getPropertyTechIdByName(String name) {
SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder();
searchSourceBuilder.query(QueryBuilders.matchQuery("name", name));
Search search = new Search.Builder(searchSourceBuilder.toString()) //
.addIndex(connection.getIndexName()) //
.addType(ElasticConstants.TYPE_PROPERTY) //
.build();
List<Hit<Map, Void>> items = connection.search(search).getHits(Map.class);
if (null != items && !items.isEmpty()) {
return connection.search(search).getHits(Map.class).get(0).source.get(JestResult.ES_METADATA_ID).toString();
}
return null;
}
public Search queryReadGroup(String groupName) {
SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder();
searchSourceBuilder.query(QueryBuilders.matchQuery("group", groupName));
return new Search.Builder(searchSourceBuilder.toString()) //
.addIndex(connection.getIndexName()) //
.addType("feature") //
.build();
}
public Update queryAddFeatureToGroup(String uid, String groupName) {
String partialDoc = "{ \"doc\" : { \"group\" : \"" + groupName + "\" } }";
return new Update.Builder(partialDoc) //
.index(connection.getIndexName()) //
.type(ElasticConstants.TYPE_FEATURE) //
.id(getFeatureTechId(uid)) //
.build();
}
public Update queryRemoveFeatureFromGroup(String uid, String groupName) {
String partialDoc = "{ \"doc\" : { \"group\" : \"\" } }";
return new Update.Builder(partialDoc) //
.index(connection.getIndexName()) //
.type(ElasticConstants.TYPE_FEATURE) //
.id(getFeatureTechId(uid)) //
.build();
}
// "Event" methods
public Index queryCreateEvent(Event event) {
return new Index.Builder(event).index(connection.getIndexName()).type(ElasticConstants.TYPE_EVENT).refresh(true)
.build();
}
public Search queryGetEventById(String uuid) {
SearchSourceBuilder source = new SearchSourceBuilder();
source.query(QueryBuilders.matchQuery("uuid", uuid));
return new Search.Builder(source.toString()).addIndex(connection.getIndexName())
.addType(ElasticConstants.TYPE_EVENT).build();
}
public Search queryGetEventQueryDefinition(EventQueryDefinition query, String action) {
BoolQueryBuilder booleanQuery = new BoolQueryBuilder();
// Optional constant for action filter
if (action != null) {
query.getActionFilters().add(action);
}
QueryBuilder typeQuery = QueryBuilders.termQuery("type", EventConstants.TARGET_FEATURE);
// Timestamp filter
RangeQueryBuilder timestampFilter = QueryBuilders.rangeQuery("timestamp") //
.gt(query.getFrom().longValue()) //
.lt(query.getTo().longValue()) //
.includeLower(false) //
.includeUpper(false);
booleanQuery.must(typeQuery);
booleanQuery.must(timestampFilter);
// Optional filters
addOptionalFilters(booleanQuery, query.getActionFilters(), "action");
addOptionalFilters(booleanQuery, query.getHostFilters(), "hostName");
addOptionalFilters(booleanQuery, query.getNamesFilter(), "name");
addOptionalFilters(booleanQuery, query.getSourceFilters(), "source");
// Warning : default size is set to 10 results, that's why it's
// overridden
SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder().size(100);
Search searchQuery = new Search.Builder(searchSourceBuilder.query(booleanQuery.toString()).toString()) //
.addIndex(connection.getIndexName()) //
.addType(ElasticConstants.TYPE_EVENT) //
.build();
return searchQuery;
}
public Search queryGetEventQueryDefinition(EventQueryDefinition query) {
return queryGetEventQueryDefinition(query, null);
}
public void addOptionalFilters(BoolQueryBuilder booleanQuery, Set<String> filters, String field) {
if (!filters.isEmpty()) {
BoolQueryBuilder subQuery = new BoolQueryBuilder();
for (String filter : filters) {
subQuery.must(QueryBuilders.matchQuery(field, filter));
}
booleanQuery.must(subQuery);
}
}
public Search queryReadAllEvents() {
return new Search.Builder(new SearchSourceBuilder().toString()).addIndex(connection.getIndexName())
.addType(ElasticConstants.TYPE_EVENT).build();
}
public Delete queryDeleteEvent(String uid) {
return new Delete.Builder(uid).index(connection.getIndexName()).type(ElasticConstants.TYPE_EVENT)
.id(getEventTechId(uid)).refresh(true).build();
}
@SuppressWarnings("rawtypes")
public String getEventTechId(String uuid) {
SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder();
searchSourceBuilder.query(QueryBuilders.matchQuery("uuid", uuid));
Search search = new Search.Builder(searchSourceBuilder.toString()) //
.addIndex(connection.getIndexName()) //
.addType(ElasticConstants.TYPE_EVENT) //
.build();
// event existence must have been checked before (technical function)
List<Hit<Map, Void>> items = connection.search(search).getHits(Map.class);
if (null != items && !items.isEmpty()) {
return connection.search(search).getHits(Map.class).get(0).source.get(JestResult.ES_METADATA_ID).toString();
}
return null;
}
}