From 79b06a470af701a92327032ddb968a4386d33282 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Gerhard=20St=C3=B6bich?= Date: Tue, 28 Feb 2023 18:08:07 +0100 Subject: [PATCH] feat: improve http span name (#3603) --- experimental/CHANGELOG.md | 3 +++ .../src/http.ts | 15 ++++++----- .../test/functionals/http-enable.test.ts | 27 ++++++++++++++++--- .../test/functionals/http-package.test.ts | 2 +- .../test/functionals/https-enable.test.ts | 6 ++--- .../test/functionals/https-package.test.ts | 2 +- .../test/integrations/http-enable.test.ts | 16 +++++------ .../test/integrations/https-enable.test.ts | 12 ++++----- .../test/utils/assertSpan.ts | 5 +--- 9 files changed, 55 insertions(+), 33 deletions(-) diff --git a/experimental/CHANGELOG.md b/experimental/CHANGELOG.md index e11fb70b016..b85f4c1d5f5 100644 --- a/experimental/CHANGELOG.md +++ b/experimental/CHANGELOG.md @@ -6,8 +6,11 @@ All notable changes to experimental packages in this project will be documented ### :boom: Breaking Change +* feat: remove HTTP/HTTPS prefix from span name [#3603](/~https://github.com/open-telemetry/opentelemetry-js/pull/3603) @Flarna + ### :rocket: (Enhancement) +* feat: use HTTP_ROUTE in span name [#3603](/~https://github.com/open-telemetry/opentelemetry-js/pull/3603) @Flarna * feat: add HTTP_ROUTE attribute to http incoming metrics if present [#3581](/~https://github.com/open-telemetry/opentelemetry-js/pull/3581) @hermogenes * feat(sdk-node): install diag logger with OTEL_LOG_LEVEL [#3627](/~https://github.com/open-telemetry/opentelemetry-js/pull/3627) @legendecas diff --git a/experimental/packages/opentelemetry-instrumentation-http/src/http.ts b/experimental/packages/opentelemetry-instrumentation-http/src/http.ts index ca5e3e8382b..7291780e869 100644 --- a/experimental/packages/opentelemetry-instrumentation-http/src/http.ts +++ b/experimental/packages/opentelemetry-instrumentation-http/src/http.ts @@ -58,6 +58,7 @@ import { } from '@opentelemetry/instrumentation'; import { RPCMetadata, RPCType, setRPCMetadata } from '@opentelemetry/core'; import { errorMonitor } from 'events'; +import { SemanticAttributes } from '@opentelemetry/semantic-conventions'; /** * Http instrumentation instrumentation for Opentelemetry @@ -489,11 +490,7 @@ export class HttpInstrumentation extends InstrumentationBase { utils.getIncomingRequestMetricAttributes(spanAttributes); const ctx = propagation.extract(ROOT_CONTEXT, headers); - const span = instrumentation._startHttpSpan( - `${component.toLocaleUpperCase()} ${method}`, - spanOptions, - ctx - ); + const span = instrumentation._startHttpSpan(method, spanOptions, ctx); const rpcMetadata: RPCMetadata = { type: RPCType.HTTP, span, @@ -622,7 +619,6 @@ export class HttpInstrumentation extends InstrumentationBase { return original.apply(this, [optionsParsed, ...args]); } - const operationName = `${component.toUpperCase()} ${method}`; const { hostname, port } = utils.extractHostnameAndPort(optionsParsed); const attributes = utils.getOutgoingRequestAttributes(optionsParsed, { @@ -643,7 +639,7 @@ export class HttpInstrumentation extends InstrumentationBase { kind: SpanKind.CLIENT, attributes, }; - const span = instrumentation._startHttpSpan(operationName, spanOptions); + const span = instrumentation._startHttpSpan(method, spanOptions); const parentContext = context.active(); const requestContext = trace.setSpan(parentContext, span); @@ -717,6 +713,11 @@ export class HttpInstrumentation extends InstrumentationBase { code: utils.parseResponseStatus(SpanKind.SERVER, response.statusCode), }); + const route = attributes[SemanticAttributes.HTTP_ROUTE]; + if (route) { + span.updateName(`${request.method || 'GET'} ${route}`); + } + if (this._getConfig().applyCustomAttributesOnSpan) { safeExecuteInTheMiddle( () => diff --git a/experimental/packages/opentelemetry-instrumentation-http/test/functionals/http-enable.test.ts b/experimental/packages/opentelemetry-instrumentation-http/test/functionals/http-enable.test.ts index 9ebfe41c80f..5a5588993fd 100644 --- a/experimental/packages/opentelemetry-instrumentation-http/test/functionals/http-enable.test.ts +++ b/experimental/packages/opentelemetry-instrumentation-http/test/functionals/http-enable.test.ts @@ -272,6 +272,13 @@ describe('HttpInstrumentation', () => { if (request.url?.includes('/ignored')) { provider.getTracer('test').startSpan('some-span').end(); } + if (request.url?.includes('/setroute')) { + const rpcData = getRPCMetadata(context.active()); + assert.ok(rpcData != null); + assert.strictEqual(rpcData.type, RPCType.HTTP); + assert.strictEqual(rpcData.route, undefined); + rpcData.route = 'TheRoute'; + } response.end('Test Server Response'); }); @@ -339,6 +346,20 @@ describe('HttpInstrumentation', () => { }); }); + it('should respect HTTP_ROUTE', async () => { + await httpRequest.get( + `${protocol}://${hostname}:${serverPort}/setroute` + ); + const span = memoryExporter.getFinishedSpans()[0]; + + assert.strictEqual(span.kind, SpanKind.SERVER); + assert.strictEqual( + span.attributes[SemanticAttributes.HTTP_ROUTE], + 'TheRoute' + ); + assert.strictEqual(span.name, 'GET TheRoute'); + }); + const httpErrorCodes = [ 400, 401, 403, 404, 429, 501, 503, 504, 500, 505, 597, ]; @@ -404,7 +425,7 @@ describe('HttpInstrumentation', () => { assert.ok(localSpan.name.indexOf('TestRootSpan') >= 0); assert.strictEqual(spans.length, 2); - assert.strictEqual(reqSpan.name, 'HTTP GET'); + assert.strictEqual(reqSpan.name, 'GET'); assert.strictEqual( localSpan.spanContext().traceId, reqSpan.spanContext().traceId @@ -449,7 +470,7 @@ describe('HttpInstrumentation', () => { assert.ok(localSpan.name.indexOf('TestRootSpan') >= 0); assert.strictEqual(spans.length, 2); - assert.strictEqual(reqSpan.name, 'HTTP GET'); + assert.strictEqual(reqSpan.name, 'GET'); assert.strictEqual( localSpan.spanContext().traceId, reqSpan.spanContext().traceId @@ -474,7 +495,7 @@ describe('HttpInstrumentation', () => { for (let i = 0; i < num; i++) { await httpRequest.get(`${protocol}://${hostname}${testPath}`); const spans = memoryExporter.getFinishedSpans(); - assert.strictEqual(spans[i].name, 'HTTP GET'); + assert.strictEqual(spans[i].name, 'GET'); assert.strictEqual( span.spanContext().traceId, spans[i].spanContext().traceId diff --git a/experimental/packages/opentelemetry-instrumentation-http/test/functionals/http-package.test.ts b/experimental/packages/opentelemetry-instrumentation-http/test/functionals/http-package.test.ts index 51b15dd6f3d..39b288ccdbb 100644 --- a/experimental/packages/opentelemetry-instrumentation-http/test/functionals/http-package.test.ts +++ b/experimental/packages/opentelemetry-instrumentation-http/test/functionals/http-package.test.ts @@ -119,7 +119,7 @@ describe('Packages', () => { }; assert.strictEqual(spans.length, 1); - assert.strictEqual(span.name, 'HTTP GET'); + assert.strictEqual(span.name, 'GET'); switch (name) { case 'axios': diff --git a/experimental/packages/opentelemetry-instrumentation-http/test/functionals/https-enable.test.ts b/experimental/packages/opentelemetry-instrumentation-http/test/functionals/https-enable.test.ts index 2f3af2aaeb3..63a0ec00f5d 100644 --- a/experimental/packages/opentelemetry-instrumentation-http/test/functionals/https-enable.test.ts +++ b/experimental/packages/opentelemetry-instrumentation-http/test/functionals/https-enable.test.ts @@ -356,7 +356,7 @@ describe('HttpsInstrumentation', () => { assert.ok(localSpan.name.indexOf('TestRootSpan') >= 0); assert.strictEqual(spans.length, 2); - assert.strictEqual(reqSpan.name, 'HTTPS GET'); + assert.strictEqual(reqSpan.name, 'GET'); assert.strictEqual( localSpan.spanContext().traceId, reqSpan.spanContext().traceId @@ -401,7 +401,7 @@ describe('HttpsInstrumentation', () => { assert.ok(localSpan.name.indexOf('TestRootSpan') >= 0); assert.strictEqual(spans.length, 2); - assert.strictEqual(reqSpan.name, 'HTTPS GET'); + assert.strictEqual(reqSpan.name, 'GET'); assert.strictEqual( localSpan.spanContext().traceId, reqSpan.spanContext().traceId @@ -426,7 +426,7 @@ describe('HttpsInstrumentation', () => { for (let i = 0; i < num; i++) { await httpsRequest.get(`${protocol}://${hostname}${testPath}`); const spans = memoryExporter.getFinishedSpans(); - assert.strictEqual(spans[i].name, 'HTTPS GET'); + assert.strictEqual(spans[i].name, 'GET'); assert.strictEqual( span.spanContext().traceId, spans[i].spanContext().traceId diff --git a/experimental/packages/opentelemetry-instrumentation-http/test/functionals/https-package.test.ts b/experimental/packages/opentelemetry-instrumentation-http/test/functionals/https-package.test.ts index 28e9fb98042..dada803e451 100644 --- a/experimental/packages/opentelemetry-instrumentation-http/test/functionals/https-package.test.ts +++ b/experimental/packages/opentelemetry-instrumentation-http/test/functionals/https-package.test.ts @@ -125,7 +125,7 @@ describe('Packages', () => { }; assert.strictEqual(spans.length, 1); - assert.strictEqual(span.name, 'HTTPS GET'); + assert.strictEqual(span.name, 'GET'); switch (name) { case 'axios': diff --git a/experimental/packages/opentelemetry-instrumentation-http/test/integrations/http-enable.test.ts b/experimental/packages/opentelemetry-instrumentation-http/test/integrations/http-enable.test.ts index b6a8d18d878..625b8b65499 100644 --- a/experimental/packages/opentelemetry-instrumentation-http/test/integrations/http-enable.test.ts +++ b/experimental/packages/opentelemetry-instrumentation-http/test/integrations/http-enable.test.ts @@ -176,7 +176,7 @@ describe('HttpInstrumentation Integration tests', () => { }; assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTP GET'); + assert.strictEqual(span.name, 'GET'); assertSpan(span, SpanKind.CLIENT, validations); }); @@ -203,7 +203,7 @@ describe('HttpInstrumentation Integration tests', () => { }; assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTP GET'); + assert.strictEqual(span.name, 'GET'); assertSpan(span, SpanKind.CLIENT, validations); }); @@ -233,7 +233,7 @@ describe('HttpInstrumentation Integration tests', () => { }; assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTP GET'); + assert.strictEqual(span.name, 'GET'); assert.strictEqual(result.reqHeaders['x-foo'], 'foo'); assert.strictEqual( span.attributes[SemanticAttributes.HTTP_FLAVOR], @@ -264,7 +264,7 @@ describe('HttpInstrumentation Integration tests', () => { }; assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTP GET'); + assert.strictEqual(span.name, 'GET'); assert.strictEqual(span.attributes['span kind'], SpanKind.CLIENT); assertSpan(span, SpanKind.CLIENT, validations); }); @@ -292,7 +292,7 @@ describe('HttpInstrumentation Integration tests', () => { }; assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTP GET'); + assert.strictEqual(span.name, 'GET'); assertSpan(span, SpanKind.CLIENT, validations); }); for (const headers of [ @@ -347,7 +347,7 @@ describe('HttpInstrumentation Integration tests', () => { const span = spans.find(s => s.kind === SpanKind.CLIENT); assert.ok(span); assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTP GET'); + assert.strictEqual(span.name, 'GET'); assert.ok(data); assert.ok(validations.reqHeaders[DummyPropagation.TRACE_CONTEXT_KEY]); assert.ok(validations.reqHeaders[DummyPropagation.SPAN_CONTEXT_KEY]); @@ -362,7 +362,7 @@ describe('HttpInstrumentation Integration tests', () => { const span = spans.find((s: any) => s.kind === SpanKind.SERVER); assert.ok(span); assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTP GET'); + assert.strictEqual(span.name, 'GET'); }); it('should have correct spans even when request timeout', async () => { @@ -381,7 +381,7 @@ describe('HttpInstrumentation Integration tests', () => { spans = memoryExporter.getFinishedSpans(); const span = spans.find(s => s.kind === SpanKind.CLIENT); assert.ok(span); - assert.strictEqual(span.name, 'HTTP GET'); + assert.strictEqual(span.name, 'GET'); assert.strictEqual( span.attributes[SemanticAttributes.HTTP_HOST], `localhost:${mockServerPort}` diff --git a/experimental/packages/opentelemetry-instrumentation-http/test/integrations/https-enable.test.ts b/experimental/packages/opentelemetry-instrumentation-http/test/integrations/https-enable.test.ts index 08d636b1646..10afe76a803 100644 --- a/experimental/packages/opentelemetry-instrumentation-http/test/integrations/https-enable.test.ts +++ b/experimental/packages/opentelemetry-instrumentation-http/test/integrations/https-enable.test.ts @@ -180,7 +180,7 @@ describe('HttpsInstrumentation Integration tests', () => { }; assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTPS GET'); + assert.strictEqual(span.name, 'GET'); assertSpan(span, SpanKind.CLIENT, validations); }); @@ -207,7 +207,7 @@ describe('HttpsInstrumentation Integration tests', () => { }; assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTPS GET'); + assert.strictEqual(span.name, 'GET'); assertSpan(span, SpanKind.CLIENT, validations); }); @@ -237,7 +237,7 @@ describe('HttpsInstrumentation Integration tests', () => { }; assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTPS GET'); + assert.strictEqual(span.name, 'GET'); assert.strictEqual(result.reqHeaders['x-foo'], 'foo'); assert.strictEqual( span.attributes[SemanticAttributes.HTTP_FLAVOR], @@ -268,7 +268,7 @@ describe('HttpsInstrumentation Integration tests', () => { }; assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTPS GET'); + assert.strictEqual(span.name, 'GET'); assert.strictEqual(span.attributes['span kind'], SpanKind.CLIENT); assertSpan(span, SpanKind.CLIENT, validations); }); @@ -296,7 +296,7 @@ describe('HttpsInstrumentation Integration tests', () => { }; assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTPS GET'); + assert.strictEqual(span.name, 'GET'); assertSpan(span, SpanKind.CLIENT, validations); }); for (const headers of [ @@ -351,7 +351,7 @@ describe('HttpsInstrumentation Integration tests', () => { const span = spans.find(s => s.kind === SpanKind.CLIENT); assert.ok(span); assert.strictEqual(spans.length, 2); - assert.strictEqual(span.name, 'HTTPS GET'); + assert.strictEqual(span.name, 'GET'); assert.ok(data); assert.ok(validations.reqHeaders[DummyPropagation.TRACE_CONTEXT_KEY]); assert.ok(validations.reqHeaders[DummyPropagation.SPAN_CONTEXT_KEY]); diff --git a/experimental/packages/opentelemetry-instrumentation-http/test/utils/assertSpan.ts b/experimental/packages/opentelemetry-instrumentation-http/test/utils/assertSpan.ts index 2576b599d62..1f1b0518c8c 100644 --- a/experimental/packages/opentelemetry-instrumentation-http/test/utils/assertSpan.ts +++ b/experimental/packages/opentelemetry-instrumentation-http/test/utils/assertSpan.ts @@ -49,10 +49,7 @@ export const assertSpan = ( assert.strictEqual(span.spanContext().traceId.length, 32); assert.strictEqual(span.spanContext().spanId.length, 16); assert.strictEqual(span.kind, kind); - assert.strictEqual( - span.name, - `${validations.component.toUpperCase()} ${validations.httpMethod}` - ); + assert.strictEqual(span.name, validations.httpMethod); assert.strictEqual( span.attributes[AttributeNames.HTTP_ERROR_MESSAGE], span.status.message