org.graylog.plugins.views.search.Search Maven / Gradle / Ivy
/*
* Copyright (C) 2020 Graylog, Inc.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the Server Side Public License, version 1,
* as published by MongoDB, Inc.
*
* 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
* Server Side Public License for more details.
*
* You should have received a copy of the Server Side Public License
* along with this program. If not, see
* .
*/
package org.graylog.plugins.views.search;
import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
import com.fasterxml.jackson.databind.annotation.JsonPOJOBuilder;
import com.google.auto.value.AutoValue;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Maps;
import com.google.common.collect.Sets;
import org.graylog.plugins.views.search.rest.ExecutionState;
import org.graylog.plugins.views.search.rest.ExecutionStateGlobalOverride;
import org.graylog.plugins.views.search.views.PluginMetadataSummary;
import org.graylog2.contentpacks.ContentPackable;
import org.graylog2.contentpacks.EntityDescriptorIds;
import org.graylog2.contentpacks.model.entities.SearchEntity;
import org.graylog2.shared.rest.exceptions.MissingStreamPermissionException;
import org.joda.time.DateTime;
import org.joda.time.DateTimeZone;
import org.mongojack.Id;
import org.mongojack.ObjectId;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.Collections;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import static com.google.common.collect.ImmutableSet.of;
import static com.google.common.collect.ImmutableSet.toImmutableSet;
import static java.util.stream.Collectors.toSet;
@AutoValue
@JsonAutoDetect
@JsonDeserialize(builder = Search.Builder.class)
public abstract class Search implements ContentPackable {
public static final String FIELD_REQUIRES = "requires";
static final String FIELD_CREATED_AT = "created_at";
public static final String FIELD_OWNER = "owner";
// generated during build to help quickly find a parameter by name.
private ImmutableMap parameterIndex;
@Id
@ObjectId
@Nullable
@JsonProperty
public abstract String id();
@JsonProperty
public abstract ImmutableSet queries();
@JsonProperty
public abstract ImmutableSet parameters();
@JsonProperty(FIELD_REQUIRES)
public abstract Map requires();
@JsonProperty(FIELD_OWNER)
public abstract Optional owner();
public Search withOwner(@Nonnull String owner) {
return toBuilder().owner(owner).build();
}
@JsonProperty(FIELD_CREATED_AT)
public abstract DateTime createdAt();
@JsonIgnore
public Optional getParameter(String parameterName) {
return Optional.ofNullable(parameterIndex.get(parameterName));
}
public Search applyExecutionState(ObjectMapper objectMapper, ExecutionState executionState) {
final Builder builder = toBuilder();
if (!executionState.parameterBindings().isEmpty()) {
final ImmutableSet parameters = parameters().stream()
.map(param -> param.applyBindings(executionState.parameterBindings()))
.collect(toImmutableSet());
builder.parameters(parameters);
}
if (executionState.queries() != null || executionState.globalOverride() != null) {
final ImmutableSet queries = queries().stream()
.map(query -> applyStateToQuery(executionState, query))
.collect(toImmutableSet());
builder.queries(queries);
}
return builder.build();
}
private Query applyStateToQuery(ExecutionState executionState, Query query) {
if (executionState.globalOverride().hasValues()) {
return query.applyExecutionState(executionState.globalOverride());
} else {
final ExecutionStateGlobalOverride queryOverride = executionState.queries().get(query.id());
return query.applyExecutionState(queryOverride);
}
}
public Search addStreamsToQueriesWithoutStreams(Supplier> defaultStreamsSupplier) {
if (!hasQueriesWithoutStreams()) {
return this;
}
final Set withStreams = queries().stream().filter(Query::hasStreams).collect(toSet());
final Set withoutStreams = Sets.difference(queries(), withStreams);
final Set defaultStreams = defaultStreamsSupplier.get();
if (defaultStreams.isEmpty()) {
throw new MissingStreamPermissionException("User doesn't have access to any streams",
Collections.emptySet());
}
final Set withDefaultStreams = withoutStreams.stream()
.map(q -> q.addStreamsToFilter(defaultStreams))
.collect(toSet());
final ImmutableSet newQueries = Sets.union(withStreams, withDefaultStreams).immutableCopy();
return toBuilder().queries(newQueries).build();
}
private boolean hasQueriesWithoutStreams() {
return !queries().stream().allMatch(Query::hasStreams);
}
public abstract Builder toBuilder();
public static Builder builder() {
return Builder.create().parameters(of()).queries(ImmutableSet.builder().build());
}
public Set usedStreamIds() {
final Set queryStreamIds = queries().stream()
.map(Query::usedStreamIds)
.reduce(Collections.emptySet(), Sets::union);
final Set searchTypeStreamIds = queries().stream()
.flatMap(q -> q.searchTypes().stream())
.map(SearchType::effectiveStreams)
.reduce(Collections.emptySet(), Sets::union);
return Sets.union(queryStreamIds, searchTypeStreamIds);
}
public Set streamIdsForPermissionsCheck() {
final Set queryStreamIds = queries().stream()
.map(Query::usedStreamIds)
.reduce(Collections.emptySet(), Sets::union);
final Set searchTypeStreamIds = queries().stream()
.flatMap(q -> q.searchTypes().stream())
.map(SearchType::streams)
.reduce(Collections.emptySet(), Sets::union);
return Sets.union(queryStreamIds, searchTypeStreamIds);
}
public Query queryForSearchType(String searchTypeId) {
return queries().stream()
.filter(q -> q.hasSearchType(searchTypeId))
.findFirst()
.orElseThrow(() -> new IllegalArgumentException("Search " + id() + " doesn't have a query for search type " + searchTypeId));
}
@AutoValue.Builder
@JsonPOJOBuilder(withPrefix = "")
public abstract static class Builder {
@Id
@JsonProperty
public abstract Builder id(String id);
public abstract String id();
@JsonProperty
public abstract Builder queries(ImmutableSet queries);
@JsonProperty
public abstract Builder parameters(ImmutableSet parameters);
@JsonProperty(FIELD_REQUIRES)
public abstract Builder requires(Map requirements);
@JsonProperty(FIELD_OWNER)
public abstract Builder owner(@Nullable String owner);
@JsonProperty(FIELD_CREATED_AT)
public abstract Builder createdAt(DateTime createdAt);
abstract Search autoBuild();
@JsonCreator
public static Builder create() {
return new AutoValue_Search.Builder()
.requires(Collections.emptyMap())
.createdAt(DateTime.now(DateTimeZone.UTC))
.parameters(of());
}
public Search build() {
if (id() == null) {
id(org.bson.types.ObjectId.get().toString());
}
final Search search = autoBuild();
search.parameterIndex = Maps.uniqueIndex(search.parameters(), Parameter::name);
return search;
}
}
@Override
public SearchEntity toContentPackEntity(EntityDescriptorIds entityDescriptorIds) {
final SearchEntity.Builder searchEntityBuilder = SearchEntity.builder()
.queries(ImmutableSet.copyOf(this.queries().stream()
.map(query -> query.toContentPackEntity(entityDescriptorIds))
.collect(Collectors.toSet())))
.parameters(this.parameters())
.requires(this.requires())
.createdAt(this.createdAt());
if (this.owner().isPresent()) {
searchEntityBuilder.owner(this.owner().get());
}
return searchEntityBuilder.build();
}
}