Next.js website for Rocky Mountain Vending company featuring: - Product catalog with Stripe integration - Service areas and parts pages - Admin dashboard with Clerk authentication - SEO optimized pages with JSON-LD structured data Co-authored-by: Cursor <cursoragent@cursor.com>
326 lines
No EOL
17 KiB
Text
326 lines
No EOL
17 KiB
Text
"use strict";
|
|
/*
|
|
* 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.
|
|
*/
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
exports.RedisInstrumentation = void 0;
|
|
const api_1 = require("@opentelemetry/api");
|
|
const instrumentation_1 = require("@opentelemetry/instrumentation");
|
|
const utils_1 = require("./utils");
|
|
const redis_common_1 = require("@opentelemetry/redis-common");
|
|
/** @knipignore */
|
|
const version_1 = require("./version");
|
|
const semantic_conventions_1 = require("@opentelemetry/semantic-conventions");
|
|
const OTEL_OPEN_SPANS = Symbol('opentelemetry.instrumentation.redis.open_spans');
|
|
const MULTI_COMMAND_OPTIONS = Symbol('opentelemetry.instrumentation.redis.multi_command_options');
|
|
const DEFAULT_CONFIG = {
|
|
requireParentSpan: false,
|
|
};
|
|
class RedisInstrumentation extends instrumentation_1.InstrumentationBase {
|
|
constructor(config = {}) {
|
|
super(version_1.PACKAGE_NAME, version_1.PACKAGE_VERSION, Object.assign(Object.assign({}, DEFAULT_CONFIG), config));
|
|
}
|
|
setConfig(config = {}) {
|
|
super.setConfig(Object.assign(Object.assign({}, DEFAULT_CONFIG), config));
|
|
}
|
|
init() {
|
|
// @node-redis/client is a new package introduced and consumed by 'redis 4.0.x'
|
|
// on redis@4.1.0 it was changed to @redis/client.
|
|
// we will instrument both packages
|
|
return [
|
|
this._getInstrumentationNodeModuleDefinition('@redis/client'),
|
|
this._getInstrumentationNodeModuleDefinition('@node-redis/client'),
|
|
];
|
|
}
|
|
_getInstrumentationNodeModuleDefinition(basePackageName) {
|
|
const commanderModuleFile = new instrumentation_1.InstrumentationNodeModuleFile(`${basePackageName}/dist/lib/commander.js`, ['^1.0.0'], (moduleExports, moduleVersion) => {
|
|
const transformCommandArguments = moduleExports.transformCommandArguments;
|
|
if (!transformCommandArguments) {
|
|
this._diag.error('internal instrumentation error, missing transformCommandArguments function');
|
|
return moduleExports;
|
|
}
|
|
// function name and signature changed in redis 4.1.0 from 'extendWithCommands' to 'attachCommands'
|
|
// the matching internal package names starts with 1.0.x (for redis 4.0.x)
|
|
const functionToPatch = (moduleVersion === null || moduleVersion === void 0 ? void 0 : moduleVersion.startsWith('1.0.'))
|
|
? 'extendWithCommands'
|
|
: 'attachCommands';
|
|
// this is the function that extend a redis client with a list of commands.
|
|
// the function patches the commandExecutor to record a span
|
|
if ((0, instrumentation_1.isWrapped)(moduleExports === null || moduleExports === void 0 ? void 0 : moduleExports[functionToPatch])) {
|
|
this._unwrap(moduleExports, functionToPatch);
|
|
}
|
|
this._wrap(moduleExports, functionToPatch, this._getPatchExtendWithCommands(transformCommandArguments));
|
|
return moduleExports;
|
|
}, (moduleExports) => {
|
|
if ((0, instrumentation_1.isWrapped)(moduleExports === null || moduleExports === void 0 ? void 0 : moduleExports.extendWithCommands)) {
|
|
this._unwrap(moduleExports, 'extendWithCommands');
|
|
}
|
|
if ((0, instrumentation_1.isWrapped)(moduleExports === null || moduleExports === void 0 ? void 0 : moduleExports.attachCommands)) {
|
|
this._unwrap(moduleExports, 'attachCommands');
|
|
}
|
|
});
|
|
const multiCommanderModule = new instrumentation_1.InstrumentationNodeModuleFile(`${basePackageName}/dist/lib/client/multi-command.js`, ['^1.0.0'], (moduleExports) => {
|
|
var _a;
|
|
const redisClientMultiCommandPrototype = (_a = moduleExports === null || moduleExports === void 0 ? void 0 : moduleExports.default) === null || _a === void 0 ? void 0 : _a.prototype;
|
|
if ((0, instrumentation_1.isWrapped)(redisClientMultiCommandPrototype === null || redisClientMultiCommandPrototype === void 0 ? void 0 : redisClientMultiCommandPrototype.exec)) {
|
|
this._unwrap(redisClientMultiCommandPrototype, 'exec');
|
|
}
|
|
this._wrap(redisClientMultiCommandPrototype, 'exec', this._getPatchMultiCommandsExec());
|
|
if ((0, instrumentation_1.isWrapped)(redisClientMultiCommandPrototype === null || redisClientMultiCommandPrototype === void 0 ? void 0 : redisClientMultiCommandPrototype.addCommand)) {
|
|
this._unwrap(redisClientMultiCommandPrototype, 'addCommand');
|
|
}
|
|
this._wrap(redisClientMultiCommandPrototype, 'addCommand', this._getPatchMultiCommandsAddCommand());
|
|
return moduleExports;
|
|
}, (moduleExports) => {
|
|
var _a;
|
|
const redisClientMultiCommandPrototype = (_a = moduleExports === null || moduleExports === void 0 ? void 0 : moduleExports.default) === null || _a === void 0 ? void 0 : _a.prototype;
|
|
if ((0, instrumentation_1.isWrapped)(redisClientMultiCommandPrototype === null || redisClientMultiCommandPrototype === void 0 ? void 0 : redisClientMultiCommandPrototype.exec)) {
|
|
this._unwrap(redisClientMultiCommandPrototype, 'exec');
|
|
}
|
|
if ((0, instrumentation_1.isWrapped)(redisClientMultiCommandPrototype === null || redisClientMultiCommandPrototype === void 0 ? void 0 : redisClientMultiCommandPrototype.addCommand)) {
|
|
this._unwrap(redisClientMultiCommandPrototype, 'addCommand');
|
|
}
|
|
});
|
|
const clientIndexModule = new instrumentation_1.InstrumentationNodeModuleFile(`${basePackageName}/dist/lib/client/index.js`, ['^1.0.0'], (moduleExports) => {
|
|
var _a;
|
|
const redisClientPrototype = (_a = moduleExports === null || moduleExports === void 0 ? void 0 : moduleExports.default) === null || _a === void 0 ? void 0 : _a.prototype;
|
|
// In some @redis/client versions 'multi' is a method. In later
|
|
// versions, as of https://github.com/redis/node-redis/pull/2324,
|
|
// 'MULTI' is a method and 'multi' is a property defined in the
|
|
// constructor that points to 'MULTI', and therefore it will not
|
|
// be defined on the prototype.
|
|
if (redisClientPrototype === null || redisClientPrototype === void 0 ? void 0 : redisClientPrototype.multi) {
|
|
if ((0, instrumentation_1.isWrapped)(redisClientPrototype === null || redisClientPrototype === void 0 ? void 0 : redisClientPrototype.multi)) {
|
|
this._unwrap(redisClientPrototype, 'multi');
|
|
}
|
|
this._wrap(redisClientPrototype, 'multi', this._getPatchRedisClientMulti());
|
|
}
|
|
if (redisClientPrototype === null || redisClientPrototype === void 0 ? void 0 : redisClientPrototype.MULTI) {
|
|
if ((0, instrumentation_1.isWrapped)(redisClientPrototype === null || redisClientPrototype === void 0 ? void 0 : redisClientPrototype.MULTI)) {
|
|
this._unwrap(redisClientPrototype, 'MULTI');
|
|
}
|
|
this._wrap(redisClientPrototype, 'MULTI', this._getPatchRedisClientMulti());
|
|
}
|
|
if ((0, instrumentation_1.isWrapped)(redisClientPrototype === null || redisClientPrototype === void 0 ? void 0 : redisClientPrototype.sendCommand)) {
|
|
this._unwrap(redisClientPrototype, 'sendCommand');
|
|
}
|
|
this._wrap(redisClientPrototype, 'sendCommand', this._getPatchRedisClientSendCommand());
|
|
this._wrap(redisClientPrototype, 'connect', this._getPatchedClientConnect());
|
|
return moduleExports;
|
|
}, (moduleExports) => {
|
|
var _a;
|
|
const redisClientPrototype = (_a = moduleExports === null || moduleExports === void 0 ? void 0 : moduleExports.default) === null || _a === void 0 ? void 0 : _a.prototype;
|
|
if ((0, instrumentation_1.isWrapped)(redisClientPrototype === null || redisClientPrototype === void 0 ? void 0 : redisClientPrototype.multi)) {
|
|
this._unwrap(redisClientPrototype, 'multi');
|
|
}
|
|
if ((0, instrumentation_1.isWrapped)(redisClientPrototype === null || redisClientPrototype === void 0 ? void 0 : redisClientPrototype.MULTI)) {
|
|
this._unwrap(redisClientPrototype, 'MULTI');
|
|
}
|
|
if ((0, instrumentation_1.isWrapped)(redisClientPrototype === null || redisClientPrototype === void 0 ? void 0 : redisClientPrototype.sendCommand)) {
|
|
this._unwrap(redisClientPrototype, 'sendCommand');
|
|
}
|
|
});
|
|
return new instrumentation_1.InstrumentationNodeModuleDefinition(basePackageName, ['^1.0.0'], (moduleExports) => {
|
|
return moduleExports;
|
|
}, () => { }, [commanderModuleFile, multiCommanderModule, clientIndexModule]);
|
|
}
|
|
// serves both for redis 4.0.x where function name is extendWithCommands
|
|
// and redis ^4.1.0 where function name is attachCommands
|
|
_getPatchExtendWithCommands(transformCommandArguments) {
|
|
const plugin = this;
|
|
return function extendWithCommandsPatchWrapper(original) {
|
|
return function extendWithCommandsPatch(config) {
|
|
var _a;
|
|
if (((_a = config === null || config === void 0 ? void 0 : config.BaseClass) === null || _a === void 0 ? void 0 : _a.name) !== 'RedisClient') {
|
|
return original.apply(this, arguments);
|
|
}
|
|
const origExecutor = config.executor;
|
|
config.executor = function (command, args) {
|
|
const redisCommandArguments = transformCommandArguments(command, args).args;
|
|
return plugin._traceClientCommand(origExecutor, this, arguments, redisCommandArguments);
|
|
};
|
|
return original.apply(this, arguments);
|
|
};
|
|
};
|
|
}
|
|
_getPatchMultiCommandsExec() {
|
|
const plugin = this;
|
|
return function execPatchWrapper(original) {
|
|
return function execPatch() {
|
|
const execRes = original.apply(this, arguments);
|
|
if (typeof (execRes === null || execRes === void 0 ? void 0 : execRes.then) !== 'function') {
|
|
plugin._diag.error('got non promise result when patching RedisClientMultiCommand.exec');
|
|
return execRes;
|
|
}
|
|
return execRes
|
|
.then((redisRes) => {
|
|
const openSpans = this[OTEL_OPEN_SPANS];
|
|
plugin._endSpansWithRedisReplies(openSpans, redisRes);
|
|
return redisRes;
|
|
})
|
|
.catch((err) => {
|
|
const openSpans = this[OTEL_OPEN_SPANS];
|
|
if (!openSpans) {
|
|
plugin._diag.error('cannot find open spans to end for redis multi command');
|
|
}
|
|
else {
|
|
const replies = err.constructor.name === 'MultiErrorReply'
|
|
? err.replies
|
|
: new Array(openSpans.length).fill(err);
|
|
plugin._endSpansWithRedisReplies(openSpans, replies);
|
|
}
|
|
return Promise.reject(err);
|
|
});
|
|
};
|
|
};
|
|
}
|
|
_getPatchMultiCommandsAddCommand() {
|
|
const plugin = this;
|
|
return function addCommandWrapper(original) {
|
|
return function addCommandPatch(args) {
|
|
return plugin._traceClientCommand(original, this, arguments, args);
|
|
};
|
|
};
|
|
}
|
|
_getPatchRedisClientMulti() {
|
|
return function multiPatchWrapper(original) {
|
|
return function multiPatch() {
|
|
const multiRes = original.apply(this, arguments);
|
|
multiRes[MULTI_COMMAND_OPTIONS] = this.options;
|
|
return multiRes;
|
|
};
|
|
};
|
|
}
|
|
_getPatchRedisClientSendCommand() {
|
|
const plugin = this;
|
|
return function sendCommandWrapper(original) {
|
|
return function sendCommandPatch(args) {
|
|
return plugin._traceClientCommand(original, this, arguments, args);
|
|
};
|
|
};
|
|
}
|
|
_getPatchedClientConnect() {
|
|
const plugin = this;
|
|
return function connectWrapper(original) {
|
|
return function patchedConnect() {
|
|
const options = this.options;
|
|
const attributes = (0, utils_1.getClientAttributes)(plugin._diag, options);
|
|
const span = plugin.tracer.startSpan(`${RedisInstrumentation.COMPONENT}-connect`, {
|
|
kind: api_1.SpanKind.CLIENT,
|
|
attributes,
|
|
});
|
|
const res = api_1.context.with(api_1.trace.setSpan(api_1.context.active(), span), () => {
|
|
return original.apply(this);
|
|
});
|
|
return res
|
|
.then((result) => {
|
|
span.end();
|
|
return result;
|
|
})
|
|
.catch((error) => {
|
|
span.recordException(error);
|
|
span.setStatus({
|
|
code: api_1.SpanStatusCode.ERROR,
|
|
message: error.message,
|
|
});
|
|
span.end();
|
|
return Promise.reject(error);
|
|
});
|
|
};
|
|
};
|
|
}
|
|
_traceClientCommand(origFunction, origThis, origArguments, redisCommandArguments) {
|
|
const hasNoParentSpan = api_1.trace.getSpan(api_1.context.active()) === undefined;
|
|
if (hasNoParentSpan && this.getConfig().requireParentSpan) {
|
|
return origFunction.apply(origThis, origArguments);
|
|
}
|
|
const clientOptions = origThis.options || origThis[MULTI_COMMAND_OPTIONS];
|
|
const commandName = redisCommandArguments[0]; // types also allows it to be a Buffer, but in practice it only string
|
|
const commandArgs = redisCommandArguments.slice(1);
|
|
const dbStatementSerializer = this.getConfig().dbStatementSerializer || redis_common_1.defaultDbStatementSerializer;
|
|
const attributes = (0, utils_1.getClientAttributes)(this._diag, clientOptions);
|
|
try {
|
|
const dbStatement = dbStatementSerializer(commandName, commandArgs);
|
|
if (dbStatement != null) {
|
|
attributes[semantic_conventions_1.SEMATTRS_DB_STATEMENT] = dbStatement;
|
|
}
|
|
}
|
|
catch (e) {
|
|
this._diag.error('dbStatementSerializer throw an exception', e, {
|
|
commandName,
|
|
});
|
|
}
|
|
const span = this.tracer.startSpan(`${RedisInstrumentation.COMPONENT}-${commandName}`, {
|
|
kind: api_1.SpanKind.CLIENT,
|
|
attributes,
|
|
});
|
|
const res = api_1.context.with(api_1.trace.setSpan(api_1.context.active(), span), () => {
|
|
return origFunction.apply(origThis, origArguments);
|
|
});
|
|
if (typeof (res === null || res === void 0 ? void 0 : res.then) === 'function') {
|
|
res.then((redisRes) => {
|
|
this._endSpanWithResponse(span, commandName, commandArgs, redisRes, undefined);
|
|
}, (err) => {
|
|
this._endSpanWithResponse(span, commandName, commandArgs, null, err);
|
|
});
|
|
}
|
|
else {
|
|
const redisClientMultiCommand = res;
|
|
redisClientMultiCommand[OTEL_OPEN_SPANS] =
|
|
redisClientMultiCommand[OTEL_OPEN_SPANS] || [];
|
|
redisClientMultiCommand[OTEL_OPEN_SPANS].push({
|
|
span,
|
|
commandName,
|
|
commandArgs,
|
|
});
|
|
}
|
|
return res;
|
|
}
|
|
_endSpansWithRedisReplies(openSpans, replies) {
|
|
if (!openSpans) {
|
|
return this._diag.error('cannot find open spans to end for redis multi command');
|
|
}
|
|
if (replies.length !== openSpans.length) {
|
|
return this._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 = replies[i];
|
|
const [res, err] = currCommandRes instanceof Error
|
|
? [null, currCommandRes]
|
|
: [currCommandRes, undefined];
|
|
this._endSpanWithResponse(span, commandName, commandArgs, res, err);
|
|
}
|
|
}
|
|
_endSpanWithResponse(span, commandName, commandArgs, response, error) {
|
|
const { responseHook } = this.getConfig();
|
|
if (!error && responseHook) {
|
|
try {
|
|
responseHook(span, commandName, commandArgs, response);
|
|
}
|
|
catch (err) {
|
|
this._diag.error('responseHook throw an exception', err);
|
|
}
|
|
}
|
|
if (error) {
|
|
span.recordException(error);
|
|
span.setStatus({ code: api_1.SpanStatusCode.ERROR, message: error === null || error === void 0 ? void 0 : error.message });
|
|
}
|
|
span.end();
|
|
}
|
|
}
|
|
exports.RedisInstrumentation = RedisInstrumentation;
|
|
RedisInstrumentation.COMPONENT = 'redis';
|
|
//# sourceMappingURL=instrumentation.js.map |