zipkin2.elasticsearch.ElasticsearchSpanStore Maven / Gradle / Ivy
The newest version!
/*
* Copyright The OpenZipkin Authors
* SPDX-License-Identifier: Apache-2.0
*/
package zipkin2.elasticsearch;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import zipkin2.Call;
import zipkin2.DependencyLink;
import zipkin2.Span;
import zipkin2.elasticsearch.internal.IndexNameFormatter;
import zipkin2.elasticsearch.internal.client.Aggregation;
import zipkin2.elasticsearch.internal.client.HttpCall;
import zipkin2.elasticsearch.internal.client.SearchCallFactory;
import zipkin2.elasticsearch.internal.client.SearchRequest;
import zipkin2.storage.GroupByTraceId;
import zipkin2.storage.QueryRequest;
import zipkin2.storage.ServiceAndSpanNames;
import zipkin2.storage.SpanStore;
import zipkin2.storage.StrictTraceId;
import zipkin2.storage.Traces;
import static java.util.Arrays.asList;
import static zipkin2.elasticsearch.VersionSpecificTemplates.TYPE_DEPENDENCY;
import static zipkin2.elasticsearch.VersionSpecificTemplates.TYPE_SPAN;
final class ElasticsearchSpanStore implements SpanStore, Traces, ServiceAndSpanNames {
/** To not produce unnecessarily long queries, we don't look back further than first ES support */
static final long EARLIEST_MS = 1456790400000L; // March 2016
final SearchCallFactory search;
final Call.Mapper, List>> groupByTraceId;
final String[] allSpanIndices;
final IndexNameFormatter indexNameFormatter;
final boolean strictTraceId, searchEnabled;
final int namesLookback;
ElasticsearchSpanStore(ElasticsearchStorage es) {
this.search = new SearchCallFactory(es.http());
this.groupByTraceId = GroupByTraceId.create(es.strictTraceId());
this.allSpanIndices = new String[] {es.indexNameFormatter().formatType(TYPE_SPAN)};
this.indexNameFormatter = es.indexNameFormatter();
this.strictTraceId = es.strictTraceId();
this.searchEnabled = es.searchEnabled();
this.namesLookback = es.namesLookback();
}
@Override
public Call>> getTraces(QueryRequest request) {
if (!searchEnabled) return Call.emptyList();
long endMillis = request.endTs();
long beginMillis = Math.max(endMillis - request.lookback(), EARLIEST_MS);
SearchRequest.Filters filters = new SearchRequest.Filters();
filters.addRange("timestamp_millis", beginMillis, endMillis);
if (request.serviceName() != null) {
filters.addTerm("localEndpoint.serviceName", request.serviceName());
}
if (request.remoteServiceName() != null) {
filters.addTerm("remoteEndpoint.serviceName", request.remoteServiceName());
}
if (request.spanName() != null) {
filters.addTerm("name", request.spanName());
}
for (Map.Entry kv : request.annotationQuery().entrySet()) {
if (kv.getValue().isEmpty()) {
filters.addTerm("_q", kv.getKey());
} else {
filters.addTerm("_q", kv.getKey() + "=" + kv.getValue());
}
}
if (request.minDuration() != null) {
filters.addRange("duration", request.minDuration(), request.maxDuration());
}
// We need to filter to traces that contain at least one span that matches the request,
// but the zipkin API is supposed to order traces by first span, regardless of if it was
// filtered or not. This is not possible without either multiple, heavyweight queries
// or complex multiple indexing, defeating much of the elegance of using elasticsearch for this.
// So we fudge and order on the first span among the filtered spans - in practice, there should
// be no significant difference in user experience since span start times are usually very
// close to each other in human time.
Aggregation traceIdTimestamp =
Aggregation.terms("traceId", request.limit())
.addSubAggregation(Aggregation.min("timestamp_millis"))
.orderBy("timestamp_millis", "desc");
List indices = indexNameFormatter.formatTypeAndRange(TYPE_SPAN, beginMillis, endMillis);
if (indices.isEmpty()) return Call.emptyList();
SearchRequest esRequest =
SearchRequest.create(indices).filters(filters).addAggregation(traceIdTimestamp);
HttpCall> traceIdsCall = search.newCall(esRequest, BodyConverters.KEYS);
Call>> result =
traceIdsCall.flatMap(new GetSpansByTraceId(search, indices)).map(groupByTraceId);
// Elasticsearch lookup by trace ID is by the full 128-bit length, but there's still a chance of
// clash on lower-64 bit. When strict trace ID is enabled, we only filter client-side on clash.
return strictTraceId ? result.map(StrictTraceId.filterTraces(request)) : result;
}
@Override
public Call> getTrace(String traceId) {
// make sure we have a 16 or 32 character trace ID
traceId = Span.normalizeTraceId(traceId);
// Unless we are strict, truncate the trace ID to 64bit (encoded as 16 characters)
if (!strictTraceId && traceId.length() == 32) traceId = traceId.substring(16);
SearchRequest request = SearchRequest.create(asList(allSpanIndices)).term("traceId", traceId);
return search.newCall(request, BodyConverters.SPANS);
}
@Override public Call>> getTraces(Iterable traceIds) {
Set normalizedTraceIds = new LinkedHashSet<>();
for (String traceId : traceIds) {
// make sure we have a 16 or 32 character trace ID
traceId = Span.normalizeTraceId(traceId);
// Unless we are strict, truncate the trace ID to 64bit (encoded as 16 characters)
if (!strictTraceId && traceId.length() == 32) traceId = traceId.substring(16);
normalizedTraceIds.add(traceId);
}
if (normalizedTraceIds.isEmpty()) return Call.emptyList();
SearchRequest request =
SearchRequest.create(asList(allSpanIndices)).terms("traceId", normalizedTraceIds);
return search.newCall(request, BodyConverters.SPANS).map(groupByTraceId);
}
@Override public Call> getServiceNames() {
if (!searchEnabled) return Call.emptyList();
long endMillis = System.currentTimeMillis();
long beginMillis = endMillis - namesLookback;
List indices = indexNameFormatter.formatTypeAndRange(TYPE_SPAN, beginMillis, endMillis);
if (indices.isEmpty()) return Call.emptyList();
SearchRequest request = SearchRequest.create(indices)
.filters(new SearchRequest.Filters().addRange("timestamp_millis", beginMillis, endMillis))
.addAggregation(Aggregation.terms("localEndpoint.serviceName", Integer.MAX_VALUE));
return search.newCall(request, BodyConverters.KEYS);
}
@Override public Call> getRemoteServiceNames(String serviceName) {
return aggregatedFieldByServiceName(serviceName, "remoteEndpoint.serviceName");
}
@Override public Call> getSpanNames(String serviceName) {
return aggregatedFieldByServiceName(serviceName, "name");
}
Call> aggregatedFieldByServiceName(String serviceName, String term) {
if (serviceName.isEmpty() || !searchEnabled) return Call.emptyList();
long endMillis = System.currentTimeMillis();
long beginMillis = endMillis - namesLookback;
List indices = indexNameFormatter.formatTypeAndRange(TYPE_SPAN, beginMillis, endMillis);
if (indices.isEmpty()) return Call.emptyList();
// A span name is only valid on a local endpoint, as a span name is defined locally
SearchRequest.Filters filters = new SearchRequest.Filters()
.addRange("timestamp_millis", beginMillis, endMillis)
.addTerm("localEndpoint.serviceName", serviceName.toLowerCase(Locale.ROOT));
SearchRequest request = SearchRequest.create(indices).filters(filters)
.addAggregation(Aggregation.terms(term, Integer.MAX_VALUE));
return search.newCall(request, BodyConverters.KEYS);
}
@Override
public Call> getDependencies(long endTs, long lookback) {
if (endTs <= 0) throw new IllegalArgumentException("endTs <= 0");
if (lookback <= 0) throw new IllegalArgumentException("lookback <= 0");
long beginMillis = Math.max(endTs - lookback, EARLIEST_MS);
// We just return all dependencies in the days that fall within endTs and lookback as
// dependency links themselves don't have timestamps.
List indices =
indexNameFormatter.formatTypeAndRange(TYPE_DEPENDENCY, beginMillis, endTs);
if (indices.isEmpty()) return Call.emptyList();
return search.newCall(SearchRequest.create(indices), BodyConverters.DEPENDENCY_LINKS);
}
static final class GetSpansByTraceId implements Call.FlatMapper, List> {
final SearchCallFactory search;
final List indices;
GetSpansByTraceId(SearchCallFactory search, List indices) {
this.search = search;
this.indices = indices;
}
@Override
public Call> map(List input) {
if (input.isEmpty()) return Call.emptyList();
SearchRequest getTraces = SearchRequest.create(indices).terms("traceId", input);
return search.newCall(getTraces, BodyConverters.SPANS);
}
@Override
public String toString() {
return "GetSpansByTraceId{indices=" + indices + "}";
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy