Skip to content

Commit

Permalink
fix(instrumentation-fs): fix instrumentation of fs/promises (#1375)
Browse files Browse the repository at this point in the history
* fix(instrumentation-fs): fix instrumentation of `fs/promises`

* Revert "fix(instrumentation-fs): fix instrumentation of `fs/promises`"

This reverts commit 90d9f0d.

* fix(instrumentation-fs): fix instrumentation of `fs/promises`

* chore(instrumentation-fs): fix lint error

* chore(instrumentation-fs): hard-code `R_OK` value for node 14

* chore(instrumentation-fs): fix supported version for `fs/promises`

* chore(instrumentation-fs): incorporate changes from #1332

* chore(instrumentation-fs): consolidate common test utilities

---------

Co-authored-by: Haddas Bronfman <85441461+haddasbronfman@users.noreply.github.com>
  • Loading branch information
mhassan1 and haddasbronfman authored Feb 27, 2023
1 parent e93a192 commit 3ca874e
Show file tree
Hide file tree
Showing 6 changed files with 386 additions and 60 deletions.
33 changes: 32 additions & 1 deletion plugins/node/instrumentation-fs/src/instrumentation.ts
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ import { promisify } from 'util';
import { indexFs } from './utils';

type FS = typeof fs;
type FSPromises = (typeof fs)['promises'];

/**
* This is important for 2-level functions like `realpath.native` to retain the 2nd-level
Expand All @@ -55,7 +56,10 @@ export default class FsInstrumentation extends InstrumentationBase<FS> {
super('@opentelemetry/instrumentation-fs', VERSION, config);
}

init(): InstrumentationNodeModuleDefinition<FS>[] {
init(): (
| InstrumentationNodeModuleDefinition<FS>
| InstrumentationNodeModuleDefinition<FSPromises>
)[] {
return [
new InstrumentationNodeModuleDefinition<FS>(
'fs',
Expand Down Expand Up @@ -129,6 +133,33 @@ export default class FsInstrumentation extends InstrumentationBase<FS> {
}
}
),
new InstrumentationNodeModuleDefinition<FSPromises>(
'fs/promises',
['*'],
(fsPromises: FSPromises) => {
this._diag.debug('Applying patch for fs/promises');
for (const fName of PROMISE_FUNCTIONS) {
if (isWrapped(fsPromises[fName])) {
this._unwrap(fsPromises, fName);
}
this._wrap(
fsPromises,
fName,
<any>this._patchPromiseFunction.bind(this, fName)
);
}
return fsPromises;
},
(fsPromises: FSPromises) => {
if (fsPromises === undefined) return;
this._diag.debug('Removing patch for fs/promises');
for (const fName of PROMISE_FUNCTIONS) {
if (isWrapped(fsPromises[fName])) {
this._unwrap(fsPromises, fName);
}
}
}
),
];
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,12 +13,11 @@
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { context, trace, SpanStatusCode, SpanKind } from '@opentelemetry/api';
import { context, trace } from '@opentelemetry/api';
import { AsyncHooksContextManager } from '@opentelemetry/context-async-hooks';
import {
BasicTracerProvider,
InMemorySpanExporter,
ReadableSpan,
SimpleSpanProcessor,
} from '@opentelemetry/sdk-trace-base';
import * as assert from 'assert';
Expand All @@ -34,6 +33,7 @@ import {
SYNC_FUNCTIONS,
} from '../src/constants';
import { indexFs, splitTwoLevels } from '../src/utils';
import { assertSpans, makeRootSpanName } from './utils';

const TEST_ATTRIBUTE = 'test.attr';
const TEST_VALUE = 'test.attr.value';
Expand Down Expand Up @@ -149,16 +149,6 @@ describe('fs instrumentation', () => {
]);
});
};
const makeRootSpanName = (name: FMember): string => {
let rsn: string;
if (Array.isArray(name)) {
rsn = `${name[0]}.${name[1]}`;
} else {
rsn = `${name}`;
}
rsn = `${rsn} test span`;
return rsn;
};

const callbackTest: TestCreator<FMember> = (
name: FMember,
Expand Down Expand Up @@ -443,50 +433,3 @@ describe('fs instrumentation', () => {
});
});
});

const assertSpans = (spans: ReadableSpan[], expected: any) => {
assert.strictEqual(
spans.length,
expected.length,
`Expected ${expected.length} spans, got ${spans.length}(${spans
.map((s: any) => `"${s.name}"`)
.join(', ')})`
);

spans.forEach((span, i) => {
assertSpan(span, expected[i]);
});
};

const assertSpan = (span: ReadableSpan, expected: any) => {
assert(span);
assert.strictEqual(span.name, expected.name);
assert.strictEqual(
span.kind,
SpanKind.INTERNAL,
'Expected to be of INTERNAL kind'
);
if (expected.parentSpan) {
assert.strictEqual(
span.parentSpanId,
expected.parentSpan.spanContext().spanId
);
}
if (expected.attributes) {
assert.deepEqual(span.attributes, expected.attributes);
}
if (expected.error) {
assert(
expected.error.test(span.status.message),
`Expected "${span.status.message}" to match ${expected.error}`
);
assert.strictEqual(span.status.code, SpanStatusCode.ERROR);
} else {
assert.strictEqual(
span.status.code,
SpanStatusCode.UNSET,
'Expected status to be unset'
);
assert.strictEqual(span.status.message, undefined);
}
};
153 changes: 153 additions & 0 deletions plugins/node/instrumentation-fs/test/fsPromises.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,153 @@
/*
* 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 { context, trace } from '@opentelemetry/api';
import { AsyncHooksContextManager } from '@opentelemetry/context-async-hooks';
import {
BasicTracerProvider,
InMemorySpanExporter,
SimpleSpanProcessor,
} from '@opentelemetry/sdk-trace-base';
import * as assert from 'assert';
import Instrumentation from '../src';
import * as sinon from 'sinon';
import type * as FSPromisesType from 'fs/promises';
import tests, { FsFunction, TestCase, TestCreator } from './definitions';
import type { FPMember, EndHook } from '../src/types';
import { assertSpans, makeRootSpanName } from './utils';

const supportsPromises =
parseInt(process.versions.node.split('.')[0], 10) >= 14;

const TEST_ATTRIBUTE = 'test.attr';
const TEST_VALUE = 'test.attr.value';

const endHook = <EndHook>sinon.spy((fnName, { args, span }) => {
span.setAttribute(TEST_ATTRIBUTE, TEST_VALUE);
});
const pluginConfig = {
endHook,
};
const provider = new BasicTracerProvider();
const tracer = provider.getTracer('default');
const memoryExporter = new InMemorySpanExporter();
provider.addSpanProcessor(new SimpleSpanProcessor(memoryExporter));

if (supportsPromises) {
describe('fs/promises instrumentation', () => {
let contextManager: AsyncHooksContextManager;
let fsPromises: typeof FSPromisesType;
let plugin: Instrumentation;

beforeEach(async () => {
contextManager = new AsyncHooksContextManager();
context.setGlobalContextManager(contextManager.enable());
plugin = new Instrumentation(pluginConfig);
plugin.setTracerProvider(provider);
plugin.enable();
fsPromises = require('fs/promises');
assert.strictEqual(memoryExporter.getFinishedSpans().length, 0);
});

afterEach(() => {
plugin.disable();
memoryExporter.reset();
context.disable();
});

const promiseTest: TestCreator<FPMember> = (
name: FPMember,
args,
{ error, result, resultAsError = null, hasPromiseVersion = true },
spans
) => {
if (!hasPromiseVersion) return;
const rootSpanName = makeRootSpanName(name);
it(`promises.${name} ${error ? 'error' : 'success'}`, async () => {
const rootSpan = tracer.startSpan(rootSpanName);

assert.strictEqual(memoryExporter.getFinishedSpans().length, 0);
await context
.with(trace.setSpan(context.active(), rootSpan), () => {
// eslint-disable-next-line node/no-unsupported-features/node-builtins
assert(
typeof fsPromises[name] === 'function',
`Expected fsPromises.${name} to be a function`
);
return Reflect.apply(fsPromises[name], fsPromises, args);
})
.then((actualResult: any) => {
if (error) {
assert.fail(`promises.${name} did not reject`);
} else {
assert.deepEqual(actualResult, result ?? resultAsError);
}
})
.catch((actualError: any) => {
assert(
actualError instanceof Error,
`Expected caugth error to be instance of Error. Got ${actualError}`
);
if (error) {
assert(
error.test(actualError?.message ?? ''),
`Expected "${actualError?.message}" to match ${error}`
);
} else {
actualError.message = `Did not expect promises.${name} to reject: ${actualError.message}`;
assert.fail(actualError);
}
});
rootSpan.end();
assertSpans(memoryExporter.getFinishedSpans(), [
...spans.map((s: any) => {
const spanName = s.name.replace(/%NAME/, name);
const attributes = {
...(s.attributes ?? {}),
};
attributes[TEST_ATTRIBUTE] = TEST_VALUE;
return {
...s,
name: spanName,
attributes,
};
}),
{ name: rootSpanName },
]);
});
};

const selection: TestCase[] = tests.filter(
([name, , , , options = {}]) =>
options.promise !== false && name !== ('exists' as FsFunction)
);

describe('Instrumentation enabled', () => {
selection.forEach(([name, args, result, spans]) => {
promiseTest(name as FPMember, args, result, spans);
});
});

describe('Instrumentation disabled', () => {
beforeEach(() => {
plugin.disable();
});

selection.forEach(([name, args, result]) => {
promiseTest(name as FPMember, args, result, []);
});
});
});
}
Loading

0 comments on commit 3ca874e

Please sign in to comment.