Skip to content

Commit

Permalink
fix(instrumentation-redis-4): fix unhandledRejection in client.multi(…
Browse files Browse the repository at this point in the history
…) handling

The instrumentation was not handling a rejection of the promise from
client.multi(), resulting in unended spans and an unhandleRejection
event.

Fixes: #1708
  • Loading branch information
trentm committed Oct 12, 2023
1 parent a8c225d commit 3a7c993
Show file tree
Hide file tree
Showing 2 changed files with 69 additions and 29 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -278,41 +278,61 @@ export class RedisInstrumentation extends InstrumentationBase<any> {
return execRes;
}

execRes.then((redisRes: unknown[]) => {
const openSpans = this[OTEL_OPEN_SPANS];
if (!openSpans) {
return plugin._diag.error(
'cannot find open spans to end for redis multi command'
);
}
if (redisRes.length !== openSpans.length) {
return plugin._diag.error(
'number of multi command spans does not match response from redis'
);
}
for (let i = 0; i < openSpans.length; i++) {
const { span, commandName, commandArgs } = openSpans[i];
const currCommandRes = redisRes[i];
if (currCommandRes instanceof Error) {
plugin._endSpanWithResponse(
span,
commandName,
commandArgs,
null,
currCommandRes
return execRes
.then((redisRes: unknown[]) => {
const openSpans = this[OTEL_OPEN_SPANS];
if (!openSpans) {
return plugin._diag.error(

Check warning on line 285 in plugins/node/opentelemetry-instrumentation-redis-4/src/instrumentation.ts

View check run for this annotation

Codecov / codecov/patch

plugins/node/opentelemetry-instrumentation-redis-4/src/instrumentation.ts#L285

Added line #L285 was not covered by tests
'cannot find open spans to end for redis multi command'
);
}
if (redisRes.length !== openSpans.length) {
return plugin._diag.error(

Check warning on line 290 in plugins/node/opentelemetry-instrumentation-redis-4/src/instrumentation.ts

View check run for this annotation

Codecov / codecov/patch

plugins/node/opentelemetry-instrumentation-redis-4/src/instrumentation.ts#L290

Added line #L290 was not covered by tests
'number of multi command spans does not match response from redis'
);
} else {
}
for (let i = 0; i < openSpans.length; i++) {
const { span, commandName, commandArgs } = openSpans[i];
const currCommandRes = redisRes[i];
if (currCommandRes instanceof Error) {
plugin._endSpanWithResponse(
span,
commandName,
commandArgs,
null,
currCommandRes
);
} else {
plugin._endSpanWithResponse(
span,
commandName,
commandArgs,
currCommandRes,
undefined
);
}
}
return redisRes;
})
.catch((err: Error) => {
const openSpans = this[OTEL_OPEN_SPANS];
if (!openSpans) {
return plugin._diag.error(

Check warning on line 320 in plugins/node/opentelemetry-instrumentation-redis-4/src/instrumentation.ts

View check run for this annotation

Codecov / codecov/patch

plugins/node/opentelemetry-instrumentation-redis-4/src/instrumentation.ts#L320

Added line #L320 was not covered by tests
'cannot find open spans to end for redis multi command'
);
}
for (let i = 0; i < openSpans.length; i++) {
const { span, commandName, commandArgs } = openSpans[i];
plugin._endSpanWithResponse(
span,
commandName,
commandArgs,
currCommandRes,
undefined
null,
err
);
}
}
});
return execRes;
return Promise.reject(err);
});
};
};
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ const instrumentation = registerInstrumentationTesting(
new RedisInstrumentation()
);

import { createClient } from 'redis';
import { createClient, WatchError } from 'redis';
import {
Span,
SpanKind,
Expand Down Expand Up @@ -398,6 +398,26 @@ describe('redis@^4.0.0', () => {
);
});

it('multi command that rejects', async () => {
const watchedKey = 'watched-key';
await client.watch(watchedKey);
await client.set(watchedKey, 'a different value');
try {
await client.multi().get(watchedKey).exec();
assert.fail('expected WatchError to be thrown and caught in try/catch');
} catch (error) {
assert.ok(error instanceof WatchError);
}

// All the multi spans' status are set to ERROR.
const [_watchSpan, _setSpan, multiGetSpan] = getTestSpans();
assert.strictEqual(multiGetSpan?.status.code, SpanStatusCode.ERROR);
assert.strictEqual(
multiGetSpan?.status.message,
'One (or more) of the watched keys has been changed'
);
});

it('duration covers create until server response', async () => {
await client.set('another-key', 'another-value');
const multiClient = client.multi();
Expand Down

0 comments on commit 3a7c993

Please sign in to comment.