From 69a1c7bfbd36ab0d5a6f5146412207db882fed24 Mon Sep 17 00:00:00 2001 From: Krystian Kruk Date: Mon, 14 Jun 2021 22:14:24 +0200 Subject: [PATCH 1/2] fix(instrumentation-user-interaction): support clicks in React apps --- .../src/instrumentation.ts | 31 +++-- .../src/types.ts | 2 +- .../test/helper.test.ts | 4 +- .../test/userInteraction.nozone.test.ts | 129 +++++++++++++++++- .../test/userInteraction.test.ts | 122 +++++++++++++++++ 5 files changed, 268 insertions(+), 20 deletions(-) diff --git a/plugins/web/opentelemetry-instrumentation-user-interaction/src/instrumentation.ts b/plugins/web/opentelemetry-instrumentation-user-interaction/src/instrumentation.ts index c9b3f5393e..0a5ef04153 100644 --- a/plugins/web/opentelemetry-instrumentation-user-interaction/src/instrumentation.ts +++ b/plugins/web/opentelemetry-instrumentation-user-interaction/src/instrumentation.ts @@ -254,7 +254,7 @@ export class UserInteractionInstrumentation extends InstrumentationBase * auto instrument the click events * This is done when zone is not available */ - private _patchElement() { + private _patchAddEventListener() { const plugin = this; return (original: Function) => { return function addEventListenerPatched( @@ -265,16 +265,19 @@ export class UserInteractionInstrumentation extends InstrumentationBase ) { const once = useCapture && useCapture.once; const patchedListener = (...args: any[]) => { - const target = this; let parentSpan: api.Span | undefined; const event: Event | undefined = args[0]; + const target = event?.target; if (event) { parentSpan = plugin._eventsSpanMap.get(event); } if (once) { plugin.removePatchedListener(this, type, listener); } - const span = plugin._createSpan(target, type, parentSpan); + const span = + target instanceof HTMLElement + ? plugin._createSpan(target, type, parentSpan) + : undefined; if (span) { if (event) { plugin._eventsSpanMap.set(event, span); @@ -439,10 +442,14 @@ export class UserInteractionInstrumentation extends InstrumentationBase applyThis?: any, applyArgs?: any ): Zone { - const target: HTMLElement | undefined = task.target; + const event = + Array.isArray(applyArgs) && applyArgs[0] instanceof Event + ? applyArgs[0] + : undefined; + const target = event?.target; let span: api.Span | undefined; const activeZone = this; - if (target) { + if (target instanceof HTMLElement) { span = plugin._createSpan(target, task.eventName); if (span) { plugin._incrementTask(span); @@ -564,23 +571,23 @@ export class UserInteractionInstrumentation extends InstrumentationBase ); } else { this._zonePatched = false; - if (isWrapped(HTMLElement.prototype.addEventListener)) { - this._unwrap(HTMLElement.prototype, 'addEventListener'); + if (isWrapped(EventTarget.prototype.addEventListener)) { + this._unwrap(EventTarget.prototype, 'addEventListener'); api.diag.debug('removing previous patch from method addEventListener'); } - if (isWrapped(HTMLElement.prototype.removeEventListener)) { - this._unwrap(HTMLElement.prototype, 'removeEventListener'); + if (isWrapped(EventTarget.prototype.removeEventListener)) { + this._unwrap(EventTarget.prototype, 'removeEventListener'); api.diag.debug( 'removing previous patch from method removeEventListener' ); } this._wrap( - HTMLElement.prototype, + EventTarget.prototype, 'addEventListener', - this._patchElement() + this._patchAddEventListener() ); this._wrap( - HTMLElement.prototype, + EventTarget.prototype, 'removeEventListener', this._patchRemoveEventListener() ); diff --git a/plugins/web/opentelemetry-instrumentation-user-interaction/src/types.ts b/plugins/web/opentelemetry-instrumentation-user-interaction/src/types.ts index a400810dd0..1de8cd3db7 100644 --- a/plugins/web/opentelemetry-instrumentation-user-interaction/src/types.ts +++ b/plugins/web/opentelemetry-instrumentation-user-interaction/src/types.ts @@ -21,7 +21,7 @@ import * as types from '@opentelemetry/api'; */ export type AsyncTask = Task & { eventName: string; - target: HTMLElement; + target: EventTarget; // Allows access to the private `_zone` property of a Zone.js Task. _zone: Zone; }; diff --git a/plugins/web/opentelemetry-instrumentation-user-interaction/test/helper.test.ts b/plugins/web/opentelemetry-instrumentation-user-interaction/test/helper.test.ts index 385b9d0ecf..863e9c9e7d 100644 --- a/plugins/web/opentelemetry-instrumentation-user-interaction/test/helper.test.ts +++ b/plugins/web/opentelemetry-instrumentation-user-interaction/test/helper.test.ts @@ -36,10 +36,8 @@ export function createButton(disabled?: boolean): HTMLElement { export function fakeInteraction( callback: Function = function () {}, - elem?: HTMLElement + element: HTMLElement = createButton() ) { - const element: HTMLElement = elem || createButton(); - element.addEventListener('click', () => { callback(); }); diff --git a/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.nozone.test.ts b/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.nozone.test.ts index dfba789dd9..0a48d71e71 100644 --- a/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.nozone.test.ts +++ b/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.nozone.test.ts @@ -337,10 +337,15 @@ describe('UserInteractionInstrumentation', () => { callCount++; }; document.body.addEventListener('click', listener1); - document.body.firstElementChild?.addEventListener('click', listener2); - document.body.firstElementChild?.dispatchEvent( - new MouseEvent('click', { bubbles: true }) - ); + try { + document.body.firstElementChild?.addEventListener('click', listener2); + document.body.firstElementChild?.dispatchEvent( + new MouseEvent('click', { bubbles: true }) + ); + } finally { + // remove added listener so we don't pollute other tests + document.body.removeEventListener('click', listener1); + } assert.strictEqual(callCount, 2); assert.strictEqual(exportSpy.args.length, 2); assert.strictEqual( @@ -410,6 +415,122 @@ describe('UserInteractionInstrumentation', () => { }); }); + it('should handle interactions listened on document - react < 17', done => { + const btn1 = document.createElement('button'); + btn1.setAttribute('id', 'btn1'); + document.body.appendChild(btn1); + const btn2 = document.createElement('button'); + btn2.setAttribute('id', 'btn2'); + document.body.appendChild(btn2); + + const listener = (event: MouseEvent) => { + switch (event.target) { + case btn1: + getData(FILE_URL, () => { + sandbox.clock.tick(10); + }).then(() => {}); + break; + case btn2: + getData(FILE_URL, () => { + sandbox.clock.tick(10); + }).then(() => {}); + break; + } + }; + + document.addEventListener('click', listener); + + try { + btn1.click(); + btn2.click(); + } finally { + // remove added listener so we don't pollute other tests + document.removeEventListener('click', listener); + } + + sandbox.clock.tick(1000); + originalSetTimeout(() => { + assert.equal(exportSpy.args.length, 4, 'should export 4 spans'); + + const span1: tracing.ReadableSpan = exportSpy.args[0][0][0]; + const span2: tracing.ReadableSpan = exportSpy.args[1][0][0]; + const span3: tracing.ReadableSpan = exportSpy.args[2][0][0]; + const span4: tracing.ReadableSpan = exportSpy.args[3][0][0]; + + assertClickSpan(span1, 'btn1'); + assertClickSpan(span2, 'btn2'); + + assert.strictEqual( + span1.spanContext().spanId, + span3.parentSpanId, + 'span3 has wrong parent' + ); + assert.strictEqual( + span2.spanContext().spanId, + span4.parentSpanId, + 'span4 has wrong parent' + ); + + done(); + }); + }); + + it('should handle interactions listened on a parent element (bubbled events) - react >= 17', done => { + const root = document.createElement('div'); + document.body.appendChild(root); + + const btn1 = document.createElement('button'); + btn1.setAttribute('id', 'btn1'); + root.appendChild(btn1); + const btn2 = document.createElement('button'); + btn2.setAttribute('id', 'btn2'); + root.appendChild(btn2); + + root.addEventListener('click', event => { + switch (event.target) { + case btn1: + getData(FILE_URL, () => { + sandbox.clock.tick(10); + }).then(() => {}); + break; + case btn2: + getData(FILE_URL, () => { + sandbox.clock.tick(10); + }).then(() => {}); + break; + } + }); + + btn1.click(); + btn2.click(); + + sandbox.clock.tick(1000); + originalSetTimeout(() => { + assert.equal(exportSpy.args.length, 4, 'should export 4 spans'); + + const span1: tracing.ReadableSpan = exportSpy.args[0][0][0]; + const span2: tracing.ReadableSpan = exportSpy.args[1][0][0]; + const span3: tracing.ReadableSpan = exportSpy.args[2][0][0]; + const span4: tracing.ReadableSpan = exportSpy.args[3][0][0]; + + assertClickSpan(span1, 'btn1'); + assertClickSpan(span2, 'btn2'); + + assert.strictEqual( + span1.spanContext().spanId, + span3.parentSpanId, + 'span3 has wrong parent' + ); + assert.strictEqual( + span2.spanContext().spanId, + span4.parentSpanId, + 'span4 has wrong parent' + ); + + done(); + }); + }); + it('should handle disable', () => { assert.strictEqual( isWrapped(HTMLElement.prototype.addEventListener), diff --git a/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.test.ts b/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.test.ts index bcdf9efd89..3e38df8600 100644 --- a/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.test.ts +++ b/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.test.ts @@ -40,6 +40,12 @@ const FILE_URL = 'https://raw.githubusercontent.com/open-telemetry/opentelemetry-js/main/package.json'; describe('UserInteractionInstrumentation', () => { + afterEach(() => { + while (document.body.lastChild) { + document.body.removeChild(document.body.lastChild); + } + }); + describe('when zone.js is available', () => { let contextManager: ZoneContextManager; let userInteractionInstrumentation: UserInteractionInstrumentation; @@ -313,6 +319,122 @@ describe('UserInteractionInstrumentation', () => { }); }); + it('should handle interactions listened on document - react < 17', done => { + const btn1 = document.createElement('button'); + btn1.setAttribute('id', 'btn1'); + document.body.appendChild(btn1); + const btn2 = document.createElement('button'); + btn2.setAttribute('id', 'btn2'); + document.body.appendChild(btn2); + + const listener = (event: MouseEvent) => { + switch (event.target) { + case btn1: + getData(FILE_URL, () => { + sandbox.clock.tick(10); + }).then(() => {}); + break; + case btn2: + getData(FILE_URL, () => { + sandbox.clock.tick(10); + }).then(() => {}); + break; + } + }; + + document.addEventListener('click', listener); + + try { + btn1.click(); + btn2.click(); + } finally { + // remove added listener so we don't pollute other tests + document.removeEventListener('click', listener); + } + + sandbox.clock.tick(1000); + originalSetTimeout(() => { + assert.equal(exportSpy.args.length, 4, 'should export 4 spans'); + + const span1: tracing.ReadableSpan = exportSpy.args[0][0][0]; + const span2: tracing.ReadableSpan = exportSpy.args[1][0][0]; + const span3: tracing.ReadableSpan = exportSpy.args[2][0][0]; + const span4: tracing.ReadableSpan = exportSpy.args[3][0][0]; + + assertClickSpan(span1, 'btn1'); + assertClickSpan(span2, 'btn2'); + + assert.strictEqual( + span1.spanContext().spanId, + span3.parentSpanId, + 'span3 has wrong parent' + ); + assert.strictEqual( + span2.spanContext().spanId, + span4.parentSpanId, + 'span4 has wrong parent' + ); + + done(); + }); + }); + + it('should handle interactions listened on a parent element (bubbled events) - react >= 17', done => { + const root = document.createElement('div'); + document.body.appendChild(root); + + const btn1 = document.createElement('button'); + btn1.setAttribute('id', 'btn1'); + root.appendChild(btn1); + const btn2 = document.createElement('button'); + btn2.setAttribute('id', 'btn2'); + root.appendChild(btn2); + + root.addEventListener('click', event => { + switch (event.target) { + case btn1: + getData(FILE_URL, () => { + sandbox.clock.tick(10); + }).then(() => {}); + break; + case btn2: + getData(FILE_URL, () => { + sandbox.clock.tick(10); + }).then(() => {}); + break; + } + }); + + btn1.click(); + btn2.click(); + + sandbox.clock.tick(1000); + originalSetTimeout(() => { + assert.equal(exportSpy.args.length, 4, 'should export 4 spans'); + + const span1: tracing.ReadableSpan = exportSpy.args[0][0][0]; + const span2: tracing.ReadableSpan = exportSpy.args[1][0][0]; + const span3: tracing.ReadableSpan = exportSpy.args[2][0][0]; + const span4: tracing.ReadableSpan = exportSpy.args[3][0][0]; + + assertClickSpan(span1, 'btn1'); + assertClickSpan(span2, 'btn2'); + + assert.strictEqual( + span1.spanContext().spanId, + span3.parentSpanId, + 'span3 has wrong parent' + ); + assert.strictEqual( + span2.spanContext().spanId, + span4.parentSpanId, + 'span4 has wrong parent' + ); + + done(); + }); + }); + it('should handle unpatch', () => { const _window: WindowWithZone = window as unknown as WindowWithZone; const ZoneWithPrototype = _window.Zone; From 7ee1d07390ba292d76b558317e13cc89116ac56e Mon Sep 17 00:00:00 2001 From: Krystian Kruk Date: Wed, 16 Jun 2021 16:45:12 +0200 Subject: [PATCH 2/2] chore: code maintenance --- .../src/instrumentation.ts | 12 ++++++------ .../test/userInteraction.test.ts | 1 + 2 files changed, 7 insertions(+), 6 deletions(-) diff --git a/plugins/web/opentelemetry-instrumentation-user-interaction/src/instrumentation.ts b/plugins/web/opentelemetry-instrumentation-user-interaction/src/instrumentation.ts index 0a5ef04153..2f5663f36c 100644 --- a/plugins/web/opentelemetry-instrumentation-user-interaction/src/instrumentation.ts +++ b/plugins/web/opentelemetry-instrumentation-user-interaction/src/instrumentation.ts @@ -98,10 +98,13 @@ export class UserInteractionInstrumentation extends InstrumentationBase * @param eventName */ private _createSpan( - element: HTMLElement, + element: EventTarget | null | undefined, eventName: string, parentSpan?: api.Span | undefined ): api.Span | undefined { + if (!(element instanceof HTMLElement)) { + return undefined; + } if (!element.getAttribute) { return undefined; } @@ -274,10 +277,7 @@ export class UserInteractionInstrumentation extends InstrumentationBase if (once) { plugin.removePatchedListener(this, type, listener); } - const span = - target instanceof HTMLElement - ? plugin._createSpan(target, type, parentSpan) - : undefined; + const span = plugin._createSpan(target, type, parentSpan); if (span) { if (event) { plugin._eventsSpanMap.set(event, span); @@ -449,7 +449,7 @@ export class UserInteractionInstrumentation extends InstrumentationBase const target = event?.target; let span: api.Span | undefined; const activeZone = this; - if (target instanceof HTMLElement) { + if (target) { span = plugin._createSpan(target, task.eventName); if (span) { plugin._incrementTask(span); diff --git a/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.test.ts b/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.test.ts index 3e38df8600..cf83d3045d 100644 --- a/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.test.ts +++ b/plugins/web/opentelemetry-instrumentation-user-interaction/test/userInteraction.test.ts @@ -41,6 +41,7 @@ const FILE_URL = describe('UserInteractionInstrumentation', () => { afterEach(() => { + // clear body from elements created by some tests to make sure they are independent while (document.body.lastChild) { document.body.removeChild(document.body.lastChild); }