diff --git a/examples/tracer-web/examples/zipkin/index.html b/examples/tracer-web/examples/zipkin/index.html new file mode 100644 index 00000000000..bf98b074336 --- /dev/null +++ b/examples/tracer-web/examples/zipkin/index.html @@ -0,0 +1,19 @@ + + + + + + Zipkin Exporter Example + + + + + + Example of using Web Tracer with Zipkin Exporter + +
+ + + + + diff --git a/examples/tracer-web/examples/zipkin/index.js b/examples/tracer-web/examples/zipkin/index.js new file mode 100644 index 00000000000..410749b00af --- /dev/null +++ b/examples/tracer-web/examples/zipkin/index.js @@ -0,0 +1,23 @@ +import { ConsoleSpanExporter, SimpleSpanProcessor } from '@opentelemetry/tracing'; +import { WebTracerProvider } from '@opentelemetry/web'; +import { ZipkinExporter } from '@opentelemetry/exporter-zipkin'; + +const provider = new WebTracerProvider(); +provider.addSpanProcessor(new SimpleSpanProcessor(new ConsoleSpanExporter())); +provider.addSpanProcessor(new SimpleSpanProcessor(new ZipkinExporter())); + +provider.register(); + +const tracer = provider.getTracer('example-tracer-web'); + +const prepareClickEvent = () => { + const element = document.getElementById('button1'); + + const onClick = () => { + const span = tracer.startSpan('foo'); + span.end(); + }; + element.addEventListener('click', onClick); +}; + +window.addEventListener('load', prepareClickEvent); diff --git a/examples/tracer-web/package.json b/examples/tracer-web/package.json index 005ca3216a1..711d2e5292f 100644 --- a/examples/tracer-web/package.json +++ b/examples/tracer-web/package.json @@ -37,6 +37,7 @@ "@opentelemetry/context-zone": "^0.11.0", "@opentelemetry/core": "^0.11.0", "@opentelemetry/exporter-collector": "^0.11.0", + "@opentelemetry/exporter-zipkin": "^0.11.0", "@opentelemetry/metrics": "^0.11.0", "@opentelemetry/plugin-document-load": "^0.9.0", "@opentelemetry/plugin-fetch": "^0.11.0", diff --git a/examples/tracer-web/webpack.config.js b/examples/tracer-web/webpack.config.js index 3859f2f8fd9..00773db28b6 100644 --- a/examples/tracer-web/webpack.config.js +++ b/examples/tracer-web/webpack.config.js @@ -12,6 +12,7 @@ const common = { fetch: 'examples/fetch/index.js', 'xml-http-request': 'examples/xml-http-request/index.js', 'user-interaction': 'examples/user-interaction/index.js', + zipkin: 'examples/zipkin/index.js', }, output: { path: path.resolve(__dirname, 'dist'), diff --git a/packages/opentelemetry-exporter-zipkin/.eslintrc.js b/packages/opentelemetry-exporter-zipkin/.eslintrc.js index f726f3becb6..9dfe62f9b8c 100644 --- a/packages/opentelemetry-exporter-zipkin/.eslintrc.js +++ b/packages/opentelemetry-exporter-zipkin/.eslintrc.js @@ -1,7 +1,9 @@ module.exports = { "env": { "mocha": true, - "node": true + "commonjs": true, + "node": true, + "browser": true }, ...require('../../eslint.config.js') } diff --git a/packages/opentelemetry-exporter-zipkin/README.md b/packages/opentelemetry-exporter-zipkin/README.md index ccd4404c677..1c1466ae5e1 100644 --- a/packages/opentelemetry-exporter-zipkin/README.md +++ b/packages/opentelemetry-exporter-zipkin/README.md @@ -16,7 +16,7 @@ OpenTelemetry Zipkin Trace Exporter allows the user to send collected traces to npm install --save @opentelemetry/exporter-zipkin ``` -## Usage +## Usage in Node and Browser Install the exporter on your application and pass the options. `serviceName` is an optional string. If omitted, the exporter will first try to get the service name from the Resource. If no service name can be detected on the Resource, a fallback name of "OpenTelemetry Service" will be used. diff --git a/packages/opentelemetry-exporter-zipkin/karma.conf.js b/packages/opentelemetry-exporter-zipkin/karma.conf.js new file mode 100644 index 00000000000..455b1437c87 --- /dev/null +++ b/packages/opentelemetry-exporter-zipkin/karma.conf.js @@ -0,0 +1,26 @@ +/*! + * 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 + * + * 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. + */ + +const karmaWebpackConfig = require('../../karma.webpack'); +const karmaBaseConfig = require('../../karma.base'); + +module.exports = (config) => { + config.set(Object.assign({}, karmaBaseConfig, { + webpack: karmaWebpackConfig, + files: ['test/browser/index-webpack.ts'], + preprocessors: { 'test/browser/index-webpack.ts': ['webpack'] } + })) +}; diff --git a/packages/opentelemetry-exporter-zipkin/package.json b/packages/opentelemetry-exporter-zipkin/package.json index a6620123912..57d002ea482 100644 --- a/packages/opentelemetry-exporter-zipkin/package.json +++ b/packages/opentelemetry-exporter-zipkin/package.json @@ -5,21 +5,29 @@ "main": "build/src/index.js", "types": "build/src/index.d.ts", "repository": "open-telemetry/opentelemetry-js", + "browser": { + "./src/platform/index.ts": "./src/platform/browser/index.ts", + "./build/src/platform/index.js": "./build/src/platform/browser/index.js" + }, "scripts": { - "test": "nyc ts-mocha -p tsconfig.json 'test/**/*.test.ts'", - "tdd": "npm run test -- --watch-extensions ts --watch", "clean": "rimraf build/*", + "codecov": "nyc report --reporter=json && codecov -f coverage/*.json -p ../../", + "codecov:browser": "nyc report --reporter=json && codecov -f coverage/*.json -p ../../", + "compile": "npm run version:update && tsc -p .", "lint": "eslint . --ext .ts", "lint:fix": "eslint . --ext .ts --fix", - "codecov": "nyc report --reporter=json && codecov -f coverage/*.json -p ../../", "precompile": "tsc --version", + "prepare": "npm run compile", + "tdd": "npm run test -- --watch-extensions ts --watch", + "test": "nyc ts-mocha -p tsconfig.json 'test/**/*.test.ts' --exclude 'test/browser/**/*.ts'", + "test:browser": "nyc karma start --single-run", "version:update": "node ../../scripts/version-update.js", - "compile": "npm run version:update && tsc -p .", - "prepare": "npm run compile" + "watch": "tsc -w" }, "keywords": [ "opentelemetry", "nodejs", + "browser", "tracing", "profiling" ], @@ -40,17 +48,33 @@ "access": "public" }, "devDependencies": { + "@babel/core": "7.11.0", "@types/mocha": "8.0.2", "@types/node": "14.0.27", + "@types/sinon": "9.0.4", + "@types/webpack-env": "1.15.2", + "babel-loader": "8.1.0", "codecov": "3.7.2", "gts": "2.0.2", + "istanbul-instrumenter-loader": "3.0.1", + "karma": "5.1.1", + "karma-chrome-launcher": "3.1.0", + "karma-coverage-istanbul-reporter": "3.0.3", + "karma-mocha": "2.0.1", + "karma-spec-reporter": "0.0.32", + "karma-webpack": "4.0.2", "mocha": "7.2.0", "nock": "12.0.3", "nyc": "15.1.0", "rimraf": "3.0.2", + "sinon": "9.0.2", + "ts-loader": "8.0.1", "ts-mocha": "7.0.0", "ts-node": "8.10.2", - "typescript": "3.9.7" + "typescript": "3.9.7", + "webpack": "4.44.1", + "webpack-cli": "3.3.12", + "webpack-merge": "5.0.9" }, "dependencies": { "@opentelemetry/api": "^0.11.0", diff --git a/packages/opentelemetry-exporter-zipkin/src/index.ts b/packages/opentelemetry-exporter-zipkin/src/index.ts index 9a35ec9d07a..4879fea44f9 100644 --- a/packages/opentelemetry-exporter-zipkin/src/index.ts +++ b/packages/opentelemetry-exporter-zipkin/src/index.ts @@ -15,3 +15,4 @@ */ export * from './zipkin'; +export * from './platform'; diff --git a/packages/opentelemetry-exporter-zipkin/test/e2e.test.ts b/packages/opentelemetry-exporter-zipkin/src/platform/browser/index.ts similarity index 87% rename from packages/opentelemetry-exporter-zipkin/test/e2e.test.ts rename to packages/opentelemetry-exporter-zipkin/src/platform/browser/index.ts index 96181daa517..3786b948273 100644 --- a/packages/opentelemetry-exporter-zipkin/test/e2e.test.ts +++ b/packages/opentelemetry-exporter-zipkin/src/platform/browser/index.ts @@ -14,6 +14,4 @@ * limitations under the License. */ -describe('Zipkin Exporter E2E', () => { - it('should report spans to Zipkin server'); -}); +export * from './util'; diff --git a/packages/opentelemetry-exporter-zipkin/src/platform/browser/util.ts b/packages/opentelemetry-exporter-zipkin/src/platform/browser/util.ts new file mode 100644 index 00000000000..a954fb2d421 --- /dev/null +++ b/packages/opentelemetry-exporter-zipkin/src/platform/browser/util.ts @@ -0,0 +1,131 @@ +/* + * 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 api from '@opentelemetry/api'; +import { ExportResult } from '@opentelemetry/core'; +import * as zipkinTypes from '../../types'; +import { OT_REQUEST_HEADER } from '../../utils'; + +/** + * Prepares send function that will send spans to the remote Zipkin service. + */ +export function prepareSend( + logger: api.Logger, + urlStr: string, + headers?: Record +) { + let xhrHeaders: Record; + const useBeacon = navigator.sendBeacon && !headers; + if (headers) { + xhrHeaders = { + Accept: 'application/json', + 'Content-Type': 'application/json', + [OT_REQUEST_HEADER]: '1', + ...headers, + }; + } + + /** + * Send spans to the remote Zipkin service. + */ + return function send( + zipkinSpans: zipkinTypes.Span[], + done: (result: ExportResult) => void + ) { + if (zipkinSpans.length === 0) { + logger.debug('Zipkin send with empty spans'); + return done(ExportResult.SUCCESS); + } + const payload = JSON.stringify(zipkinSpans); + if (useBeacon) { + sendWithBeacon(payload, done, urlStr, logger); + } else { + sendWithXhr(payload, done, urlStr, logger, xhrHeaders); + } + }; +} + +/** + * Sends data using beacon + * @param data + * @param done + * @param urlStr + * @param logger + */ +function sendWithBeacon( + data: string, + done: (result: ExportResult) => void, + urlStr: string, + logger: api.Logger +) { + if (navigator.sendBeacon(urlStr, data)) { + logger.debug('sendBeacon - can send', data); + done(ExportResult.SUCCESS); + } else { + logger.error('sendBeacon - cannot send', data); + done(ExportResult.FAILED_NOT_RETRYABLE); + } +} + +/** + * Sends data using XMLHttpRequest + * @param data + * @param done + * @param urlStr + * @param logger + * @param xhrHeaders + */ +function sendWithXhr( + data: string, + done: (result: ExportResult) => void, + urlStr: string, + logger: api.Logger, + xhrHeaders: Record = {} +) { + const xhr = new window.XMLHttpRequest(); + xhr.open('POST', urlStr); + Object.entries(xhrHeaders).forEach(([k, v]) => { + xhr.setRequestHeader(k, v); + }); + + xhr.onreadystatechange = () => { + if (xhr.readyState === XMLHttpRequest.DONE) { + const statusCode = xhr.status || 0; + logger.debug( + 'Zipkin response status code: %d, body: %s', + statusCode, + data + ); + + if (xhr.status >= 200 && xhr.status < 400) { + return done(ExportResult.SUCCESS); + } else if (statusCode < 500) { + return done(ExportResult.FAILED_NOT_RETRYABLE); + } else { + return done(ExportResult.FAILED_RETRYABLE); + } + } + }; + + xhr.onerror = err => { + logger.error('Zipkin request error', err); + return done(ExportResult.FAILED_RETRYABLE); + }; + + // Issue request to remote service + logger.debug('Zipkin request payload: %s', data); + xhr.send(data); +} diff --git a/packages/opentelemetry-exporter-zipkin/src/platform/index.ts b/packages/opentelemetry-exporter-zipkin/src/platform/index.ts new file mode 100644 index 00000000000..cdaf8858ce5 --- /dev/null +++ b/packages/opentelemetry-exporter-zipkin/src/platform/index.ts @@ -0,0 +1,17 @@ +/* + * 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. + */ + +export * from './node'; diff --git a/packages/opentelemetry-exporter-zipkin/src/platform/node/index.ts b/packages/opentelemetry-exporter-zipkin/src/platform/node/index.ts new file mode 100644 index 00000000000..3786b948273 --- /dev/null +++ b/packages/opentelemetry-exporter-zipkin/src/platform/node/index.ts @@ -0,0 +1,17 @@ +/* + * 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. + */ + +export * from './util'; diff --git a/packages/opentelemetry-exporter-zipkin/src/platform/node/util.ts b/packages/opentelemetry-exporter-zipkin/src/platform/node/util.ts new file mode 100644 index 00000000000..89f740b8e2b --- /dev/null +++ b/packages/opentelemetry-exporter-zipkin/src/platform/node/util.ts @@ -0,0 +1,97 @@ +/* + * 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 api from '@opentelemetry/api'; +import { ExportResult } from '@opentelemetry/core'; +import * as http from 'http'; +import * as https from 'https'; +import * as url from 'url'; +import * as zipkinTypes from '../../types'; +import { OT_REQUEST_HEADER } from '../../utils'; + +/** + * Prepares send function that will send spans to the remote Zipkin service. + */ +export function prepareSend( + logger: api.Logger, + urlStr: string, + headers?: Record +) { + const urlOpts = url.parse(urlStr); + + const reqOpts: http.RequestOptions = Object.assign( + { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + [OT_REQUEST_HEADER]: 1, + ...headers, + }, + }, + urlOpts + ); + + /** + * Send spans to the remote Zipkin service. + */ + return function send( + zipkinSpans: zipkinTypes.Span[], + done: (result: ExportResult) => void + ) { + if (zipkinSpans.length === 0) { + logger.debug('Zipkin send with empty spans'); + return done(ExportResult.SUCCESS); + } + + const { request } = reqOpts.protocol === 'http:' ? http : https; + const req = request(reqOpts, (res: http.IncomingMessage) => { + let rawData = ''; + res.on('data', chunk => { + rawData += chunk; + }); + res.on('end', () => { + const statusCode = res.statusCode || 0; + logger.debug( + 'Zipkin response status code: %d, body: %s', + statusCode, + rawData + ); + + // Consider 2xx and 3xx as success. + if (statusCode < 400) { + return done(ExportResult.SUCCESS); + // Consider 4xx as failed non-retriable. + } else if (statusCode < 500) { + return done(ExportResult.FAILED_NOT_RETRYABLE); + // Consider 5xx as failed retriable. + } else { + return done(ExportResult.FAILED_RETRYABLE); + } + }); + }); + + req.on('error', (err: Error) => { + logger.error('Zipkin request error', err); + return done(ExportResult.FAILED_RETRYABLE); + }); + + // Issue request to remote service + const payload = JSON.stringify(zipkinSpans); + logger.debug('Zipkin request payload: %s', payload); + req.write(payload, 'utf8'); + req.end(); + }; +} diff --git a/packages/opentelemetry-exporter-zipkin/src/types.ts b/packages/opentelemetry-exporter-zipkin/src/types.ts index 0433351491d..7eb7eaeaf29 100644 --- a/packages/opentelemetry-exporter-zipkin/src/types.ts +++ b/packages/opentelemetry-exporter-zipkin/src/types.ts @@ -15,6 +15,7 @@ */ import * as api from '@opentelemetry/api'; +import { ExportResult } from '@opentelemetry/core'; /** * Exporter config @@ -177,3 +178,11 @@ export enum SpanKind { CONSUMER = 'CONSUMER', PRODUCER = 'PRODUCER', } + +/** + * interface for function that will send zipkin spans + */ +export type SendFunction = ( + zipkinSpans: Span[], + done: (result: ExportResult) => void +) => void; diff --git a/packages/opentelemetry-exporter-zipkin/src/zipkin.ts b/packages/opentelemetry-exporter-zipkin/src/zipkin.ts index 0a84cb58b8d..bb1083bb232 100644 --- a/packages/opentelemetry-exporter-zipkin/src/zipkin.ts +++ b/packages/opentelemetry-exporter-zipkin/src/zipkin.ts @@ -15,19 +15,17 @@ */ import * as api from '@opentelemetry/api'; -import * as http from 'http'; -import * as https from 'https'; -import * as url from 'url'; import { ExportResult, NoopLogger } from '@opentelemetry/core'; import { SpanExporter, ReadableSpan } from '@opentelemetry/tracing'; +import { prepareSend } from './platform/index'; import * as zipkinTypes from './types'; import { toZipkinSpan, statusCodeTagName, statusDescriptionTagName, } from './transform'; -import { OT_REQUEST_HEADER } from './utils'; import { SERVICE_RESOURCE } from '@opentelemetry/resources'; + /** * Zipkin Exporter */ @@ -37,27 +35,15 @@ export class ZipkinExporter implements SpanExporter { private readonly _logger: api.Logger; private readonly _statusCodeTagName: string; private readonly _statusDescriptionTagName: string; - private readonly _reqOpts: http.RequestOptions; + private _send: zipkinTypes.SendFunction; private _serviceName?: string; private _isShutdown: boolean; private _sendingPromises: Promise[] = []; constructor(config: zipkinTypes.ExporterConfig = {}) { const urlStr = config.url || ZipkinExporter.DEFAULT_URL; - const urlOpts = url.parse(urlStr); - this._logger = config.logger || new NoopLogger(); - this._reqOpts = Object.assign( - { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - [OT_REQUEST_HEADER]: 1, - ...config.headers, - }, - }, - urlOpts - ); + this._send = prepareSend(this._logger, urlStr, config.headers); this._serviceName = config.serviceName; this._statusCodeTagName = config.statusCodeTagName || statusCodeTagName; this._statusDescriptionTagName = @@ -129,55 +115,4 @@ export class ZipkinExporter implements SpanExporter { } }); } - - /** - * Send spans to the remote Zipkin service. - */ - private _send( - zipkinSpans: zipkinTypes.Span[], - done: (result: ExportResult) => void - ) { - if (zipkinSpans.length === 0) { - this._logger.debug('Zipkin send with empty spans'); - return done(ExportResult.SUCCESS); - } - - const { request } = this._reqOpts.protocol === 'http:' ? http : https; - const req = request(this._reqOpts, (res: http.IncomingMessage) => { - let rawData = ''; - res.on('data', chunk => { - rawData += chunk; - }); - res.on('end', () => { - const statusCode = res.statusCode || 0; - this._logger.debug( - 'Zipkin response status code: %d, body: %s', - statusCode, - rawData - ); - - // Consider 2xx and 3xx as success. - if (statusCode < 400) { - return done(ExportResult.SUCCESS); - // Consider 4xx as failed non-retriable. - } else if (statusCode < 500) { - return done(ExportResult.FAILED_NOT_RETRYABLE); - // Consider 5xx as failed retriable. - } else { - return done(ExportResult.FAILED_RETRYABLE); - } - }); - }); - - req.on('error', (err: Error) => { - this._logger.error('Zipkin request error', err); - return done(ExportResult.FAILED_RETRYABLE); - }); - - // Issue request to remote service - const payload = JSON.stringify(zipkinSpans); - this._logger.debug('Zipkin request payload: %s', payload); - req.write(payload, 'utf8'); - req.end(); - } } diff --git a/packages/opentelemetry-exporter-zipkin/test/browser/index-webpack.ts b/packages/opentelemetry-exporter-zipkin/test/browser/index-webpack.ts new file mode 100644 index 00000000000..99100a0f6ee --- /dev/null +++ b/packages/opentelemetry-exporter-zipkin/test/browser/index-webpack.ts @@ -0,0 +1,23 @@ +/* + * 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. + */ +const testsContext = require.context('../browser', true, /test$/); +testsContext.keys().forEach(testsContext); + +const testsContextCommon = require.context('../common', true, /test$/); +testsContextCommon.keys().forEach(testsContextCommon); + +const srcContext = require.context('.', true, /src$/); +srcContext.keys().forEach(srcContext); diff --git a/packages/opentelemetry-exporter-zipkin/test/browser/zipkin.test.ts b/packages/opentelemetry-exporter-zipkin/test/browser/zipkin.test.ts new file mode 100644 index 00000000000..c47e69fa572 --- /dev/null +++ b/packages/opentelemetry-exporter-zipkin/test/browser/zipkin.test.ts @@ -0,0 +1,157 @@ +/* + * 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 { NoopLogger } from '@opentelemetry/core'; +import { ReadableSpan } from '@opentelemetry/tracing'; +import * as assert from 'assert'; +import * as sinon from 'sinon'; +import { ZipkinExporter } from '../../src'; +import * as zipkinTypes from '../../src/types'; +import { + ensureHeadersContain, + ensureSpanIsCorrect, + mockedReadableSpan, +} from '../helper'; + +const sendBeacon = navigator.sendBeacon; + +describe('Zipkin Exporter - web', () => { + let zipkinExporter: ZipkinExporter; + let zipkinConfig: zipkinTypes.ExporterConfig = {}; + let spySend: sinon.SinonSpy; + let spyBeacon: sinon.SinonSpy; + let spans: ReadableSpan[]; + let sandbox: sinon.SinonSandbox; + + beforeEach(() => { + sandbox = sinon.createSandbox(); + spySend = sandbox.stub(XMLHttpRequest.prototype, 'send'); + spyBeacon = sandbox.stub(navigator, 'sendBeacon'); + spans = []; + spans.push(Object.assign({}, mockedReadableSpan)); + }); + + afterEach(() => { + sandbox.restore(); + navigator.sendBeacon = sendBeacon; + }); + + describe('export', () => { + describe('when "sendBeacon" is available', () => { + beforeEach(() => { + zipkinExporter = new ZipkinExporter(zipkinConfig); + }); + + it('should successfully send the spans using sendBeacon', done => { + zipkinExporter.export(spans, () => {}); + + setTimeout(() => { + const args = spyBeacon.args[0]; + const body = args[1]; + const json = JSON.parse(body) as any; + ensureSpanIsCorrect(json[0]); + assert.strictEqual(spyBeacon.callCount, 1); + assert.strictEqual(spySend.callCount, 0); + + done(); + }); + }); + }); + + describe('when "sendBeacon" is NOT available', () => { + let server: any; + beforeEach(() => { + (window.navigator as any).sendBeacon = false; + zipkinExporter = new ZipkinExporter(zipkinConfig); + server = sinon.fakeServer.create(); + }); + afterEach(() => { + server.restore(); + }); + + it('should successfully send the spans using XMLHttpRequest', done => { + zipkinExporter.export(spans, () => {}); + + setTimeout(() => { + const request = server.requests[0]; + const body = request.requestBody; + const json = JSON.parse(body) as any; + ensureSpanIsCorrect(json[0]); + + done(); + }); + }); + }); + }); + + describe('export with custom headers', () => { + let server: any; + const customHeaders = { + foo: 'bar', + bar: 'baz', + }; + + beforeEach(() => { + zipkinConfig = { + logger: new NoopLogger(), + headers: customHeaders, + }; + server = sinon.fakeServer.create(); + }); + + afterEach(() => { + server.restore(); + }); + + describe('when "sendBeacon" is available', () => { + beforeEach(() => { + zipkinExporter = new ZipkinExporter(zipkinConfig); + }); + it('should successfully send custom headers using XMLHTTPRequest', done => { + zipkinExporter.export(spans, () => {}); + + setTimeout(() => { + const [{ requestHeaders }] = server.requests; + + ensureHeadersContain(requestHeaders, customHeaders); + assert.strictEqual(spyBeacon.callCount, 0); + + done(); + }); + }); + }); + + describe('when "sendBeacon" is NOT available', () => { + beforeEach(() => { + (window.navigator as any).sendBeacon = false; + zipkinExporter = new ZipkinExporter(zipkinConfig); + }); + + it('should successfully send custom headers using XMLHTTPRequest', done => { + zipkinExporter.export(spans, () => {}); + + setTimeout(() => { + const [{ requestHeaders }] = server.requests; + + ensureHeadersContain(requestHeaders, customHeaders); + assert.strictEqual(spyBeacon.callCount, 0); + + done(); + }); + }); + }); + }); +}); diff --git a/packages/opentelemetry-exporter-zipkin/test/transform.test.ts b/packages/opentelemetry-exporter-zipkin/test/common/transform.test.ts similarity index 95% rename from packages/opentelemetry-exporter-zipkin/test/transform.test.ts rename to packages/opentelemetry-exporter-zipkin/test/common/transform.test.ts index 05cbbbfd0d9..ad4472066fa 100644 --- a/packages/opentelemetry-exporter-zipkin/test/transform.test.ts +++ b/packages/opentelemetry-exporter-zipkin/test/common/transform.test.ts @@ -29,14 +29,16 @@ import { _toZipkinAnnotations, statusCodeTagName, statusDescriptionTagName, -} from '../src/transform'; -import * as zipkinTypes from '../src/types'; -import { Resource } from '@opentelemetry/resources'; - +} from '../../src/transform'; +import * as zipkinTypes from '../../src/types'; +import { Resource, TELEMETRY_SDK_RESOURCE } from '@opentelemetry/resources'; const logger = new NoopLogger(); const tracer = new BasicTracerProvider({ logger, }).getTracer('default'); + +const language = tracer.resource.attributes[TELEMETRY_SDK_RESOURCE.LANGUAGE]; + const parentId = '5c1c63257de34c67'; const spanContext: api.SpanContext = { traceId: 'd4cda95b652f4a1592b449d5929fda1b', @@ -94,7 +96,7 @@ describe('transform', () => { key1: 'value1', key2: 'value2', [statusCodeTagName]: 'OK', - 'telemetry.sdk.language': 'nodejs', + 'telemetry.sdk.language': language, 'telemetry.sdk.name': 'opentelemetry', 'telemetry.sdk.version': VERSION, }, @@ -131,7 +133,7 @@ describe('transform', () => { parentId: undefined, tags: { [statusCodeTagName]: 'OK', - 'telemetry.sdk.language': 'nodejs', + 'telemetry.sdk.language': language, 'telemetry.sdk.name': 'opentelemetry', 'telemetry.sdk.version': VERSION, }, @@ -173,7 +175,7 @@ describe('transform', () => { parentId: undefined, tags: { [statusCodeTagName]: 'OK', - 'telemetry.sdk.language': 'nodejs', + 'telemetry.sdk.language': language, 'telemetry.sdk.name': 'opentelemetry', 'telemetry.sdk.version': VERSION, }, diff --git a/packages/opentelemetry-exporter-zipkin/test/helper.ts b/packages/opentelemetry-exporter-zipkin/test/helper.ts new file mode 100644 index 00000000000..8fa1cfe1e4b --- /dev/null +++ b/packages/opentelemetry-exporter-zipkin/test/helper.ts @@ -0,0 +1,78 @@ +/* + * 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 { TraceFlags } from '@opentelemetry/api'; +import { ReadableSpan } from '@opentelemetry/tracing'; +import { Resource } from '@opentelemetry/resources'; +import * as assert from 'assert'; +import { Span } from '../src/types'; + +export const mockedReadableSpan: ReadableSpan = { + name: 'documentFetch', + kind: 0, + spanContext: { + traceId: '1f1008dc8e270e85c40a0d7c3939b278', + spanId: '5e107261f64fa53e', + traceFlags: TraceFlags.SAMPLED, + }, + parentSpanId: '78a8915098864388', + startTime: [1574120165, 429803070], + endTime: [1574120165, 438688070], + ended: true, + status: { code: 0 }, + attributes: { component: 'foo' }, + links: [], + events: [], + duration: [0, 8885000], + resource: new Resource({ + service: 'ui', + version: 1, + cost: 112.12, + }), + instrumentationLibrary: { name: 'default', version: '0.0.1' }, +}; + +export function ensureHeadersContain( + actual: { [key: string]: string }, + expected: { [key: string]: string } +) { + Object.entries(expected).forEach(([k, v]) => { + assert.strictEqual( + v, + actual[k], + `Expected ${actual} to contain ${k}: ${v}` + ); + }); +} + +export function ensureSpanIsCorrect(span: Span) { + assert.deepStrictEqual(span, { + traceId: '1f1008dc8e270e85c40a0d7c3939b278', + parentId: '78a8915098864388', + name: 'documentFetch', + id: '5e107261f64fa53e', + timestamp: 1574120165429803, + duration: 8885, + localEndpoint: { serviceName: 'OpenTelemetry Service' }, + tags: { + component: 'foo', + 'ot.status_code': 'OK', + service: 'ui', + version: 1, + cost: 112.12, + }, + }); +} diff --git a/packages/opentelemetry-exporter-zipkin/test/zipkin.test.ts b/packages/opentelemetry-exporter-zipkin/test/node/zipkin.test.ts similarity index 95% rename from packages/opentelemetry-exporter-zipkin/test/zipkin.test.ts rename to packages/opentelemetry-exporter-zipkin/test/node/zipkin.test.ts index 7dd50b83818..803f88838d2 100644 --- a/packages/opentelemetry-exporter-zipkin/test/zipkin.test.ts +++ b/packages/opentelemetry-exporter-zipkin/test/node/zipkin.test.ts @@ -24,9 +24,9 @@ import { } from '@opentelemetry/core'; import * as api from '@opentelemetry/api'; import { Resource } from '@opentelemetry/resources'; -import { ZipkinExporter } from '../src'; -import * as zipkinTypes from '../src/types'; -import { OT_REQUEST_HEADER } from '../src/utils'; +import { ZipkinExporter } from '../../src'; +import * as zipkinTypes from '../../src/types'; +import { OT_REQUEST_HEADER } from '../../src/utils'; import { TraceFlags } from '@opentelemetry/api'; import { SERVICE_RESOURCE } from '@opentelemetry/resources'; @@ -59,7 +59,7 @@ function getReadableSpan() { return readableSpan; } -describe('ZipkinExporter', () => { +describe('Zipkin Exporter - node', () => { describe('constructor', () => { it('should construct an exporter', () => { const exporter = new ZipkinExporter({ serviceName: 'my-service' }); @@ -98,21 +98,6 @@ describe('ZipkinExporter', () => { assert.ok(typeof exporter.export === 'function'); assert.ok(typeof exporter.shutdown === 'function'); }); - it('should construct an exporter with headers', () => { - const exporter = new ZipkinExporter({ - headers: { - foo: 'bar', - }, - }); - interface ExporterWithHeaders { - _reqOpts: { - headers: { [key: string]: string }; - }; - } - const exporterWithHeaders = (exporter as unknown) as ExporterWithHeaders; - - assert.ok(exporterWithHeaders._reqOpts.headers['foo'] === 'bar'); - }); }); describe('export', () => { @@ -510,8 +495,5 @@ describe('ZipkinExporter', () => { after(() => { nock.enableNetConnect(); }); - - // @todo: implement - it('should send by default'); }); });