From fc94e4669a94e688204a919165a020323fc1d5c2 Mon Sep 17 00:00:00 2001 From: Marc Pichler Date: Thu, 28 Nov 2024 18:42:15 +0100 Subject: [PATCH] feat(otlp-exporter-base): implement partial success handling (#5183) --- experimental/CHANGELOG.md | 1 + .../src/logging-response-handler.ts | 49 ++++++ .../src/otlp-export-delegate.ts | 17 ++ .../src/response-handler.ts | 27 ++++ .../common/logging-response-handler.test.ts | 86 +++++++++++ .../test/common/otlp-export-delegate.test.ts | 145 +++++++++++++++++- .../test/common/test-utils.ts | 32 ++++ ...vert-legacy-node-otlp-http-options.test.ts | 14 +- 8 files changed, 356 insertions(+), 15 deletions(-) create mode 100644 experimental/packages/otlp-exporter-base/src/logging-response-handler.ts create mode 100644 experimental/packages/otlp-exporter-base/src/response-handler.ts create mode 100644 experimental/packages/otlp-exporter-base/test/common/logging-response-handler.test.ts create mode 100644 experimental/packages/otlp-exporter-base/test/common/test-utils.ts diff --git a/experimental/CHANGELOG.md b/experimental/CHANGELOG.md index eebd013262b..a888113b1e4 100644 --- a/experimental/CHANGELOG.md +++ b/experimental/CHANGELOG.md @@ -22,6 +22,7 @@ All notable changes to experimental packages in this project will be documented ### :rocket: (Enhancement) +* feat(otlp-exporter-base): handle OTLP partial success [#5183](https://github.com/open-telemetry/opentelemetry-js/pull/5183) @pichlermarc * feat(otlp-exporter-base): internally accept a http header provider function only [#5179](https://github.com/open-telemetry/opentelemetry-js/pull/5179) @pichlermarc * refactor(otlp-exporter-base): don't create blob before sending xhr [#5193](https://github.com/open-telemetry/opentelemetry-js/pull/5193) @pichlermarc * improves compatibility with some unsupported runtimes diff --git a/experimental/packages/otlp-exporter-base/src/logging-response-handler.ts b/experimental/packages/otlp-exporter-base/src/logging-response-handler.ts new file mode 100644 index 00000000000..3984eb6cba1 --- /dev/null +++ b/experimental/packages/otlp-exporter-base/src/logging-response-handler.ts @@ -0,0 +1,49 @@ +/* + * Copyright The OpenTelemetry Authors + * + * Licensed 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 + * + * https://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. + */ +import { diag } from '@opentelemetry/api'; +import { IOtlpResponseHandler } from './response-handler'; + +function isPartialSuccessResponse( + response: unknown +): response is { partialSuccess: never } { + return Object.prototype.hasOwnProperty.call(response, 'partialSuccess'); +} + +/** + * Default response handler that logs a partial success to the console. + */ +export function createLoggingPartialSuccessResponseHandler< + T, +>(): IOtlpResponseHandler { + return { + handleResponse(response: T) { + // Partial success MUST never be an empty object according the specification + // see https://opentelemetry.io/docs/specs/otlp/#partial-success + if ( + response == null || + !isPartialSuccessResponse(response) || + response.partialSuccess == null || + Object.keys(response.partialSuccess).length === 0 + ) { + return; + } + diag.warn( + 'Received Partial Success response:', + JSON.stringify(response.partialSuccess) + ); + }, + }; +} diff --git a/experimental/packages/otlp-exporter-base/src/otlp-export-delegate.ts b/experimental/packages/otlp-exporter-base/src/otlp-export-delegate.ts index df42eb6061a..00d8113a799 100644 --- a/experimental/packages/otlp-exporter-base/src/otlp-export-delegate.ts +++ b/experimental/packages/otlp-exporter-base/src/otlp-export-delegate.ts @@ -19,6 +19,8 @@ import { IExporterTransport } from './exporter-transport'; import { IExportPromiseHandler } from './bounded-queue-export-promise-handler'; import { ISerializer } from '@opentelemetry/otlp-transformer'; import { OTLPExporterError } from './types'; +import { IOtlpResponseHandler } from './response-handler'; +import { createLoggingPartialSuccessResponseHandler } from './logging-response-handler'; import { diag, DiagLogger } from '@opentelemetry/api'; /** @@ -40,6 +42,7 @@ class OTLPExportDelegate constructor( private _transport: IExporterTransport, private _serializer: ISerializer, + private _responseHandler: IOtlpResponseHandler, private _promiseQueue: IExportPromiseHandler, private _timeout: number ) { @@ -79,6 +82,19 @@ class OTLPExportDelegate this._transport.send(serializedRequest, this._timeout).then( response => { if (response.status === 'success') { + if (response.data != null) { + try { + this._responseHandler.handleResponse( + this._serializer.deserializeResponse(response.data) + ); + } catch (e) { + this._diagLogger.warn( + 'Export succeeded but could not deserialize response - is the response specification compliant?', + e, + response.data + ); + } + } // No matter the response, we can consider the export still successful. resultCallback({ code: ExportResultCode.SUCCESS, @@ -139,6 +155,7 @@ export function createOtlpExportDelegate( return new OTLPExportDelegate( components.transport, components.serializer, + createLoggingPartialSuccessResponseHandler(), components.promiseHandler, settings.timeout ); diff --git a/experimental/packages/otlp-exporter-base/src/response-handler.ts b/experimental/packages/otlp-exporter-base/src/response-handler.ts new file mode 100644 index 00000000000..69f6213efa0 --- /dev/null +++ b/experimental/packages/otlp-exporter-base/src/response-handler.ts @@ -0,0 +1,27 @@ +/* + * Copyright The OpenTelemetry Authors + * + * Licensed 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 + * + * https://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. + */ + +/** + * Generic export response handler. Can be implemented to handle export responses like partial success. + */ +export interface IOtlpResponseHandler { + /** + * Handles an OTLP export response. + * Implementations MUST NOT throw. + * @param response + */ + handleResponse(response: Response): void; +} diff --git a/experimental/packages/otlp-exporter-base/test/common/logging-response-handler.test.ts b/experimental/packages/otlp-exporter-base/test/common/logging-response-handler.test.ts new file mode 100644 index 00000000000..7ab8a6981d9 --- /dev/null +++ b/experimental/packages/otlp-exporter-base/test/common/logging-response-handler.test.ts @@ -0,0 +1,86 @@ +/* + * Copyright The OpenTelemetry Authors + * + * Licensed 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 + * + * https://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. + */ +import { createLoggingPartialSuccessResponseHandler } from '../../src/logging-response-handler'; +import * as sinon from 'sinon'; +import { IExportTraceServiceResponse } from '@opentelemetry/otlp-transformer'; +import { registerMockDiagLogger } from './test-utils'; + +describe('loggingResponseHandler', function () { + afterEach(function () { + sinon.restore(); + }); + + it('should diag warn if a partial success is passed', function () { + // arrange + const { warn } = registerMockDiagLogger(); + const handler = + createLoggingPartialSuccessResponseHandler(); + const partialSuccessResponse: IExportTraceServiceResponse = { + partialSuccess: { + errorMessage: 'error', + rejectedSpans: 10, + }, + }; + + // act + handler.handleResponse(partialSuccessResponse); + + //assert + sinon.assert.calledOnceWithExactly( + warn, + 'Received Partial Success response:', + JSON.stringify(partialSuccessResponse.partialSuccess) + ); + }); + + it('should not warn when a response is undefined', function () { + // arrange + const { warn } = registerMockDiagLogger(); + const handler = createLoggingPartialSuccessResponseHandler(); + + // act + handler.handleResponse(undefined); + + //assert + sinon.assert.notCalled(warn); + }); + + it('should not warn when a response is defined but partialSuccess is undefined', function () { + // arrange + const { warn } = registerMockDiagLogger(); + const handler = createLoggingPartialSuccessResponseHandler(); + + // act + handler.handleResponse({ partialSuccess: undefined }); + + //assert + sinon.assert.notCalled(warn); + }); + + it('should not warn when a response is defined but partialSuccess is empty object', function () { + // note: this is the common case for the OTel collector's OTLP receiver, client should treat is as full success + // arrange + const { warn } = registerMockDiagLogger(); + const handler = createLoggingPartialSuccessResponseHandler(); + const response = { partialSuccess: {} }; + + // act + handler.handleResponse(response); + + //assert + sinon.assert.notCalled(warn); + }); +}); diff --git a/experimental/packages/otlp-exporter-base/test/common/otlp-export-delegate.test.ts b/experimental/packages/otlp-exporter-base/test/common/otlp-export-delegate.test.ts index 96a636c7075..de27eaaf05e 100644 --- a/experimental/packages/otlp-exporter-base/test/common/otlp-export-delegate.test.ts +++ b/experimental/packages/otlp-exporter-base/test/common/otlp-export-delegate.test.ts @@ -22,13 +22,14 @@ import { createOtlpExportDelegate } from '../../src/otlp-export-delegate'; import { ExportResponse } from '../../src'; import { ISerializer } from '@opentelemetry/otlp-transformer'; import { IExportPromiseHandler } from '../../src/bounded-queue-export-promise-handler'; +import { registerMockDiagLogger } from './test-utils'; interface FakeInternalRepresentation { foo: string; } interface FakeSignalResponse { - baz: string; + partialSuccess?: { foo: string }; } type FakeSerializer = ISerializer< @@ -491,9 +492,7 @@ describe('OTLPExportDelegate', function () { }; const mockTransport = transportStubs; - const response: FakeSignalResponse = { - baz: 'partial success', - }; + const response: FakeSignalResponse = {}; const serializerStubs = { // simulate that the serializer returns something to send @@ -541,6 +540,144 @@ describe('OTLPExportDelegate', function () { }); }); + it('returns success even if response cannot be deserialized', function (done) { + const { warn } = registerMockDiagLogger(); + // returns mock success response (empty body) + const exportResponse: ExportResponse = { + data: Uint8Array.from([]), + status: 'success', + }; + + // transport does not need to do anything in this case. + const transportStubs = { + send: sinon.stub().returns(Promise.resolve(exportResponse)), + shutdown: sinon.stub(), + }; + const mockTransport = transportStubs; + + const serializerStubs = { + // simulate that the serializer returns something to send + serializeRequest: sinon.stub().returns(Uint8Array.from([1])), + // simulate that it returns a partial success (response with contents) + deserializeResponse: sinon.stub().throws(new Error()), + }; + const mockSerializer = serializerStubs; + + // mock a queue that has not yet reached capacity + const promiseHandlerStubs = { + pushPromise: sinon.stub(), + hasReachedLimit: sinon.stub().returns(false), + awaitAll: sinon.stub(), + }; + const promiseHandler = promiseHandlerStubs; + + const exporter = createOtlpExportDelegate( + { + promiseHandler: promiseHandler, + serializer: mockSerializer, + transport: mockTransport, + }, + { + timeout: 1000, + } + ); + + exporter.export(internalRepresentation, result => { + try { + assert.strictEqual(result.code, ExportResultCode.SUCCESS); + assert.strictEqual(result.error, undefined); + + // assert here as otherwise the promise will not have executed yet + sinon.assert.calledOnceWithMatch( + warn, + 'OTLPExportDelegate', + 'Export succeeded but could not deserialize response - is the response specification compliant?', + sinon.match.instanceOf(Error), + exportResponse.data + ); + sinon.assert.calledOnce(serializerStubs.serializeRequest); + sinon.assert.calledOnce(transportStubs.send); + sinon.assert.calledOnce(promiseHandlerStubs.pushPromise); + sinon.assert.calledOnce(promiseHandlerStubs.hasReachedLimit); + sinon.assert.notCalled(promiseHandlerStubs.awaitAll); + done(); + } catch (err) { + // ensures we throw if there are more calls to result; + done(err); + } + }); + }); + + it('returns success and warns on partial success response', function (done) { + const { warn } = registerMockDiagLogger(); + // returns mock success response (empty body) + const exportResponse: ExportResponse = { + data: Uint8Array.from([]), + status: 'success', + }; + + // transport does not need to do anything in this case. + const transportStubs = { + send: sinon.stub().returns(Promise.resolve(exportResponse)), + shutdown: sinon.stub(), + }; + const mockTransport = transportStubs; + + const partialSuccessResponse: FakeSignalResponse = { + partialSuccess: { foo: 'bar' }, + }; + + const serializerStubs = { + // simulate that the serializer returns something to send + serializeRequest: sinon.stub().returns(Uint8Array.from([1])), + // simulate that it returns a partial success (response with contents) + deserializeResponse: sinon.stub().returns(partialSuccessResponse), + }; + const mockSerializer = serializerStubs; + + // mock a queue that has not yet reached capacity + const promiseHandlerStubs = { + pushPromise: sinon.stub(), + hasReachedLimit: sinon.stub().returns(false), + awaitAll: sinon.stub(), + }; + const promiseHandler = promiseHandlerStubs; + + const exporter = createOtlpExportDelegate( + { + promiseHandler: promiseHandler, + serializer: mockSerializer, + transport: mockTransport, + }, + { + timeout: 1000, + } + ); + + exporter.export(internalRepresentation, result => { + try { + assert.strictEqual(result.code, ExportResultCode.SUCCESS); + assert.strictEqual(result.error, undefined); + + // assert here as otherwise the promise will not have executed yet + sinon.assert.calledOnceWithMatch( + warn, + 'Received Partial Success response:', + JSON.stringify(partialSuccessResponse.partialSuccess) + ); + sinon.assert.calledOnce(serializerStubs.serializeRequest); + sinon.assert.calledOnce(transportStubs.send); + sinon.assert.calledOnce(promiseHandlerStubs.pushPromise); + sinon.assert.calledOnce(promiseHandlerStubs.hasReachedLimit); + sinon.assert.notCalled(promiseHandlerStubs.awaitAll); + done(); + } catch (err) { + // ensures we throw if there are more calls to result; + done(err); + } + }); + }); + it('returns failure when send rejects', function (done) { const transportStubs = { // make transport reject diff --git a/experimental/packages/otlp-exporter-base/test/common/test-utils.ts b/experimental/packages/otlp-exporter-base/test/common/test-utils.ts new file mode 100644 index 00000000000..b8752f6b94c --- /dev/null +++ b/experimental/packages/otlp-exporter-base/test/common/test-utils.ts @@ -0,0 +1,32 @@ +/* + * Copyright The OpenTelemetry Authors + * + * Licensed 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 + * + * https://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. + */ +import * as sinon from 'sinon'; +import { diag } from '@opentelemetry/api'; + +export function registerMockDiagLogger() { + // arrange + const stubs = { + verbose: sinon.stub(), + debug: sinon.stub(), + info: sinon.stub(), + warn: sinon.stub(), + error: sinon.stub(), + }; + diag.setLogger(stubs); + stubs.warn.resetHistory(); // reset history setLogger will warn if another has already been set + + return stubs; +} diff --git a/experimental/packages/otlp-exporter-base/test/node/configuration/convert-legacy-node-otlp-http-options.test.ts b/experimental/packages/otlp-exporter-base/test/node/configuration/convert-legacy-node-otlp-http-options.test.ts index 73fe38b52e5..d436933161d 100644 --- a/experimental/packages/otlp-exporter-base/test/node/configuration/convert-legacy-node-otlp-http-options.test.ts +++ b/experimental/packages/otlp-exporter-base/test/node/configuration/convert-legacy-node-otlp-http-options.test.ts @@ -14,11 +14,10 @@ * limitations under the License. */ -import { diag } from '@opentelemetry/api'; - import * as sinon from 'sinon'; import * as assert from 'assert'; import { convertLegacyHttpOptions } from '../../../src/configuration/convert-legacy-node-http-options'; +import { registerMockDiagLogger } from '../../common/test-utils'; describe('convertLegacyHttpOptions', function () { afterEach(function () { @@ -26,14 +25,7 @@ describe('convertLegacyHttpOptions', function () { }); it('should warn when used with metadata', function () { - const warnStub = sinon.stub(); - diag.setLogger({ - verbose: sinon.stub(), - debug: sinon.stub(), - info: sinon.stub(), - warn: warnStub, - error: sinon.stub(), - }); + const { warn } = registerMockDiagLogger(); convertLegacyHttpOptions( { metadata: { foo: 'bar' } } as any, @@ -43,7 +35,7 @@ describe('convertLegacyHttpOptions', function () { ); sinon.assert.calledOnceWithExactly( - warnStub, + warn, 'Metadata cannot be set when using http' ); });