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

Astro v5 ClientRouter duplicating scripts when deployed #12804

Open
1 task
Boston343 opened this issue Dec 21, 2024 · 13 comments
Open
1 task

Astro v5 ClientRouter duplicating scripts when deployed #12804

Boston343 opened this issue Dec 21, 2024 · 13 comments
Labels
- P2: has workaround Bug, but has workaround (priority)

Comments

@Boston343
Copy link

Boston343 commented Dec 21, 2024

Astro Info

Astro                    v5.1.1
Node                     v20.18.1
System                   Windows (x64)
Package Manager          pnpm
Output                   static
Adapter                  none
Integrations             @astrojs/tailwind

If this issue only occurs in one browser, which browser is a problem?

All

Describe the Bug

In preview and deployment, ClientRouter adds additional copies of any scripts on page when navigating to a new page. This causes issues with scripts on page. Things work as expected in dev mode.

You can see this issue by running pnpm preview with this demo repo, and navigating between the "Home" and "Other Page" with the dev tools Console open to see console logs. Repo here for demo: Client Router Issue Demo

What's the expected result?

Navigating to a page with a script, where the script has previously been loaded, should not add an additional copy of the script.

Link to Minimal Reproducible Example

https://stackblitz.com/github/Boston343/astro-5-client-router-demo

Participation

  • I am willing to submit a pull request for this issue.
@github-actions github-actions bot added the needs triage Issue needs to be triaged label Dec 21, 2024
@Boston343 Boston343 changed the title Issues with ClientRouter in Astro V5 Astro v5 ClientRouter duplicating scripts when deployed Dec 21, 2024
@martrapp
Copy link
Member

Hi @Boston343 👋🏼!

Astro 5 issues short scripts as inline module scripts. The client router re-executes inline scripts if they where not also included on the previous page.

You can force them to be external by adding this to your astro.config:

export default defineConfig({
  vite: {
    build: {
      assetsInlineLimit: 0,
    },
  },
});

@martrapp martrapp removed the needs triage Issue needs to be triaged label Dec 21, 2024
@martrapp
Copy link
Member

We should make that clearer in the docs.

@edard3v
Copy link

edard3v commented Dec 21, 2024

I had the same problem. You saved my life bro, now it works as I expected, I was close to quitting Astro 😢 thanks. Although I didn't understand the explanation very well (I use google translate, I'm from Colombia). It seems super strange to me that it works in development and not in production, isn't that counterproductive? @martrapp

@Boston343
Copy link
Author

Boston343 commented Dec 21, 2024

Hey @martrapp, thanks for looking at it.

That's interesting - I always thought you had to opt-in to inlining with is:inline on the script tag. Or maybe I'm misunderstanding. But this does seem to fix the issue and it's an easy addition - appreciated!

Although with this solution, what are the negatives of doing it?

@martrapp
Copy link
Member

Hi @edard3v happy that you didn't quit!

For a bit more background: dev and build are two different beasts for astro. We try to have them behave the same, but "build for production" runs optimizations that can't be done during dev mode with hot module replacement, which allows you to directly see your changes while you develop.

@martrapp
Copy link
Member

@Boston343 the downside is that you also switch off base 64 urls for small assets like images < 4k.

@martrapp martrapp added the - P2: has workaround Bug, but has workaround (priority) label Dec 21, 2024
@Boston343
Copy link
Author

@martrapp As this doesn't look like a new Vite feature, did Astro disable it by default in v4?

@martrapp
Copy link
Member

In contact with @bluwy about that question. Just scanning the code ;-)

@martrapp
Copy link
Member

@martrapp As this doesn't look like a new Vite feature, did Astro disable it by default in v4?

Seems that we have more optimization opportunities with Astro 5. ;-)
The optimization is not new, but Astro 5 rather generates more smaller scripts instead of few big ones and more inlining is the consequence. Formerly, scripts used with the client side router were bundled with the router code and together always were larger than 4k.

@martrapp
Copy link
Member

martrapp commented Dec 21, 2024

Note that this issue never seems to occur with scripts with imports like import from astro:transitions/client.

This doesn’t significantly increase your script size (beyond a few bytes for the import declaration), but it ensures Astro doesn’t inline the script because its bundled version would exceed 4 KB. Also note that this module is already cached and loaded because the client router is in use, so there should be no extra runtime cost.

@Boston343
Copy link
Author

Note that this issue never seems to occur with scripts with imports like import from astro:transitions/client.

This doesn’t significantly increase your script size (beyond a few bytes for the import declaration), but it ensures Astro doesn’t inline the script because its bundled version would exceed 4 KB. Also note that this module is already cached and loaded because the client router is in use, so there should be no extra runtime cost.

So if you have any import statement in the script tag, it won't be inlined, even if whatever you import isn't actually used? Just want to make sure I understand that correctly.

Suggestion: I think the preferred option would be if there's an Astro config option to disable inlining of scripts (at least for scripts that don't have the is:inline attribute), which would allow for the Vite optimization to have effect over other things like CSS and small images, while allowing the old behavior of astro component <script> tags to play nicely with ClientRouter.

@bluwy
Copy link
Member

bluwy commented Jan 9, 2025

@martrapp I'm looking into a fix here and I wonder, would this issue also happen if someone does <script is:inline>? Does the router already handle this case properly?

In any case we could look into supporting build.inlineScripts like build.inlineStylesheets (even though scripts aren't exactly assets, but it'll help those who wants it).

@martrapp
Copy link
Member

martrapp commented Jan 9, 2025

Hi Blu!
Yes, build.inlineScripts with a default of 'never' for compatibility would imho be fine.
Or a different limit for scripts that defaults to 0.

Here is some background on what's happening:

In this issue, we are always talking about scripts that are present on the current page but not on the previous page. In this case the client router tries to execute the script. It does not matter what kind of script it is.

So yes, re-execution would also happen with is:inline scripts, and this fits the documentation.
But the docs also state that module scripts are only executed once.
(Which always should have been "external module scripts are only executed once".)

And the reason for this is not the client router but the browser's module loader:

The browser's module loader will ignore an external module script if it encountered it before since the last full page load, no matter how often the client router wants to execute it.

This behavior is changed by inlining the module script: When the client router executes the inlined module script it succeeds as the module loader treats each occurrence of an inlined module script as a new, different module, even if the code is exactly the same.

Users who wrote <script type="module" is:inline>...</script> in Astro 4.x, had exactly the same situation, but there it was obvious that this is an inline script.

Astro's script execution rules grew complicated over time. I'd would keep those rules as they are and rather not inline module scripts when using view transitions. If the user explicitly enables script inlining, that is fine as she/he know what she/he does.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
- P2: has workaround Bug, but has workaround (priority)
Projects
None yet
Development

No branches or pull requests

4 participants