-
Notifications
You must be signed in to change notification settings - Fork 426
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
LoAF missed for some INP events #485
Comments
Update: we discovered that touch emulation in DevTools on desktop requires some extra IPC hops between Renderer and Browser, and that this can lead to events being processed in a different LoAF frame than if it were real input processing (essentially: the browser adds an extra yield() point which then often schedules after-next-paint). We wanted to check if that could be related to this issue -- did you have a chance? |
Re-reading the initial description, I thought I should point out:
(After the feedback we got from the folks working on v4 + attribution, we are working towards cleaning up the event timing stream to remove the out-of-order reporting complexity) |
I was finally able to reproduce your test page by NOT turning on 4x throttling (and now its fairly reliable). I did it many times with 4x throttling enabled and kept getting the other case you mentioned, with 2x events each with a 1 LoAF. (I also at least once was a single event entry). My machine is pretty slow already so I guess too much throttling is as bad as none at all ? ;) |
🎉 |
I thought about this a bit, and I think that's not "report all changes" anymore (as you say). One way to handle this would be to not report until after all entries in a given |
Hmmm… not entirely sure about that. For a start it’s not "report all changes" anyway, it’s “report all metric changes”. I.e. it doesn’t report candidates that are smaller than the current metric (a point of confusion often raised). To me this isn’t a change either since it’s reporting the same value. So it’s kinda pointless and confusing. I’ve seen this issue with web-vitals extension and been meaning to figure it out. I presumed it was an issue with the extension. So ideally we wouldn’t do this. Maybe by noting when reporting a value that’s already been reported (and losing the “change”). Or by cloning the metric at observation time if it’s liable to change. Them again I think |
My understanding of the situation is that there IS a metric change, but by the time the attribution callback is run (since it's behind an idle callback) the values on the respective metric objects have updated and now appear to be the same when they're reported. My comment above was meant to say that if there actually is a metric value change, then (to be consistent with how this work for other metrics) we should fire two callbacks, even if they're part of the same interaction. My hope is that we can eventually remove the idle callback once Michal and team implement the buffering behavior he describes, and so then this would no longer be an issue. |
could be related: #491 |
Another option would be to move the |
I ran across this issue while playing with the demo for the Measuring INP codelab. After trying to find a minimal repro, it still depends on a number of very specific circumstances, but
so it still seems worth fixing.
tl;dr
LoAFs can sometimes arrive from the PerfObserver before the interaction events they contain. If a
requestIdleCallback
fires between the PerfObserver callbacks, a LoAF can be discarded and won't be able to be paired with the interaction events when they come in later.From out-of-band discussion, a reasonable fix would be to not discard any LoAFs that have a
startTime
after any existing event'sprocessingEnd
(orstartTime
+duration
?). I'll open a PR to address this.This turned out to be relatively simple, but I spent a decent amount of time debugging because it didn't appear to be simple at first, so I'm going to write down way too much about it for posterity. Feel free to skip the rest :)
To reproduce
The demo does two things:
:hover
style (since this is emulated mobile) and the hamburger SVG stroke starts a CSS color transitionclick
event triggers a 400ms busy task in arequestAnimationFrame
callbackThings that are necessary but I don't totally understand:
touch-action: manipulation
(originally included because I had copy/pasted some Adam Argyle styles into the page). I thought that was just useful for preventing DTZ, but the page also has a proper mobile viewport set so that shouldn't be a factor regardless. Does that property interact with mobile:hover
in some way? Speaking to @mmocny, setting this may affect the timing/order of event dispatch, so it may just be a matter of PerfObserver timing, not a difference in behaviorrequestIdleCallback
underlyingwhenIdle
triggering LoAF cleanup, but it's unclear if it just makes it easier to repro and isn't necessary, or if there's other task ordering/triggering that could have the same effect without DevTools.Behavior
Usually two INP events will be logged to console (
reportAllChanges
is set totrue
). On my machine:Sometimes there will be two identical INPs logged.
What's actually happening is two different INPs are reported from the unattributed
onINP
, but thewhenIdle
on the callback in the attributedonINP
is causing the first report to not be made (and logged to console) until after themetric
has been updated for the second report callback. Since it's the same object being updated and passed to the two callbacks, the same values are logged twice. In this case the attribution will have a LoAF, though, so everything is good (I don't know if it's worth doing anything about the slightly confusing apparent double logging...maybe debounce whenreportAllChanges
is true? Abort the idle task when a new report comes in? Or is that not "report all changes" anymore?)Much more often, a short INP is logged with just a
pointerdown
event inentries
, then a little later a longer INP comes in with the ~400msclick
event. Neither of these INP values will have an attributed LoAF attached.I can very reliably reproduce the second case, which is why I noticed this in the first place.
Looking into what's happening, it appears that
pointerdown
comes inLoAF
comes in that starts after thepointerdown
queueCleanup
'swhenIdle
fires. The LoAF doesn't overlap with thepointerdown
, so is discardedclick
event comes in, which would have overlapped with theLoAF
LoAF
for attribution now,longAnimationFrameEntries
is left empty.Example run of bug
This can be difficult to debug because recording via the performance panel and even console logging seem to affect when the
whenIdle
callbacks fire. I ended up using a global array to push log messages to and then logged everything at once at the end.Here's a trace with the different PerformanceObserver callbacks added as performance marks and the LoAF as a measure:
https://trace.cafe/t/Q3usrDo9C7
Console output:
new events (at 1930.9):
'pointerdown'
new LoAFs (at 2354.3):
LoAFs discarded (at 2356.9):
web-vitals: INP of 24ms
new events (at 2371.8):
'mouseover'
'mousedown'
'mouseup'
'click'
web-vitals: INP of 448ms
The text was updated successfully, but these errors were encountered: