Opened 8 weeks ago
Last modified 6 days ago
#61734 accepted enhancement
Add the ability to handle "fetchpriority" to ES Modules and Import Maps
Reported by: | dennysdionigi | Owned by: | westonruter |
---|---|---|---|
Milestone: | 6.7 | Priority: | normal |
Severity: | minor | Version: | 6.5 |
Component: | Script Loader | Keywords: | needs-patch 2nd-opinion |
Focuses: | javascript, performance | Cc: |
Description
This is a follow-up to #56313.
Following this ticket, seems that adding fetchpriority to module scripts gives no luck.
So any imported script has high priority, even when they should have low fetching prioritize.
I guess that should be a plus, to allow that choise / filter also for modules and importmaps, while actually loader_tag , script_add_data, not even replace or HTML API seem having effect.
I guess that giving a low priority hint, sparingly and with proper controls, can bring an improvement in overall performance.
Reference documentation: https://developer.mozilla.org/en-US/docs/Web/API/HTMLScriptElement/fetchPriority
Change History (10)
This ticket was mentioned in Slack in #core-performance by mukeshpanchal27. View the logs.
7 weeks ago
#2
@
7 weeks ago
- Keywords needs-patch added
- Milestone changed from Awaiting Review to 6.7
- Owner set to westonruter
- Status changed from new to accepted
- Version changed from 6.6.1 to 6.5
This ticket was mentioned in Slack in #core-performance by westonruter. View the logs.
7 weeks ago
#4
follow-up:
↓ 5
@
6 weeks ago
I'm having a hard time reproducing the results I saw above. Now I'm seeing the addition of fetchpriority=low
to be hurting LCP when the LCP element is a large Image block.
#5
in reply to:
↑ 4
@
6 weeks ago
Replying to westonruter:
I'm having a hard time reproducing the results I saw above. Now I'm seeing the addition of
fetchpriority=low
to be hurting LCP when the LCP element is a large Image block. This doesn't make sense to me.
First LCP image should have always fetchpriority high, the perf hint combo:
- Heavy non essential resources: imgs, videos, links, scripts (example heavy animation scripts) low priority
- imgs, iframe and videos can/should be further optimized by lazy-loading, decoding async - something WP already someway does...
- Heavy essential resources (like lcp image), hypothetically also fonts, and primary static content, can get high priority hint, with additionally extra links prefetch/preconnect/preload - or newest speculation api for scripts.
- fonts than can have an additionally optimisation trick: using a font-display optional (with no swap timing, with no-blocking timing), and a link rel preload to that font, gives a really huge performance boost, fixing some loading issues; but this is more like a "pro-hack"; font-display fallback, usually is enough. 😅
Otherwise adding wrong priority, or adding it to everything can lead the opposite results, as you said for the lcp image...
#6
@
6 weeks ago
@dennysdionigi yes, I'm well familiar, which is why I'm confused why I'm not seeing that loading the modules with low priority is not improving LCP on a page that has an image with fetchpriority=high
. I'm sure I saw an improvement when I tested before. We need to put together a test page that shows consistent performance improvements.
This ticket was mentioned in Slack in #core-performance by westonruter. View the logs.
4 weeks ago
This ticket was mentioned in Slack in #core-performance by mukeshpanchal27. View the logs.
3 weeks ago
#9
@
3 weeks ago
- Keywords 2nd-opinion added
This ticket was discussed in today's performance bug scrub. Someone needs to verify Weston's findings https://core.trac.wordpress.org/ticket/61734#comment:2 before any implementation.
Props to @joemcgill.
I do indeed see that on a page with the Interactivity API present for the Image block and the Navigation block, that these scripts are all loaded with high priority in Chrome:
/wp-includes/blocks/navigation/view.js
/wp-includes/blocks/image/view.js
/wp-includes/js/dist/interactivity.js
The HTML in question:
This doesn't seem ideal, considering that server-side rendering should be employed by interactive blocks, meaning that the scripting should not be in the critical path for rendering the page. By loading these scripts with high priority, they may be hurting LCP by adding network contention for fetching the LCP image.
For adding
fetchpriority
for a script module, either this could be added to thescript
tag itself or it could be added as alink[rel=modulepreload]
tag that the samesrc
.For traditional scripts and module scripts alike, you can use the
wp_script_attributes
filter to inject thefetchpriority
attribute onto thescript
tag. For example, to mark all module scripts to load withfetchpriority=low
:However, this does not help when a script module is not directly enqueued in which case a
link[rel=modulepreload]
is added viaWP_Script_Modules::print_script_module_preloads()
which normally occurs for the@wordpress/interactivity
script. And there is no way to insertfetchpriority=low
into the generatedlink
tag without some hacking:I put these snippets in a plugin you can use for testing as well: https://gist.github.com/westonruter/471111a891f43e0f48bc7e0ca478623d
Given a test post using the TT4 theme that contains one lightbox-enabled Image block which is the LCP element, where the responsive Navigation block is also present:
I tried benchmarking page loads with the existing behavior where all scripts are loaded with high priority, and over 100 requests the median LCP-TTFB was 48.15 ms.
Then I tried adding
fetchpriority=low
to the scripts, and over 100 requests the median LCP-TTFB was 44.6 ms.So adding
fetchpriority=low
makes the LCP metric here 7.37% faster.(The TTFB-LCP metric here is the LCP discounting variations in TTFB.)
I think the Interactivity API should add
fetchpriority=low
to all of the module scripts it registers.