Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Shared JSON spec for span types/subtypes #1812

Merged
merged 31 commits into from
Jun 2, 2021
Merged
Show file tree
Hide file tree
Changes from 19 commits
Commits
Show all changes
31 commits
Select commit Hold shift + click to select a range
0cad766
first draft for JSON spec
SylvainJuge May 12, 2021
0a57169
fix servlet test
SylvainJuge May 12, 2021
78e6cac
make opentracing tests pass
SylvainJuge May 12, 2021
3f48a47
fix tests for OpenTracing
SylvainJuge May 12, 2021
692a412
fix external plugin tests
SylvainJuge May 20, 2021
f273c19
Merge branch 'master' into span-type-subtype-spec
SylvainJuge May 20, 2021
06228b0
Merge branch 'master' into span-type-subtype-spec
SylvainJuge May 20, 2021
71b19cb
update grpc version
SylvainJuge May 26, 2021
33e6ed3
allow to opt-out check + modify tests
SylvainJuge May 26, 2021
de77665
Merge branch 'span-type-subtype-spec' of github.com:SylvainJuge/apm-a…
SylvainJuge May 26, 2021
c459151
fix span tests
SylvainJuge May 26, 2021
3b84f9c
normalize span type/subtype/action + fix tests
SylvainJuge May 26, 2021
749dd20
add aws/azure services to shared spec
SylvainJuge May 26, 2021
9facbab
fix storage sub-types & remove hibernate
SylvainJuge May 26, 2021
0ef9b33
make rabbitmq tests fit spec
SylvainJuge May 26, 2021
94ecb24
disable span type check for hibernate-search 6.x
SylvainJuge May 26, 2021
ae62520
make opentracing bridge fit spec
SylvainJuge May 27, 2021
bc1a863
make opentracing fit the spec
SylvainJuge May 27, 2021
6d1b48f
fix integration test for process spans
SylvainJuge May 27, 2021
74e5bce
make subtypes less optional
SylvainJuge Jun 1, 2021
1daef8b
enforce null subtype when no subtype attribute in spec
SylvainJuge Jun 1, 2021
9802357
add description to json spec
SylvainJuge Jun 1, 2021
12f69cb
Merge branch 'master' into span-type-subtype-spec
SylvainJuge Jun 1, 2021
e3b8ba1
split subtype matching & null subtype
SylvainJuge Jun 1, 2021
d412933
Merge branch 'span-type-subtype-spec' of github.com:SylvainJuge/apm-a…
SylvainJuge Jun 1, 2021
f766835
another attempt to properly check
SylvainJuge Jun 1, 2021
5126d37
fit OpenTracing bridge tests
SylvainJuge Jun 2, 2021
c318803
comment -> __description
SylvainJuge Jun 2, 2021
4d3668c
code cleanup
SylvainJuge Jun 2, 2021
899215d
Update apm-agent-core/src/test/resources/json-specs/span_types.json
SylvainJuge Jun 2, 2021
7c27ff3
Merge branch 'master' into span-type-subtype-spec
SylvainJuge Jun 2, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -157,26 +157,31 @@ public SpanContext getContext() {
* Keywords of specific relevance in the span's domain (eg: 'db', 'template', 'ext', etc)
*/
public Span withType(@Nullable String type) {
this.type = type;
this.type = normalizeEmpty(type);
return this;
}

/**
* Sets the span's subtype, related to the (eg: 'mysql', 'postgresql', 'jsf' etc)
*/
public Span withSubtype(@Nullable String subtype) {
this.subtype = subtype;
this.subtype = normalizeEmpty(subtype);
return this;
}

/**
* Action related to this span (eg: 'query', 'render' etc)
*/
public Span withAction(@Nullable String action) {
this.action = action;
this.action = normalizeEmpty(action);
return this;
}

@Nullable
private static String normalizeEmpty(@Nullable String value) {
return value == null || value.isEmpty() ? null : value;
}

/**
* Sets span.type, span.subtype and span.action. If no subtype and action are provided, assumes the legacy usage of hierarchical
* typing system and attempts to split the type by dots to discover subtype and action.
Expand All @@ -199,9 +204,9 @@ public void setType(@Nullable String type, @Nullable String subtype, @Nullable S
}
}
}
this.type = type;
this.subtype = subtype;
this.action = action;
withType(type);
withSubtype(subtype);
withAction(action);
}

@Nullable
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -705,14 +705,14 @@ private void serializeSpanType(Span span) {
replace(replaceBuilder, ".", "_", 0);
String subtype = span.getSubtype();
String action = span.getAction();
if ((subtype != null && !subtype.isEmpty()) || (action != null && !action.isEmpty())) {
if (subtype != null || action != null) {
replaceBuilder.append('.');
int replaceStartIndex = replaceBuilder.length() + 1;
if (subtype != null && !subtype.isEmpty()) {
if (subtype != null) {
replaceBuilder.append(subtype);
replace(replaceBuilder, ".", "_", replaceStartIndex);
}
if (action != null && !action.isEmpty()) {
if (action != null) {
replaceBuilder.append('.');
replaceStartIndex = replaceBuilder.length() + 1;
replaceBuilder.append(action);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,9 @@ public static ConfigurationRegistry getConfig() {
@After
@AfterEach
public final void cleanUp() {
// re-enable all reporter checks
reporter.resetChecks();

SpyConfiguration.reset(config);
try {
if (validateRecycling) {
Expand Down
112 changes: 92 additions & 20 deletions apm-agent-core/src/test/java/co/elastic/apm/agent/MockReporter.java
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@
import com.networknt.schema.ValidationMessage;
import org.awaitility.core.ThrowingRunnable;
import org.stagemonitor.configuration.ConfigurationRegistry;
import specs.TestJsonSpec;

import java.io.IOException;
import java.util.ArrayList;
Expand Down Expand Up @@ -78,17 +79,24 @@ public class MockReporter implements Reporter {
// A map of exit span type to actions that that do not support address and port discovery
private static final Map<String, Collection<String>> SPAN_ACTIONS_WITHOUT_ADDRESS;
// And for any case the disablement of the check cannot rely on subtype (eg Redis, where Jedis supports and Lettuce does not)
private boolean disableDestinationAddressCheck;
private boolean checkDestinationAddress = true;
// Allows optional opt-out for unknown outcome
private boolean checkUnknownOutcomes = true;
// Allows optional opt-out from strick span type/sub-type checking
private boolean checkStrictSpanType = true;


private final List<Transaction> transactions = Collections.synchronizedList(new ArrayList<>());
private final List<Span> spans = Collections.synchronizedList(new ArrayList<>());
private final List<ErrorCapture> errors = Collections.synchronizedList(new ArrayList<>());
private final List<byte[]> bytes = new CopyOnWriteArrayList<>();
private final ObjectMapper objectMapper;
private final boolean verifyJsonSchema;
private boolean checkUnknownOutcomes = true;

private boolean closed;

private static final JsonNode SPAN_TYPES_SPEC = TestJsonSpec.getJson("span_types.json");

static {
transactionSchema = getSchema("/schema/transactions/transaction.json");
spanSchema = getSchema("/schema/transactions/span.json");
Expand Down Expand Up @@ -119,22 +127,38 @@ private static JsonSchema getSchema(String resource) {
}

/**
* @param enable {@literal true} to enable unknown outcome check, {@literal false} to allow for unknown outcome
* Sets all optional checks to their default value (enabled), should be used as a shortcut to reset mock reporter state
* after/before using it for a single test execution
*/
public void resetChecks() {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

❤️

checkDestinationAddress = true;
checkUnknownOutcomes = true;
checkStrictSpanType = true;
}

/**
* Disables unknown outcome check
*/
public void checkUnknownOutcome(boolean enable) {
checkUnknownOutcomes = enable;
public void disableCheckUnknownOutcome() {
checkUnknownOutcomes = false;
}

public void disableDestinationAddressCheck() {
disableDestinationAddressCheck = true;
/**
* Disables destination address check
*/
public void disableCheckDestinationAddress() {
checkDestinationAddress = false;
}

public void enableDestinationAddressCheck() {
disableDestinationAddressCheck = false;
public boolean checkDestinationAddress(){
return checkDestinationAddress;
}

public boolean isDisableDestinationAddressCheck() {
return disableDestinationAddressCheck;
/**
* Disables strict span type and sub-type check (against shared spec)
*/
public void disableCheckStrictSpanType() {
checkStrictSpanType = false;
}

@Override
Expand Down Expand Up @@ -164,27 +188,59 @@ public synchronized void report(Span span) {
if (closed) {
return;
}
verifySpanSchema(asJson(dslJsonSerializer.toJsonString(span)));
verifyDestinationFields(span);

String type = span.getType();
assertThat(type).isNotNull();
try {
verifySpanSchema(asJson(dslJsonSerializer.toJsonString(span)));
verifySpanType(span);
verifyDestinationFields(span);

if (checkUnknownOutcomes) {
assertThat(span.getOutcome())
.describedAs("span outcome should be either success or failure for type = %s", type)
.isNotEqualTo(Outcome.UNKNOWN);
String type = span.getType();
assertThat(type).isNotNull();

if (checkUnknownOutcomes) {
assertThat(span.getOutcome())
.describedAs("span outcome should be either success or failure for type = %s", type)
SylvainJuge marked this conversation as resolved.
Show resolved Hide resolved
.isNotEqualTo(Outcome.UNKNOWN);
}
} catch (Exception e) {
// in case a validation error occurs, ensure that it's properly logged for easier debugging
e.printStackTrace(System.err);
throw e;
}

spans.add(span);
}


private void verifySpanType(Span span) {
String type = span.getType();
assertThat(type)
.describedAs("span type is mandatory")
.isNotNull();

if (checkStrictSpanType) {
JsonNode typeJson = getMandatoryJson(SPAN_TYPES_SPEC, type, String.format("span type '%s'", type));
SylvainJuge marked this conversation as resolved.
Show resolved Hide resolved

String typeComment = getOptionalComment(typeJson);

String subType = span.getSubtype();
if (subType != null) {
SylvainJuge marked this conversation as resolved.
Show resolved Hide resolved
JsonNode subTypesJson = typeJson.get("subtypes");
if (null != subTypesJson) {
JsonNode subTypeJson = getMandatoryJson(subTypesJson, subType, String.format("span subtype '%s' for type '%s' (%s)", subType, type, typeComment));
SylvainJuge marked this conversation as resolved.
Show resolved Hide resolved
assertThat(subTypeJson).isNotNull();
SylvainJuge marked this conversation as resolved.
Show resolved Hide resolved
}
}
}

}

private void verifyDestinationFields(Span span) {
if (!span.isExit()) {
return;
}
Destination destination = span.getContext().getDestination();
if (!disableDestinationAddressCheck && !SPAN_TYPES_WITHOUT_ADDRESS.contains(span.getSubtype())) {
if (checkDestinationAddress && !SPAN_TYPES_WITHOUT_ADDRESS.contains(span.getSubtype())) {
// see if this span's action is not supported for its subtype
Collection<String> unsupportedActions = SPAN_ACTIONS_WITHOUT_ADDRESS.getOrDefault(span.getSubtype(), Collections.emptySet());
if (!unsupportedActions.contains(span.getAction())) {
Expand Down Expand Up @@ -486,4 +542,20 @@ public void awaitUntilAsserted(long timeoutMs, ThrowingRunnable runnable) {
private static boolean hasEmptyTraceContext(AbstractSpan<?> item) {
return item.getTraceContext().getId().isEmpty();
}

private static JsonNode getMandatoryJson(JsonNode json, String name, String desc) {
JsonNode jsonNode = json.get(name);
assertThat(jsonNode)
.describedAs(desc)
.isNotNull();
assertThat(jsonNode.isObject())
.isTrue();
return jsonNode;
}

private static String getOptionalComment(JsonNode json) {
return Optional.ofNullable(json.get("comment"))
.map(JsonNode::asText)
.orElse("");
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,6 @@

import co.elastic.apm.agent.MockTracer;
import org.junit.jupiter.api.Test;
import org.mockito.Mock;

import static org.assertj.core.api.Assertions.assertThat;

Expand Down Expand Up @@ -69,4 +68,14 @@ void testOutcomeExplicitlyToUnknown() {

assertThat(span.getOutcome()).isEqualTo(Outcome.UNKNOWN);
}

@Test
void normalizeEmptyFields() {
Span span = new Span(MockTracer.create())
.withName("span");

assertThat(span.withType("").getType()).isNull();
assertThat(span.withSubtype("").getSubtype()).isNull();
assertThat(span.withAction("").getAction()).isNull();
}
}
114 changes: 114 additions & 0 deletions apm-agent-core/src/test/resources/json-specs/span_types.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,114 @@
{
"app": {
"subtypes": {
"inferred": {
"comment": "Sampling profiler inferred spans"
}
}
},
"custom": {
"comment": "API custom instrumentation"
},
"db": {
"comment": "database span",
"subtypes": {
"cassandra": {
"comment": "Cassandra"
},
"cosmosdb": {
"comment": "Azure CosmosDB"
},
"db2": {
"comment": "IBM DB2"
},
"dynamodb": {
"comment": "AWS DynamoDB"
},
"elasticsearch": {
"comment": "Elasticsearch"
},
"h2": {
"comment": "H2"
},
"mariadb": {
"comment": "MariaDB"
},
"mongodb": {
"comment": "MongoDB"
},
"mysql": {
"comment": "MySQL"
},
"oracle": {
"comment": "Oracle Database"
},
"postgresql": {
"comment": "PostgreSQL"
},
"redis": {
"comment": "Redis"
},
"sqlserver": {
"comment": "Microsoft SQL Server"
}
}
},
"external": {
"subtypes": {
"dubbo": {
"comment": "Apache Dubbo"
},
"grpc": {
"comment": "gRPC"
},
"http": {
"comment": "HTTP client"
}
}
},
"messaging": {
"subtypes": {
"azurequeue": {
"comment": "Azure Queue"
},
"azureservicebus": {
"comment": "Azure Service Bus"
},
"jms": {
"comment": "Java Messaging Service"
},
"kafka": {
"comment": "Apache Kafka"
},
"rabbitmq": {
"comment": "RabbitMQ"
},
"sns": {
"comment": "AWS Simple Notification Service"
},
"sqs": {
"comment": "AWS Simple Queue Service"
}
}
},
"process": {
"comment": "External process"
},
"storage": {
"subtypes": {
"azurefile": {
"comment": "Azure Files"
},
"azuresblob": {
"comment": "Azure Blob Storage"
},
"s3": {
"comment": "AWS S3"
}
}
},
"template": {
"comment": "Template engines (no sub-type as really platform-specific)"
}
}

Loading