blob: 5dd6bdb857715e6ca39bb12a0528f698111931ab [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.
*/
package org.apache.htrace.core;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonDeserializer;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.ObjectReader;
import com.fasterxml.jackson.databind.ObjectWriter;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
import java.io.IOException;
import java.io.StringWriter;
import java.io.UnsupportedEncodingException;
import java.math.BigInteger;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
/**
* A Span implementation that stores its information in milliseconds since the
* epoch.
*/
@JsonDeserialize(using = MilliSpan.MilliSpanDeserializer.class)
public class MilliSpan implements Span {
private static ObjectMapper OBJECT_MAPPER = new ObjectMapper();
private static ObjectReader JSON_READER = OBJECT_MAPPER.reader(MilliSpan.class);
private static ObjectWriter JSON_WRITER = OBJECT_MAPPER.writer();
private static final SpanId EMPTY_PARENT_ARRAY[] = new SpanId[0];
private static final String EMPTY_STRING = "";
private long begin;
private long end;
private final String description;
private SpanId parents[];
private final SpanId spanId;
private Map<String, String> traceInfo = null;
private String tracerId;
private List<TimelineAnnotation> timeline = null;
@Override
public Span child(String childDescription) {
return new MilliSpan.Builder().
begin(System.currentTimeMillis()).
end(0).
description(childDescription).
parents(new SpanId[] {spanId}).
spanId(spanId.newChildId()).
tracerId(tracerId).
build();
}
/**
* The public interface for constructing a MilliSpan.
*/
public static class Builder {
private long begin;
private long end;
private String description = EMPTY_STRING;
private SpanId parents[] = EMPTY_PARENT_ARRAY;
private SpanId spanId = SpanId.INVALID;
private Map<String, String> traceInfo = null;
private String tracerId = EMPTY_STRING;
private List<TimelineAnnotation> timeline = null;
public Builder() {
}
public Builder begin(long begin) {
this.begin = begin;
return this;
}
public Builder end(long end) {
this.end = end;
return this;
}
public Builder description(String description) {
this.description = description;
return this;
}
public Builder parents(SpanId parents[]) {
this.parents = parents;
return this;
}
public Builder parents(List<SpanId> parentList) {
SpanId[] parents = new SpanId[parentList.size()];
for (int i = 0; i < parentList.size(); i++) {
parents[i] = parentList.get(i);
}
this.parents = parents;
return this;
}
public Builder spanId(SpanId spanId) {
this.spanId = spanId;
return this;
}
public Builder traceInfo(Map<String, String> traceInfo) {
this.traceInfo = traceInfo.isEmpty() ? null : traceInfo;
return this;
}
public Builder tracerId(String tracerId) {
this.tracerId = tracerId;
return this;
}
public Builder timeline(List<TimelineAnnotation> timeline) {
this.timeline = timeline.isEmpty() ? null : timeline;
return this;
}
public MilliSpan build() {
return new MilliSpan(this);
}
}
public MilliSpan() {
this.begin = 0;
this.end = 0;
this.description = EMPTY_STRING;
this.parents = EMPTY_PARENT_ARRAY;
this.spanId = SpanId.INVALID;
this.traceInfo = null;
this.tracerId = EMPTY_STRING;
this.timeline = null;
}
private MilliSpan(Builder builder) {
this.begin = builder.begin;
this.end = builder.end;
this.description = builder.description;
this.parents = builder.parents;
this.spanId = builder.spanId;
this.traceInfo = builder.traceInfo;
this.tracerId = builder.tracerId;
this.timeline = builder.timeline;
}
@Override
public synchronized void stop() {
if (end == 0) {
if (begin == 0)
throw new IllegalStateException("Span for " + description
+ " has not been started");
end = System.currentTimeMillis();
}
}
protected long currentTimeMillis() {
return System.currentTimeMillis();
}
@Override
public synchronized boolean isRunning() {
return begin != 0 && end == 0;
}
@Override
public synchronized long getAccumulatedMillis() {
if (begin == 0)
return 0;
if (end > 0)
return end - begin;
return currentTimeMillis() - begin;
}
@Override
public String toString() {
return toJson();
}
@Override
public String getDescription() {
return description;
}
@Override
public SpanId getSpanId() {
return spanId;
}
@Override
public SpanId[] getParents() {
return parents;
}
@Override
public void setParents(SpanId[] parents) {
this.parents = parents;
}
@Override
public long getStartTimeMillis() {
return begin;
}
@Override
public long getStopTimeMillis() {
return end;
}
@Override
public void addKVAnnotation(String key, String value) {
if (traceInfo == null)
traceInfo = new HashMap<String, String>();
traceInfo.put(key, value);
}
@Override
public void addTimelineAnnotation(String msg) {
if (timeline == null) {
timeline = new ArrayList<TimelineAnnotation>();
}
timeline.add(new TimelineAnnotation(System.currentTimeMillis(), msg));
}
@Override
public Map<String, String> getKVAnnotations() {
if (traceInfo == null)
return Collections.emptyMap();
return Collections.unmodifiableMap(traceInfo);
}
@Override
public List<TimelineAnnotation> getTimelineAnnotations() {
if (timeline == null) {
return Collections.emptyList();
}
return Collections.unmodifiableList(timeline);
}
@Override
public String getTracerId() {
return tracerId;
}
@Override
public void setTracerId(String tracerId) {
this.tracerId = tracerId;
}
@Override
public String toJson() {
StringWriter writer = new StringWriter();
try {
JSON_WRITER.writeValue(writer, this);
} catch (IOException e) {
// An IOException should not be possible when writing to a string.
throw new RuntimeException(e);
}
return writer.toString();
}
public static class MilliSpanDeserializer
extends JsonDeserializer<MilliSpan> {
@Override
public MilliSpan deserialize(JsonParser jp, DeserializationContext ctxt)
throws IOException, JsonProcessingException {
JsonNode root = jp.getCodec().readTree(jp);
Builder builder = new Builder();
JsonNode bNode = root.get("b");
if (bNode != null) {
builder.begin(bNode.asLong());
}
JsonNode eNode = root.get("e");
if (eNode != null) {
builder.end(eNode.asLong());
}
JsonNode dNode = root.get("d");
if (dNode != null) {
builder.description(dNode.asText());
}
JsonNode sNode = root.get("a");
if (sNode != null) {
builder.spanId(SpanId.fromString(sNode.asText()));
}
JsonNode rNode = root.get("r");
if (rNode != null) {
builder.tracerId(rNode.asText());
}
JsonNode parentsNode = root.get("p");
LinkedList<SpanId> parents = new LinkedList<SpanId>();
if (parentsNode != null) {
for (Iterator<JsonNode> iter = parentsNode.elements();
iter.hasNext(); ) {
JsonNode parentIdNode = iter.next();
parents.add(SpanId.fromString(parentIdNode.asText()));
}
}
builder.parents(parents);
JsonNode traceInfoNode = root.get("n");
if (traceInfoNode != null) {
HashMap<String, String> traceInfo = new HashMap<String, String>();
for (Iterator<String> iter = traceInfoNode.fieldNames();
iter.hasNext(); ) {
String field = iter.next();
traceInfo.put(field, traceInfoNode.get(field).asText());
}
builder.traceInfo(traceInfo);
}
JsonNode timelineNode = root.get("t");
if (timelineNode != null) {
LinkedList<TimelineAnnotation> timeline =
new LinkedList<TimelineAnnotation>();
for (Iterator<JsonNode> iter = timelineNode.elements();
iter.hasNext(); ) {
JsonNode ann = iter.next();
timeline.add(new TimelineAnnotation(ann.get("t").asLong(),
ann.get("m").asText()));
}
builder.timeline(timeline);
}
return builder.build();
}
}
public static MilliSpan fromJson(String json) throws IOException {
return JSON_READER.readValue(json);
}
}