@sentry/nextjs
Version:
Official Sentry SDK for Next.js
133 lines (116 loc) • 5.45 kB
JavaScript
Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
const core = require('@sentry/core');
const nextNavigationErrorUtils = require('./nextNavigationErrorUtils.js');
const spanAttributesWithLogicAttached = require('./span-attributes-with-logic-attached.js');
const responseEnd = require('./utils/responseEnd.js');
const tracingUtils = require('./utils/tracingUtils.js');
const urls = require('./utils/urls.js');
/**
* Wraps an `app` directory server component with Sentry error instrumentation.
*/
// eslint-disable-next-line @typescript-eslint/no-explicit-any
function wrapServerComponentWithSentry(
appDirComponent,
context,
) {
const { componentRoute, componentType } = context;
// Even though users may define server components as async functions, for the client bundles
// Next.js will turn them into synchronous functions and it will transform any `await`s into instances of the `use`
// hook. 🤯
return new Proxy(appDirComponent, {
apply: (originalFunction, thisArg, args) => {
const requestTraceId = core.getActiveSpan()?.spanContext().traceId;
const isolationScope = tracingUtils.commonObjectToIsolationScope(context.headers);
let pathname = undefined ;
const activeSpan = core.getActiveSpan();
if (activeSpan) {
const rootSpan = core.getRootSpan(activeSpan);
const { scope } = core.getCapturedScopesOnSpan(rootSpan);
core.setCapturedScopesOnSpan(rootSpan, scope ?? new core.Scope(), isolationScope);
const spanData = core.spanToJSON(rootSpan);
if (spanData.data && 'http.target' in spanData.data) {
pathname = spanData.data['http.target']?.toString();
}
}
const headersDict = context.headers ? core.winterCGHeadersToDict(context.headers) : undefined;
let params = undefined;
if (core.getClient()?.getOptions().sendDefaultPii) {
const props = args[0];
params =
props && typeof props === 'object' && 'params' in props
? (props.params )
: undefined;
}
isolationScope.setSDKProcessingMetadata({
normalizedRequest: {
headers: headersDict,
url: urls.getSanitizedRequestUrl(componentRoute, params, headersDict, pathname),
} ,
});
return core.withIsolationScope(isolationScope, () => {
return core.withScope(scope => {
scope.setTransactionName(`${componentType} Server Component (${componentRoute})`);
if (process.env.NEXT_RUNTIME === 'edge') {
const propagationContext = tracingUtils.commonObjectToPropagationContext(
context.headers,
core.propagationContextFromHeaders(headersDict?.['sentry-trace'], headersDict?.['baggage']),
);
if (requestTraceId) {
propagationContext.traceId = requestTraceId;
}
scope.setPropagationContext(propagationContext);
}
const activeSpan = core.getActiveSpan();
if (activeSpan) {
const rootSpan = core.getRootSpan(activeSpan);
const sentryTrace = headersDict?.['sentry-trace'];
if (sentryTrace) {
rootSpan.setAttribute(spanAttributesWithLogicAttached.TRANSACTION_ATTR_SENTRY_TRACE_BACKFILL, sentryTrace);
}
}
return core.startSpanManual(
{
op: 'function.nextjs',
name: `${componentType} Server Component (${componentRoute})`,
attributes: {
[core.SEMANTIC_ATTRIBUTE_SENTRY_SOURCE]: 'component',
[core.SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN]: 'auto.function.nextjs',
'sentry.nextjs.ssr.function.type': componentType,
'sentry.nextjs.ssr.function.route': componentRoute,
},
},
span => {
return core.handleCallbackErrors(
() => originalFunction.apply(thisArg, args),
error => {
// When you read this code you might think: "Wait a minute, shouldn't we set the status on the root span too?"
// The answer is: "No." - The status of the root span is determined by whatever status code Next.js decides to put on the response.
if (nextNavigationErrorUtils.isNotFoundNavigationError(error)) {
// We don't want to report "not-found"s
span.setStatus({ code: core.SPAN_STATUS_ERROR, message: 'not_found' });
} else if (nextNavigationErrorUtils.isRedirectNavigationError(error)) {
// We don't want to report redirects
span.setStatus({ code: core.SPAN_STATUS_OK });
} else {
span.setStatus({ code: core.SPAN_STATUS_ERROR, message: 'internal_error' });
core.captureException(error, {
mechanism: {
handled: false,
},
});
}
},
() => {
span.end();
core.vercelWaitUntil(responseEnd.flushSafelyWithTimeout());
},
);
},
);
});
});
},
});
}
exports.wrapServerComponentWithSentry = wrapServerComponentWithSentry;
//# sourceMappingURL=wrapServerComponentWithSentry.js.map