Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

feat(browser): Send CLS as standalone span (experimental) #13056

Merged
merged 13 commits into from
Aug 13, 2024

Conversation

Lms24
Copy link
Member

@Lms24 Lms24 commented Jul 25, 2024

This PR adds an experimental feature to browserTracingIntegration to no longer tie CLS reporting to the ongoing pageload span but instead send a standalone CLS span similarly to how we report INP.

The big advantage of this reporting strategy is that layout shifts happening after the pageload idle span ended, will also get reported. This should give users more accurate CLS values in the web vitals performance insights module.

There is a big challenge though we somehow needed to adress: CLS really is cumulative 😅 - the web-vitals library does not stop reporting CLS or reset the score on soft navigations (like very commonly used in SPAs). This is discussed a lot in the web-vitals library repo (GoogleChrome/web-vitals#119) but general consensus seems to be that it is not going to change (for actually good reasons).

However, for our use case, where we specifically link CLS to the page that was initially loaded, it would be quite weird to report a high CLS score where a lot of layout shift only might have happened after a soft navigation.

To address this, we opted for the following, slightly customized, CLS reporting strategy:

  • on page load, we start collecting CLS values
  • on a navigation or the first page hide (whichever comes first), we collect the CLS value and send it off as a standalone span
  • all later CLS emissions from the web-vitals library are ignored (as previously fwiw)

To better illustrate the difference, here's a fancy diagram:

image

So to sum it up:

  • we more accurately report layout shift, especially LS that happens a long time after the page was loaded
  • we still don't take LS into account that happened after a navigation

Span envelope Payload

Our integration tests show the full span envelope item payload:

For ingestion and product, there are still some open questions:

  • is the span envelope payload correct - do we have all the data we need?
  • what about the html node attribution? For now, I added it as a string like we do in INP. I could also add it as a span attribute if that makes more sense. Though I wonder what we do with this data, given we won't show CLS spans in the UI.

ref #13015

Copy link
Contributor

github-actions bot commented Jul 25, 2024

size-limit report 📦

Path Size
@sentry/browser 22.5 KB (added)
@sentry/browser (incl. Tracing) 34.85 KB (added)
@sentry/browser (incl. Tracing, Replay) 71.18 KB (added)
@sentry/browser (incl. Tracing, Replay) - with treeshaking flags 64.45 KB (added)
@sentry/browser (incl. Tracing, Replay with Canvas) 75.53 KB (added)
@sentry/browser (incl. Tracing, Replay, Feedback) 88.17 KB (added)
@sentry/browser (incl. Tracing, Replay, Feedback, metrics) 90 KB (added)
@sentry/browser (incl. metrics) 26.81 KB (added)
@sentry/browser (incl. Feedback) 39.47 KB (added)
@sentry/browser (incl. sendFeedback) 27.13 KB (added)
@sentry/browser (incl. FeedbackAsync) 31.79 KB (added)
@sentry/react 25.26 KB (added)
@sentry/react (incl. Tracing) 37.83 KB (added)
@sentry/vue 26.65 KB (added)
@sentry/vue (incl. Tracing) 36.67 KB (added)
@sentry/svelte 22.64 KB (added)
CDN Bundle 23.73 KB (added)
CDN Bundle (incl. Tracing) 36.49 KB (added)
CDN Bundle (incl. Tracing, Replay) 70.81 KB (added)
CDN Bundle (incl. Tracing, Replay, Feedback) 76.06 KB (added)
CDN Bundle - uncompressed 69.61 KB (added)
CDN Bundle (incl. Tracing) - uncompressed 108.27 KB (added)
CDN Bundle (incl. Tracing, Replay) - uncompressed 219.52 KB (added)
CDN Bundle (incl. Tracing, Replay, Feedback) - uncompressed 232.41 KB (added)
@sentry/nextjs (client) 37.59 KB (added)
@sentry/sveltekit (client) 35.45 KB (added)
@sentry/node 115.61 KB (added)
@sentry/node - without tracing 90 KB (added)
@sentry/aws-serverless 99.42 KB (added)

@@ -44,6 +44,85 @@ export function startAndEndSpan(
});
}

interface StandaloneWebVitalSpanOptions {
Copy link
Member Author

@Lms24 Lms24 Jul 25, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I extracted this from the INP code and only left vital-specific logic in the CLS and INP handlers. Most attributes and span properties are the same for INP and CLS (+soon LCP), so it makes sense to dedupe and save a couple of bytes.

@Lms24
Copy link
Member Author

Lms24 commented Jul 25, 2024

So this definitely collides with Replay because I set reportAllChanges: false. I think for now we need to register two onLCP callbacks, one with reportAllChanges: false and one with reportAllChanges: true.

Update: I changed my approach here to also work with reportAllChanges: true. This allows us to keep the amount of onCLS calls at a minimum. Plus it removes the web-vitals modification I had to apply before.

@Lms24 Lms24 self-assigned this Jul 25, 2024
Comment on lines 27 to 36
let standaloneCLsValue = 0;
let standaloneClsEntry: LayoutShift | undefined;

if (!supportsLayoutShift()) {
return;
}
Copy link
Member Author

@Lms24 Lms24 Jul 26, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Starting with 0 solves the 0 CLS value problem (see #12989). However, since this happens outside the web-vitals library, we need to add a browser support check for CLS to avoid sending 0 values on browsers that don't support collecting CLS.

@Lms24 Lms24 marked this pull request as ready for review July 26, 2024 14:42
@Lms24
Copy link
Member Author

Lms24 commented Jul 26, 2024

This does not block reviews but to follow up:

After syncing with @c298lee we can also switch back to reportAllChanges: false as Replay is fine with getting one CLS emission per navigation or pagehide. We can make that work with re-adding the web-vitals modification to emit on navigation.

But we need to keep reportAllChanges: true for the current stable way of collecting CLS.

Copy link
Contributor

@edwardgou-sentry edwardgou-sentry left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Functionally I think this makes sense to me. Tested this out locally and the payload + behaviour looks correct.

I wasn't able to get webvital scoring to run on these standalone CLS spans, but I think that's because there are probably some relay changes to be made to not ignore these spans (I can take a look at this next week).

I think the cls html node in the description is fine. We probably will need to make some UI changes in the product to surface the description/span in some way.

1 question, should we also be attaching the pageload span id as an attribute? We may want a way to connect the standalone cls span to a pageload in the UI. Maybe there is a better otel mechanism for this?

@Lms24 Lms24 force-pushed the lms/feat-browser-cls-standalone-spans branch 2 times, most recently from 883bb92 to 69adb32 Compare August 13, 2024 08:30
@Lms24 Lms24 force-pushed the lms/feat-browser-cls-standalone-spans branch from 69adb32 to 585a7ac Compare August 13, 2024 08:54
@Lms24 Lms24 merged commit 5af8eb8 into develop Aug 13, 2024
123 checks passed
@Lms24 Lms24 deleted the lms/feat-browser-cls-standalone-spans branch August 13, 2024 11:00
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants