Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

[fastify] thousands of requests are grouped under the same trace ID #15534

Closed
punkpeye opened this issue Feb 27, 2025 · 4 comments
Closed

[fastify] thousands of requests are grouped under the same trace ID #15534

punkpeye opened this issue Feb 27, 2025 · 4 comments
Labels
Feature: Tracing Integration: fastify Issues related to Fastify support for the Sentry Node SDK Package: node Issues related to the Sentry Node SDK Stale

Comments

@punkpeye
Copy link

punkpeye commented Feb 27, 2025

Environment

SaaS (https://sentry.io/)

Steps to Reproduce

I have pretty standard setup:

server.ts

const app = fastify({
  bodyLimit: 32 * 1_024 * 1_024,
  logger: config.FASTIFY_LOGGING,
  pluginTimeout: 60_000,
  return503OnClosing: false,
  trustProxy: true,
});

setupFastifyErrorHandler(app);

app.setErrorHandler((error, request, reply) => {
  captureException({
    error,
    extra: {
      request,
    },
    message: 'could not handle request',
  });

  return replyWithErrorPage(reply, error);
});

await app.register(checksFastifyPlugin, {
  getStatus: () => {
    return shutdownHandler.getStatus();
  },
  prefix: '/checks',
});

await app.register(ingestLogsFastifyPlugin);

instrument.ts

const integrations: NodeOptions['integrations'] = [];

if (config.SENTRY_PROFILE) {
  integrations.push(nodeProfilingIntegration());
}

integrations.push(
  Sentry.extraErrorDataIntegration({
    captureErrorCause: true,
    depth: 7,
  }),
);

const SENTRY_SAMPLE_RATE = 0.1;

const sentryOptions = {
  dsn: config.SENTRY_DSN,
  environment: config.ENVIRONMENT,
  integrations,
  maxValueLength: 1_024 * 10,
  normalizeDepth: 7,
  openTelemetryInstrumentations: [new OpenAIInstrumentation()],
  profilesSampleRate: config.SENTRY_PROFILE ? SENTRY_SAMPLE_RATE : 0,
  registerEsmLoaderHooks: config.SENTRY_TRACE,
  release: config.RELEASE_VERSION,
  skipOpenTelemetrySetup: !config.SENTRY_TRACE,
  tracesSampler: () => {
    if (!config.SENTRY_TRACE) {
      return false;
    }

    if (Math.random() > SENTRY_SAMPLE_RATE) {
      return false;
    }

    return true;
  },
} satisfies NodeOptions;

Sentry.init(sentryOptions);

/**
 * The difference between `beforeSend` and `addEventProcessor` is that `beforeSend` is called the last.
 * We have convert Request instance to Sentry request object here or else it will be stringified as {}.
 */
Sentry.addEventProcessor(async (event) => {
  const visitorSession = event.extra?.visitorSession as
    | PublicVisitorSession
    | undefined;

  // https://develop.sentry.dev/sdk/data-model/event-payloads/user/
  if (visitorSession?.userAccount) {
    event.user = {
      ...event.user,
      email: visitorSession.userAccount.emailAddress,
      id: visitorSession.userAccount.uid,
      username: visitorSession.userAccount.fullName,
    };
  }

  if (typeof event.extra?.ipAddress === 'string') {
    event.user = {
      ...event.user,
      ip_address: event.extra.ipAddress,
    };
  }

  if (event.extra?.request instanceof Request) {
    const request = event.extra.request;

    const headers = Object.fromEntries(request.headers.entries());

    delete headers.cookie;

    // https://develop.sentry.dev/sdk/data-model/event-payloads/request/
    event.request = {
      data: normalize(request.body) as unknown,
      headers: { user_agent: headers['user-agent'] },
      method: request.method,
      query_string: new URL(request.url).search.replace('?', ''),
      url: request.url,
    };

    event.transaction = `${request.method} ${request.url}`;
  }

  // TODO research when values would be more than 1
  if (event.exception?.values?.length === 1) {
    const exception = event.exception.values[0];

    // New frames are added to the end of the stacktrace.
    const frames = exception.stacktrace?.frames?.reverse();

    const route = frames
      ?.map((frame) => frame.filename)
      .find((filename) => filename?.includes('/app/routes/'));

    if (route) {
      // https://docs.sentry.io/platforms/javascript/enriching-events/fingerprinting/
      event.fingerprint = ['{{ default }}', route];
    }
  }

  return event;
});

Expected Result

I expect every traceId to be associated with a unique request.

Actual Result

Thousands of different requests are associated with the same traceId.

Here is one example:

Image

Product Area

Issues

Link

No response

DSN

https://fc82aa9f808b13927ff29c56bda4d486@o4507975513735168.ingest.us.sentry.io/4507975514849280

Version

No response

@getsantry
Copy link

getsantry bot commented Feb 27, 2025

Assigning to @getsentry/support for routing ⏲️

@getsantry getsantry bot moved this to Waiting for: Support in GitHub Issues with 👀 3 Feb 27, 2025
@punkpeye
Copy link
Author

@AbhiPrasad as per your request

#4784 (comment)

@AbhiPrasad
Copy link
Member

@punkpeye could you share your Node.js, Fastify, and Sentry SDK versions? Going to transfer this to our JavaScript SDK repo so that we can track it as part of SDK tasks.

@AbhiPrasad AbhiPrasad transferred this issue from getsentry/sentry Feb 27, 2025
@getsantry getsantry bot moved this from Waiting for: Support to Waiting for: Product Owner in GitHub Issues with 👀 3 Feb 27, 2025
@AbhiPrasad AbhiPrasad added Package: node Issues related to the Sentry Node SDK Feature: Tracing Integration: fastify Issues related to Fastify support for the Sentry Node SDK labels Feb 27, 2025
@getsantry getsantry bot moved this from Waiting for: Product Owner to Waiting for: Community in GitHub Issues with 👀 3 Mar 3, 2025
@getsantry
Copy link

getsantry bot commented Mar 25, 2025

This issue has gone three weeks without activity. In another week, I will close it.

But! If you comment or otherwise update it, I will reset the clock, and if you remove the label Waiting for: Community, I will leave it alone ... forever!


"A weed is but an unloved flower." ― Ella Wheeler Wilcox 🥀

@getsantry getsantry bot added the Stale label Mar 25, 2025
@getsantry getsantry bot closed this as not planned Won't fix, can't repro, duplicate, stale Apr 2, 2025
# for free to join this conversation on GitHub. Already have an account? # to comment
Labels
Feature: Tracing Integration: fastify Issues related to Fastify support for the Sentry Node SDK Package: node Issues related to the Sentry Node SDK Stale
Projects
Archived in project
Development

No branches or pull requests

3 participants