All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.ff4j.elastic.ElasticQueryBuilder Maven / Gradle / Ivy

There is a newer version: 2.1
Show newest version
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 ([email protected])
 */
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 Search queryReadAllFeatures(Integer totalCount) {
        return new Search.Builder(new SearchSourceBuilder().size(totalCount).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> 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 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 metadatas = new HashSet();
		if (null != result && result.isSucceeded()) {
			List> features = result.getHits(Map.class);
			for (Hit 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 queryReadAllProperties(Integer totalCount) {
		return new Search.Builder(new SearchSourceBuilder().size(totalCount).toString()).addIndex(connection.getIndexName())
				.addType(ElasticConstants.TYPE_FEATURE).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> 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 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> 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;
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy