Skip to content

Commit

Permalink
feat(instrumentation): implement require-in-the-middle singleton
Browse files Browse the repository at this point in the history
  • Loading branch information
mhassan1 committed Sep 10, 2022
1 parent 32cb123 commit 285439f
Show file tree
Hide file tree
Showing 4 changed files with 239 additions and 17 deletions.
1 change: 1 addition & 0 deletions experimental/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ All notable changes to experimental packages in this project will be documented

* Add `resourceDetectors` option to `NodeSDK` [#3210](https://github.com/open-telemetry/opentelemetry-js/issues/3210)
* feat: add Logs API @mkuba [#3117](https://github.com/open-telemetry/opentelemetry-js/pull/3117)
* feat(instrumentation): implement `require-in-the-middle` singleton [#3161](https://github.com/open-telemetry/opentelemetry-js/pull/3161) @mhassan1

### :bug: (Bug Fix)

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,9 +16,9 @@

import * as types from '../../types';
import * as path from 'path';
import * as RequireInTheMiddle from 'require-in-the-middle';
import { satisfies } from 'semver';
import { InstrumentationAbstract } from '../../instrumentation';
import { requireInTheMiddleSingleton } from './requireInTheMiddleSingleton';
import { InstrumentationModuleDefinition } from './types';
import { diag } from '@opentelemetry/api';

Expand All @@ -29,7 +29,7 @@ export abstract class InstrumentationBase<T = any>
extends InstrumentationAbstract
implements types.Instrumentation {
private _modules: InstrumentationModuleDefinition<T>[];
private _hooks: RequireInTheMiddle.Hooked[] = [];
private _hooked = false;
private _enabled = false;

constructor(
Expand Down Expand Up @@ -142,7 +142,7 @@ export abstract class InstrumentationBase<T = any>
this._enabled = true;

// already hooked, just call patch again
if (this._hooks.length > 0) {
if (this._hooked) {
for (const module of this._modules) {
if (typeof module.patch === 'function' && module.moduleExports) {
module.patch(module.moduleExports, module.moduleVersion);
Expand All @@ -158,22 +158,20 @@ export abstract class InstrumentationBase<T = any>

this._warnOnPreloadedModules();
for (const module of this._modules) {
this._hooks.push(
RequireInTheMiddle(
[module.name],
{ internals: true },
(exports, name, baseDir) => {
return this._onRequire<typeof exports>(
(module as unknown) as InstrumentationModuleDefinition<
typeof exports
requireInTheMiddleSingleton.register(
module.name,
(exports, name, baseDir) => {
return this._onRequire<typeof exports>(
(module as unknown) as InstrumentationModuleDefinition<
typeof exports
>,
exports,
name,
baseDir
);
}
)
exports,
name,
baseDir
);
}
);
this._hooked = true;
}
}

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,88 @@
/*
* 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 RequireInTheMiddle from 'require-in-the-middle';
import * as path from 'path';

const RITM_SINGLETON_SYM = Symbol.for('OpenTelemetry.js.sdk.require-in-the-middle');

/**
* Singleton class for `require-in-the-middle`
* Allows instrumentation plugins to patch modules with only a single `require` patch
*/
class RequireInTheMiddleSingleton {
private _modulesToHook: Array<{ moduleName: string, onRequire: RequireInTheMiddle.OnRequireFn }> = [];

constructor() {
this._initialize();
}

private _initialize() {
RequireInTheMiddle(
// Intercept all `require` calls; we will filter the matching ones below
null,
{ internals: true },
(exports, name, basedir) => {
const matches = this._modulesToHook.filter(({ moduleName: hookedModuleName }) => {
return shouldHook(hookedModuleName, name);
});

for (const { onRequire } of matches) {
exports = onRequire(exports, name, basedir);
}

return exports;
}
);
}

register(moduleName: string, onRequire: RequireInTheMiddle.OnRequireFn) {
this._modulesToHook.push({ moduleName, onRequire });
}

static getGlobalInstance(): RequireInTheMiddleSingleton {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
return (global as any)[RITM_SINGLETON_SYM] = (global as any)[RITM_SINGLETON_SYM] ?? new RequireInTheMiddleSingleton();
}
}

/**
* Determine whether a `require`d module should be hooked
*
* @param {string} hookedModuleName Hooked module name
* @param {string} requiredModuleName Required module name
* @returns {boolean} Whether to hook the required module
* @private
*/
export function shouldHook(hookedModuleName: string, requiredModuleName: string): boolean {
const normalizedRequiredModuleName = normalizePathSeparators(requiredModuleName);

return normalizedRequiredModuleName === hookedModuleName || normalizedRequiredModuleName.startsWith(hookedModuleName + '/');
}

/**
* Normalize the path separators to forward slash in a module name or path
*
* @param {string} moduleNameOrPath Module name or path
* @returns {string} Normalized module name or path
*/
function normalizePathSeparators(moduleNameOrPath: string): string {
return path.sep !== '/'
? moduleNameOrPath.split(path.sep).join('/')
: moduleNameOrPath;
}

export const requireInTheMiddleSingleton = RequireInTheMiddleSingleton.getGlobalInstance();
Original file line number Diff line number Diff line change
@@ -0,0 +1,135 @@
/*
* 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 assert from 'assert';
import * as sinon from 'sinon';
import * as path from 'path';
import * as RequireInTheMiddle from 'require-in-the-middle';
import { requireInTheMiddleSingleton, shouldHook } from '../../src/platform/node/requireInTheMiddleSingleton';

type AugmentedExports = {
__ritmOnRequires?: string[]
};

const makeOnRequiresStub = (label: string): sinon.SinonStub => sinon.stub().callsFake(((exports: AugmentedExports) => {
exports.__ritmOnRequires ??= [];
exports.__ritmOnRequires.push(label);
return exports;
}) as RequireInTheMiddle.OnRequireFn);

describe('requireInTheMiddleSingleton', () => {
describe('register', () => {
const onRequireTimersStub = makeOnRequiresStub('timers');
const onRequireTimersPromisesStub = makeOnRequiresStub('timers-promises');
const onRequireCodecovStub = makeOnRequiresStub('codecov');
const onRequireCodecovLibStub = makeOnRequiresStub('codecov-lib');
const onRequireCpxStub = makeOnRequiresStub('cpx');
const onRequireCpxLibStub = makeOnRequiresStub('cpx-lib');

before(() => {
requireInTheMiddleSingleton.register('timers', onRequireTimersStub);
requireInTheMiddleSingleton.register('timers/promises', onRequireTimersPromisesStub);
requireInTheMiddleSingleton.register('codecov', onRequireCodecovStub);
requireInTheMiddleSingleton.register('codecov/lib/codecov.js', onRequireCodecovLibStub);
requireInTheMiddleSingleton.register('cpx', onRequireCpxStub);
requireInTheMiddleSingleton.register('cpx/lib/copy-sync.js', onRequireCpxLibStub);
});

beforeEach(() => {
onRequireTimersStub.resetHistory();
onRequireTimersPromisesStub.resetHistory();
onRequireCodecovStub.resetHistory();
onRequireCodecovLibStub.resetHistory();
onRequireCpxStub.resetHistory();
onRequireCpxLibStub.resetHistory();
});

describe('core module', () => {
describe('AND module name matches', () => {
it('should call `onRequire`', () => {
const exports = require('timers');
assert.deepEqual(exports.__ritmOnRequires, ['timers']);
sinon.assert.calledOnceWithExactly(onRequireTimersStub, exports, 'timers', undefined);
sinon.assert.notCalled(onRequireTimersPromisesStub);
});
});
describe('AND module name does not match', () => {
it('should not call `onRequire`', () => {
const exports = require('crypto');
assert.equal(exports.__ritmOnRequires, undefined);
sinon.assert.notCalled(onRequireTimersStub);
});
});
});

describe('core module with sub-path', () => {
describe('AND module name matches', () => {
it('should call `onRequire`', () => {
const exports = require('timers/promises');
assert.deepEqual(exports.__ritmOnRequires, ['timers', 'timers-promises']);
sinon.assert.calledOnceWithExactly(onRequireTimersPromisesStub, exports, 'timers/promises', undefined);
sinon.assert.calledOnceWithMatch(onRequireTimersStub, { __ritmOnRequires: ['timers', 'timers-promises'] }, 'timers/promises', undefined);
});
});
});

describe('non-core module', () => {
describe('AND module name matches', () => {
const baseDir = path.dirname(require.resolve('codecov'));
const modulePath = path.join('codecov', 'lib', 'codecov.js');
it('should call `onRequire`', () => {
const exports = require('codecov');
assert.deepEqual(exports.__ritmOnRequires, ['codecov']);
sinon.assert.calledWithExactly(onRequireCodecovStub, exports, 'codecov', baseDir);
sinon.assert.calledWithMatch(onRequireCodecovStub, { __ritmOnRequires: ['codecov', 'codecov-lib'] }, modulePath, baseDir);
sinon.assert.calledWithMatch(onRequireCodecovLibStub, { __ritmOnRequires: ['codecov', 'codecov-lib'] }, modulePath, baseDir);
});
});
});

describe('non-core module with sub-path', () => {
describe('AND module name matches', () => {
const baseDir = path.resolve(path.dirname(require.resolve('cpx')), '..');
const modulePath = path.join('cpx', 'lib', 'copy-sync.js');
it('should call `onRequire`', () => {
const exports = require('cpx/lib/copy-sync');
assert.deepEqual(exports.__ritmOnRequires, ['cpx', 'cpx-lib']);
sinon.assert.calledWithMatch(onRequireCpxStub, { __ritmOnRequires: ['cpx', 'cpx-lib'] }, modulePath, baseDir);
sinon.assert.calledWithExactly(onRequireCpxStub, exports, modulePath, baseDir);
sinon.assert.calledWithExactly(onRequireCpxLibStub, exports, modulePath, baseDir);
});
});
});
});

describe('shouldHook', () => {
describe('module that matches', () => {
it('should be hooked', () => {
assert.equal(shouldHook('c', 'c'), true);
assert.equal(shouldHook('c', path.join('c', 'd')), true);
assert.equal(shouldHook('c', path.posix.join('c', 'd')), true);
assert.equal(shouldHook('c.js', 'c.js'), true);
});
});
describe('module that does not match', () => {
it('should not be hooked', () => {
assert.equal(shouldHook('c', 'c.js'), false);
assert.equal(shouldHook('c', 'e'), false);
assert.equal(shouldHook('c.js', 'c'), false);
});
});
});
});

0 comments on commit 285439f

Please sign in to comment.