Rocky_Mountain_Vending/.pnpm-store/v10/files/35/d6da7f7414ff26a303fdc1544d77471cfd82f97dc6154e0485b66d1f05d8f0dbb5eb509c4e25d54ff11d43f78fae3beaaeda866a48f3a60bac37a8d3578266
DMleadgen 46d973904b
Initial commit: Rocky Mountain Vending website
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>
2026-02-12 16:22:15 -07:00

270 lines
No EOL
12 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.ExpressInstrumentation = void 0;
const core_1 = require("@opentelemetry/core");
const api_1 = require("@opentelemetry/api");
const ExpressLayerType_1 = require("./enums/ExpressLayerType");
const AttributeNames_1 = require("./enums/AttributeNames");
const utils_1 = require("./utils");
/** @knipignore */
const version_1 = require("./version");
const instrumentation_1 = require("@opentelemetry/instrumentation");
const semantic_conventions_1 = require("@opentelemetry/semantic-conventions");
const internal_types_1 = require("./internal-types");
/** Express instrumentation for OpenTelemetry */
class ExpressInstrumentation extends instrumentation_1.InstrumentationBase {
constructor(config = {}) {
super(version_1.PACKAGE_NAME, version_1.PACKAGE_VERSION, config);
}
init() {
return [
new instrumentation_1.InstrumentationNodeModuleDefinition('express', ['>=4.0.0 <5'], moduleExports => {
const routerProto = moduleExports.Router;
// patch express.Router.route
if ((0, instrumentation_1.isWrapped)(routerProto.route)) {
this._unwrap(routerProto, 'route');
}
this._wrap(routerProto, 'route', this._getRoutePatch());
// patch express.Router.use
if ((0, instrumentation_1.isWrapped)(routerProto.use)) {
this._unwrap(routerProto, 'use');
}
// eslint-disable-next-line @typescript-eslint/no-explicit-any
this._wrap(routerProto, 'use', this._getRouterUsePatch());
// patch express.Application.use
if ((0, instrumentation_1.isWrapped)(moduleExports.application.use)) {
this._unwrap(moduleExports.application, 'use');
}
this._wrap(moduleExports.application, 'use',
// eslint-disable-next-line @typescript-eslint/no-explicit-any
this._getAppUsePatch());
return moduleExports;
}, moduleExports => {
if (moduleExports === undefined)
return;
const routerProto = moduleExports.Router;
this._unwrap(routerProto, 'route');
this._unwrap(routerProto, 'use');
this._unwrap(moduleExports.application, 'use');
}),
];
}
/**
* Get the patch for Router.route function
*/
_getRoutePatch() {
const instrumentation = this;
return function (original) {
return function route_trace(...args) {
const route = original.apply(this, args);
const layer = this.stack[this.stack.length - 1];
instrumentation._applyPatch(layer, (0, utils_1.getLayerPath)(args));
return route;
};
};
}
/**
* Get the patch for Router.use function
*/
_getRouterUsePatch() {
const instrumentation = this;
return function (original) {
return function use(...args) {
const route = original.apply(this, args);
const layer = this.stack[this.stack.length - 1];
instrumentation._applyPatch(layer, (0, utils_1.getLayerPath)(args));
return route;
};
};
}
/**
* Get the patch for Application.use function
*/
_getAppUsePatch() {
const instrumentation = this;
return function (original) {
return function use(...args) {
const route = original.apply(this, args);
const layer = this._router.stack[this._router.stack.length - 1];
instrumentation._applyPatch(layer, (0, utils_1.getLayerPath)(args));
return route;
};
};
}
/** Patch each express layer to create span and propagate context */
_applyPatch(layer, layerPath) {
const instrumentation = this;
// avoid patching multiple times the same layer
if (layer[internal_types_1.kLayerPatched] === true)
return;
layer[internal_types_1.kLayerPatched] = true;
this._wrap(layer, 'handle', original => {
// TODO: instrument error handlers
if (original.length === 4)
return original;
const patched = function (req, res) {
(0, utils_1.storeLayerPath)(req, layerPath);
const route = req[internal_types_1._LAYERS_STORE_PROPERTY]
.filter(path => path !== '/' && path !== '/*')
.join('')
// remove duplicate slashes to normalize route
.replace(/\/{2,}/g, '/');
const attributes = {
[semantic_conventions_1.SEMATTRS_HTTP_ROUTE]: route.length > 0 ? route : '/',
};
const metadata = (0, utils_1.getLayerMetadata)(route, layer, layerPath);
const type = metadata.attributes[AttributeNames_1.AttributeNames.EXPRESS_TYPE];
const rpcMetadata = (0, core_1.getRPCMetadata)(api_1.context.active());
if ((rpcMetadata === null || rpcMetadata === void 0 ? void 0 : rpcMetadata.type) === core_1.RPCType.HTTP) {
rpcMetadata.route = route || '/';
}
// verify against the config if the layer should be ignored
if ((0, utils_1.isLayerIgnored)(metadata.name, type, instrumentation.getConfig())) {
if (type === ExpressLayerType_1.ExpressLayerType.MIDDLEWARE) {
req[internal_types_1._LAYERS_STORE_PROPERTY].pop();
}
return original.apply(this, arguments);
}
if (api_1.trace.getSpan(api_1.context.active()) === undefined) {
return original.apply(this, arguments);
}
const spanName = instrumentation._getSpanName({
request: req,
layerType: type,
route,
}, metadata.name);
const span = instrumentation.tracer.startSpan(spanName, {
attributes: Object.assign(attributes, metadata.attributes),
});
const { requestHook } = instrumentation.getConfig();
if (requestHook) {
(0, instrumentation_1.safeExecuteInTheMiddle)(() => requestHook(span, {
request: req,
layerType: type,
route,
}), e => {
if (e) {
api_1.diag.error('express instrumentation: request hook failed', e);
}
}, true);
}
let spanHasEnded = false;
if (metadata.attributes[AttributeNames_1.AttributeNames.EXPRESS_TYPE] !==
ExpressLayerType_1.ExpressLayerType.MIDDLEWARE) {
span.end();
spanHasEnded = true;
}
// listener for response.on('finish')
const onResponseFinish = () => {
if (spanHasEnded === false) {
spanHasEnded = true;
span.end();
}
};
// verify we have a callback
const args = Array.from(arguments);
const callbackIdx = args.findIndex(arg => typeof arg === 'function');
if (callbackIdx >= 0) {
arguments[callbackIdx] = function () {
var _a;
// express considers anything but an empty value, "route" or "router"
// passed to its callback to be an error
const maybeError = arguments[0];
const isError = ![undefined, null, 'route', 'router'].includes(maybeError);
if (!spanHasEnded && isError) {
const [error, message] = (0, utils_1.asErrorAndMessage)(maybeError);
span.recordException(error);
span.setStatus({
code: api_1.SpanStatusCode.ERROR,
message,
});
}
if (spanHasEnded === false) {
spanHasEnded = true;
(_a = req.res) === null || _a === void 0 ? void 0 : _a.removeListener('finish', onResponseFinish);
span.end();
}
if (!(req.route && isError)) {
req[internal_types_1._LAYERS_STORE_PROPERTY].pop();
}
const callback = args[callbackIdx];
return callback.apply(this, arguments);
};
}
try {
return original.apply(this, arguments);
}
catch (anyError) {
const [error, message] = (0, utils_1.asErrorAndMessage)(anyError);
span.recordException(error);
span.setStatus({
code: api_1.SpanStatusCode.ERROR,
message,
});
throw anyError;
}
finally {
/**
* At this point if the callback wasn't called, that means either the
* layer is asynchronous (so it will call the callback later on) or that
* the layer directly end the http response, so we'll hook into the "finish"
* event to handle the later case.
*/
if (!spanHasEnded) {
res.once('finish', onResponseFinish);
}
}
};
// `handle` isn't just a regular function in some cases. It also contains
// some properties holding metadata and state so we need to proxy them
// through through patched function
// ref: https://github.com/open-telemetry/opentelemetry-js-contrib/issues/1950
// Also some apps/libs do their own patching before OTEL and have these properties
// in the proptotype. So we use a `for...in` loop to get own properties and also
// any enumerable prop in the prototype chain
// ref: https://github.com/open-telemetry/opentelemetry-js-contrib/issues/2271
for (const key in original) {
Object.defineProperty(patched, key, {
get() {
return original[key];
},
set(value) {
original[key] = value;
},
});
}
return patched;
});
}
_getSpanName(info, defaultName) {
var _a;
const { spanNameHook } = this.getConfig();
if (!(spanNameHook instanceof Function)) {
return defaultName;
}
try {
return (_a = spanNameHook(info, defaultName)) !== null && _a !== void 0 ? _a : defaultName;
}
catch (err) {
api_1.diag.error('express instrumentation: error calling span name rewrite hook', err);
return defaultName;
}
}
}
exports.ExpressInstrumentation = ExpressInstrumentation;
//# sourceMappingURL=instrumentation.js.map