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

App Router in Next v14.1.0 slower than v13.4.6 #61259

Closed
amavisse opened this issue Jan 27, 2024 · 8 comments
Closed

App Router in Next v14.1.0 slower than v13.4.6 #61259

amavisse opened this issue Jan 27, 2024 · 8 comments
Labels
bug Issue was opened via the bug report template. locked Navigation Related to Next.js linking (e.g., <Link>) and navigation.

Comments

@amavisse
Copy link

Link to the code that reproduces this issue

https://github.com/amavisse/next14-test

To Reproduce

I reproduced the issue here:
https://next13-test-teal.vercel.app/ - repo: https://github.com/amavisse/next13-test
https://next14-test-delta.vercel.app/ - repo: https://github.com/amavisse/next14-test

  1. Visit both links
  2. Click on any of the top links to change route and notice the difference in loading page

Current vs. Expected behavior

Current: page loading is very slow on next v14.1.0

Expected: fast loading as in https://next13-test-teal.vercel.app/

Provide environment information

Operating System:
  Platform: win32
  Arch: x64
  Version: Windows 10 Home
Binaries:
  Node: 20.2.0
  npm: N/A
  Yarn: N/A
  pnpm: N/A
Relevant packages:
  next: 13.4.6
  eslint-config-next: 14.1.0
  react: 18.2.0
  react-dom: 18.2.0
  typescript: 5.3.3

-------------------------------------
Operating System:
  Platform: win32
  Arch: x64
  Version: Windows 10 Home  
Binaries:
  Node: 20.2.0
  npm: N/A
  Yarn: N/A
  pnpm: N/A
Relevant Packages:
  next: 14.1.0
  eslint-config-next: 14.1.0
  react: 18.2.0
  react-dom: 18.2.0
  typescript: 5.3.3
Next.js Config:
  output: N/A

Which area(s) are affected? (Select all that apply)

App Router, Routing (next/router, next/navigation, next/link)

Which stage(s) are affected? (Select all that apply)

next dev (local), Vercel (Deployed)

Additional context

In the next v13.4.6, everything is going smoothly, while the v14.1.0 one is very slow when changing routes
Inspecting the dev console, inside the network tab, you can see the routes are correctly pre-fetched but when moving to the route, it's fetching it again, making it extremely slower.

Am I doing something wrong? bad practices? is it a bug?

@amavisse amavisse added the bug Issue was opened via the bug report template. label Jan 27, 2024
@github-actions github-actions bot added the Navigation Related to Next.js linking (e.g., <Link>) and navigation. label Jan 27, 2024
@OlegLustenko
Copy link

Could be related #61203

@repcolding
Copy link

14.1.0

2024-01-29.15.04.38.mp4

14.0.3.canarey-7:

2024-01-29.15.03.44.mp4

A bug with caching routes, when hovering over next/link, the route is loaded again. Slow 3g is selected in the video, but even with fast Internet, the behavior is exactly the same.

Prefetch also does not load css.

It took a lot of time to figure out what the problem was.

The biggest mistake is to believe that the new versions are stable.

@Yhozen
Copy link

Yhozen commented Feb 5, 2024

Related to #48748 and I believe it it related to optimizePackageImports not working properly (I'm compiling 10k+ modules per route aprox)

@ztanner
Copy link
Member

ztanner commented Feb 7, 2024

Hi @amavisse -- thanks for raising this issue. Referencing the Prefetching Docs here, it seems like this is working as intended (in the latest version) and was actually a bug in the older version.

Specifically, note the following section of docs:

The Link's prefetching behavior is different for static and dynamic routes:
Static Routes: prefetch defaults to true. The entire route is prefetched and cached.
Dynamic Routes: prefetch default to automatic. Only the shared layout, down the rendered "tree" of components until the first loading.js file, is prefetched and cached for 30s. This reduces the cost of fetching an entire dynamic route, and it means you can show an instant loading state for better visual feedback to users.

In other words, if the prefetch argument to the Link tag is left unspecified, it will default to a partial prefetch (essentially up to the nearest loading boundary, if there is one). This is happening in your case, because each [...slug] page is considered a dynamic route.

Fortunately there's a way to opt-in to the behavior you're describing: if you pass prefetch={true} to your Link components, it will tell the router to perform a full prefetch of each of the pages (including the dynamic page data), and will instantly navigate without triggering a follow-up fetch.

In other words, update your Navigation component to look like this:

function Navigation() {
  const links = [
    { href: '/', name: 'Home' },
    { href: '/page1', name: 'Page1' },
    { href: '/page2', name: 'Page2' },
    { href: '/page3', name: 'Page3' },
  ]
  return (
    <nav className="flex flex-row justify-center gap-5 py-20">
      {links.map((link, i) => (
        <Link
          className="hover:underline"
          key={link.name + i}
          href={link.href}
          prefetch={true} // note the prefetch prop here
        >
          {link.name}
        </Link>
      ))}
    </nav>
  )
}

I'm going to close this as it seems to be working as designed, but if you have any questions about it, please don't hesitate to reply to this or raise in our Discussions forum.

@ztanner ztanner closed this as not planned Won't fix, can't repro, duplicate, stale Feb 7, 2024
@repcolding
Copy link

repcolding commented Feb 8, 2024

@ztanner Hi, if everything works as it should, that's suck

I checked the version [email protected] and 14.0.3-canary.7. Specifying prefetch={true}. And the result is terrible:

[email protected]: No preloading of css
[email protected]: css is preloaded

const Folder: FC<IFolder> = ({ children, label, className, href = '#' }) => {
  return (
    <Link
      prefetch={true} // [email protected] and [email protected]
      scroll={false}
      href={href}
      className={cn(styles.folder, className)}
    >
      {children}

      <div className={styles.label}>{label}</div>
    </Link>
  )
}

[email protected]

next@14.0.3-canary.7

prefetch={true} preload css is working


[email protected]

next@14.1.0

prefetch={true} preload css doesn't working

@ztanner
Copy link
Member

ztanner commented Feb 8, 2024

Hi @repcolding -- I didn't state that everything is working as expected, just that prefetching semantics in the version you mentioned were not the intended behavior. If there's a separate bug with CSS loading, we can look into that.

@jinowac
Copy link

jinowac commented Feb 19, 2024

I am facing the same issue , Next js 14 version Link component is terribly slow , navigating between pages takes hell out of you . @ztanner Please make sure higher versions are stable .

Copy link
Contributor

github-actions bot commented Mar 5, 2024

This closed issue has been automatically locked because it had no new activity for 2 weeks. If you are running into a similar issue, please create a new issue with the steps to reproduce. Thank you.

@github-actions github-actions bot added the locked label Mar 5, 2024
@github-actions github-actions bot locked as resolved and limited conversation to collaborators Mar 5, 2024
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
bug Issue was opened via the bug report template. locked Navigation Related to Next.js linking (e.g., <Link>) and navigation.
Projects
None yet
Development

No branches or pull requests

6 participants