Skip to content

[Cache Components] Prevent streaming fetch calls from hanging in dev#89171

Merged
unstubbable merged 2 commits intocanaryfrom
hl/fix-hmr-fetch-cache-with-abort
Jan 28, 2026
Merged

[Cache Components] Prevent streaming fetch calls from hanging in dev#89171
unstubbable merged 2 commits intocanaryfrom
hl/fix-hmr-fetch-cache-with-abort

Conversation

@unstubbable
Copy link
Contributor

With Cache Components enabled in development, fetches without an explicit cache config were incorrectly routed through createCachedPrerenderResponse during the cache filling phase due to the serverComponentsHmrCache condition. This function awaits the entire response before returning to the caller, which prevents infinitely streaming responses from ever being read or aborted.

This is a regression that was introduced in #84088.

These fetches should be treated as dynamic and use createCachedDynamicResponse instead, which returns immediately and caches in the background with proper error handling. The fix adds isCacheableRevalidate to the condition so only explicitly cacheable fetches use the blocking path.

Additionally, the "Failed to set fetch cache" warning is now suppressed when the request's abort signal was triggered, since aborting is intentional.

@unstubbable unstubbable changed the base branch from canary to graphite-base/89171 January 28, 2026 15:44
@unstubbable unstubbable force-pushed the hl/fix-hmr-fetch-cache-with-abort branch from 6f4ccda to 27910fe Compare January 28, 2026 15:44
@unstubbable unstubbable changed the base branch from graphite-base/89171 to hl/silence-baseline-browser-mapping-warning January 28, 2026 15:45
Copy link
Contributor Author

unstubbable commented Jan 28, 2026

@unstubbable unstubbable changed the base branch from hl/silence-baseline-browser-mapping-warning to graphite-base/89171 January 28, 2026 15:55
@unstubbable unstubbable force-pushed the hl/fix-hmr-fetch-cache-with-abort branch from 27910fe to 41b1402 Compare January 28, 2026 15:55
@unstubbable unstubbable changed the base branch from graphite-base/89171 to hl/prettier-ignore-bundle-analyzer-next-env-d-ts January 28, 2026 15:55
@nextjs-bot
Copy link
Collaborator

nextjs-bot commented Jan 28, 2026

Tests Passed

@unstubbable unstubbable changed the base branch from hl/prettier-ignore-bundle-analyzer-next-env-d-ts to graphite-base/89171 January 28, 2026 16:05
@unstubbable unstubbable force-pushed the hl/fix-hmr-fetch-cache-with-abort branch from 41b1402 to bc21670 Compare January 28, 2026 16:05
@unstubbable unstubbable changed the base branch from graphite-base/89171 to hl/silence-baseline-browser-mapping-warning January 28, 2026 16:06
@codspeed-hq
Copy link

codspeed-hq bot commented Jan 28, 2026

Merging this PR will not alter performance

✅ 17 untouched benchmarks
⏩ 3 skipped benchmarks1


Comparing hl/fix-hmr-fetch-cache-with-abort (27910fe) with canary (1ce62f2)2

Open in CodSpeed

Footnotes

  1. 3 benchmarks were skipped, so the baseline results were used instead. If they were deleted from the codebase, click here and archive them to remove them from the performance reports.

  2. No successful run was found on hl/silence-baseline-browser-mapping-warning (1ce62f2) during the generation of this report, so canary (1ce62f2) was used instead as the comparison base. There might be some changes unrelated to this pull request in this report.

@unstubbable unstubbable force-pushed the hl/silence-baseline-browser-mapping-warning branch from 99e7bfa to e731c0b Compare January 28, 2026 16:32
@unstubbable unstubbable force-pushed the hl/fix-hmr-fetch-cache-with-abort branch from bc21670 to 7fa7f13 Compare January 28, 2026 16:32
@nextjs-bot
Copy link
Collaborator

nextjs-bot commented Jan 28, 2026

Stats from current PR

✅ No significant changes detected

📊 All Metrics
📖 Metrics Glossary

Dev Server Metrics:

  • Listen = TCP port starts accepting connections
  • First Request = HTTP server returns successful response
  • Cold = Fresh build (no cache)
  • Warm = With cached build artifacts

Build Metrics:

  • Fresh = Clean build (no .next directory)
  • Cached = With existing .next directory

Change Thresholds:

  • Time: Changes < 50ms AND < 10%, OR < 2% are insignificant
  • Size: Changes < 1KB AND < 1% are insignificant
  • All other changes are flagged to catch regressions

⚡ Dev Server

Metric Canary PR Change Trend
Cold (Listen) 455ms 455ms ▁▁▁▁█
Cold (Ready in log) 440ms 439ms ▇▇▇▄▆
Cold (First Request) 1.165s 1.154s ███▇▄
Warm (Listen) 456ms 456ms ▁▁▁▁█
Warm (Ready in log) 444ms 443ms ▁▁▂▁█
Warm (First Request) 346ms 341ms ███▇▅
📦 Dev Server (Webpack) (Legacy)

📦 Dev Server (Webpack)

Metric Canary PR Change Trend
Cold (Listen) 456ms 455ms ▁▁▁▁█
Cold (Ready in log) 439ms 438ms ▃▃▁▃█
Cold (First Request) 1.930s 1.859s ▃▂▁▂█
Warm (Listen) 456ms 456ms ▁▁▁▁█
Warm (Ready in log) 438ms 439ms ▃▃▁▃█
Warm (First Request) 1.896s 1.878s ▃▂▁▂█

⚡ Production Builds

Metric Canary PR Change Trend
Fresh Build 4.349s 4.391s ▁▁▁▁█
Cached Build 4.340s 4.272s ▁▁▁▁█
📦 Production Builds (Webpack) (Legacy)

📦 Production Builds (Webpack)

Metric Canary PR Change Trend
Fresh Build 14.128s 14.172s ▃▁▁▁█
Cached Build 14.213s 14.234s ▃▁▁▂█
node_modules Size 462 MB 462 MB ▁████
📦 Bundle Sizes

Bundle Sizes

⚡ Turbopack

Client

Main Bundles: **432 kB** → **432 kB** ✅ -42 B

81 files with content-based hashes (individual files not comparable between builds)

Server

Middleware
Canary PR Change
middleware-b..fest.js gzip 770 B 763 B
Total 770 B 763 B ✅ -7 B
Build Details
Build Manifests
Canary PR Change
_buildManifest.js gzip 451 B 450 B
Total 451 B 450 B ✅ -1 B

📦 Webpack

Client

Main Bundles
Canary PR Change
5528-HASH.js gzip 5.47 kB N/A -
6280-HASH.js gzip 52.7 kB N/A -
6335.HASH.js gzip 169 B N/A -
912-HASH.js gzip 4.53 kB N/A -
e8aec2e4-HASH.js gzip 62.3 kB N/A -
framework-HASH.js gzip 59.7 kB 59.7 kB
main-app-HASH.js gzip 256 B 254 B
main-HASH.js gzip 39 kB 39 kB
webpack-HASH.js gzip 1.68 kB 1.68 kB
262-HASH.js gzip N/A 4.52 kB -
2889.HASH.js gzip N/A 169 B -
5602-HASH.js gzip N/A 5.48 kB -
6948ada0-HASH.js gzip N/A 62.3 kB -
9544-HASH.js gzip N/A 53.2 kB -
Total 226 kB 226 kB ⚠️ +477 B
Polyfills
Canary PR Change
polyfills-HASH.js gzip 39.4 kB 39.4 kB
Total 39.4 kB 39.4 kB
Pages
Canary PR Change
_app-HASH.js gzip 194 B 194 B
_error-HASH.js gzip 183 B 180 B 🟢 3 B (-2%)
css-HASH.js gzip 331 B 330 B
dynamic-HASH.js gzip 1.81 kB 1.81 kB
edge-ssr-HASH.js gzip 256 B 256 B
head-HASH.js gzip 351 B 352 B
hooks-HASH.js gzip 384 B 383 B
image-HASH.js gzip 580 B 581 B
index-HASH.js gzip 260 B 260 B
link-HASH.js gzip 2.49 kB 2.49 kB
routerDirect..HASH.js gzip 320 B 319 B
script-HASH.js gzip 386 B 386 B
withRouter-HASH.js gzip 315 B 315 B
1afbb74e6ecf..834.css gzip 106 B 106 B
Total 7.97 kB 7.97 kB ✅ -1 B

Server

Edge SSR
Canary PR Change
edge-ssr.js gzip 126 kB 126 kB
page.js gzip 246 kB 247 kB
Total 373 kB 373 kB ⚠️ +264 B
Middleware
Canary PR Change
middleware-b..fest.js gzip 615 B 615 B
middleware-r..fest.js gzip 156 B 155 B
middleware.js gzip 32.9 kB 33.1 kB
edge-runtime..pack.js gzip 842 B 842 B
Total 34.5 kB 34.8 kB ⚠️ +271 B
Build Details
Build Manifests
Canary PR Change
_buildManifest.js gzip 732 B 736 B
Total 732 B 736 B ⚠️ +4 B
Build Cache
Canary PR Change
0.pack gzip 3.71 MB 3.72 MB 🔴 +11 kB (+0%)
index.pack gzip 101 kB 103 kB 🔴 +1.77 kB (+2%)
index.pack.old gzip 99.2 kB 102 kB 🔴 +2.63 kB (+3%)
Total 3.91 MB 3.93 MB ⚠️ +15.4 kB

🔄 Shared (bundler-independent)

Runtimes
Canary PR Change
app-page-exp...dev.js gzip 308 kB 308 kB
app-page-exp..prod.js gzip 164 kB 164 kB
app-page-tur...dev.js gzip 308 kB 308 kB
app-page-tur..prod.js gzip 164 kB 164 kB
app-page-tur...dev.js gzip 305 kB 305 kB
app-page-tur..prod.js gzip 162 kB 162 kB
app-page.run...dev.js gzip 305 kB 305 kB
app-page.run..prod.js gzip 162 kB 162 kB
app-route-ex...dev.js gzip 70.2 kB 70.2 kB
app-route-ex..prod.js gzip 48.8 kB 48.8 kB
app-route-tu...dev.js gzip 70.2 kB 70.2 kB
app-route-tu..prod.js gzip 48.8 kB 48.8 kB
app-route-tu...dev.js gzip 69.8 kB 69.8 kB
app-route-tu..prod.js gzip 48.6 kB 48.6 kB
app-route.ru...dev.js gzip 69.8 kB 69.8 kB
app-route.ru..prod.js gzip 48.6 kB 48.6 kB
dist_client_...dev.js gzip 324 B 324 B
dist_client_...dev.js gzip 326 B 326 B
dist_client_...dev.js gzip 318 B 318 B
dist_client_...dev.js gzip 317 B 317 B
pages-api-tu...dev.js gzip 43.1 kB 43.1 kB
pages-api-tu..prod.js gzip 32.8 kB 32.8 kB
pages-api.ru...dev.js gzip 43.1 kB 43.1 kB
pages-api.ru..prod.js gzip 32.8 kB 32.8 kB
pages-turbo....dev.js gzip 52.4 kB 52.4 kB
pages-turbo...prod.js gzip 39.3 kB 39.3 kB
pages.runtim...dev.js gzip 52.3 kB 52.3 kB
pages.runtim..prod.js gzip 39.3 kB 39.3 kB
server.runti..prod.js gzip 62.5 kB 62.5 kB
Total 2.75 MB 2.75 MB ⚠️ +147 B
📝 Changed Files (8 files)

Files with changes:

  • app-route-ex..ntime.dev.js
  • app-route-ex..time.prod.js
  • app-route-tu..ntime.dev.js
  • app-route-tu..time.prod.js
  • app-route-tu..ntime.dev.js
  • app-route-tu..time.prod.js
  • app-route.runtime.dev.js
  • app-route.ru..time.prod.js
View diffs
app-route-ex..ntime.dev.js

Diff too large to display

app-route-ex..time.prod.js

Diff too large to display

app-route-tu..ntime.dev.js

Diff too large to display

app-route-tu..time.prod.js

Diff too large to display

app-route-tu..ntime.dev.js

Diff too large to display

app-route-tu..time.prod.js

Diff too large to display

app-route.runtime.dev.js

Diff too large to display

app-route.ru..time.prod.js

Diff too large to display

@unstubbable unstubbable changed the base branch from hl/silence-baseline-browser-mapping-warning to graphite-base/89171 January 28, 2026 18:24
@unstubbable unstubbable force-pushed the hl/fix-hmr-fetch-cache-with-abort branch from 7fa7f13 to f578636 Compare January 28, 2026 18:24
@graphite-app graphite-app bot changed the base branch from graphite-base/89171 to canary January 28, 2026 18:25
@unstubbable unstubbable force-pushed the hl/fix-hmr-fetch-cache-with-abort branch from f578636 to b6bcd91 Compare January 28, 2026 18:25
With Cache Components enabled in development, fetches without an
explicit cache config were incorrectly routed through
`createCachedPrerenderResponse` during the cache filling phase due to
the `serverComponentsHmrCache` condition. This function awaits the
entire response before returning to the caller, which prevents
infinitely streaming responses from ever being read or aborted.

This is a regression that was introduced in #84088.

These fetches should be treated as dynamic and use
`createCachedDynamicResponse` instead, which returns immediately and
caches in the background with proper error handling. The fix adds
`isCacheableRevalidate` to the condition so only explicitly cacheable
fetches use the blocking path.

Additionally, the "Failed to set fetch cache" warning is now suppressed
when the request's abort signal was triggered, since aborting is
intentional.
@unstubbable unstubbable force-pushed the hl/fix-hmr-fetch-cache-with-abort branch from b6bcd91 to d9c7cf3 Compare January 28, 2026 18:32
@unstubbable unstubbable marked this pull request as ready for review January 28, 2026 20:27
@unstubbable unstubbable merged commit 5ec4d7a into canary Jan 28, 2026
285 of 288 checks passed
@unstubbable unstubbable deleted the hl/fix-hmr-fetch-cache-with-abort branch January 28, 2026 20:53
unstubbable added a commit that referenced this pull request Jan 28, 2026
…89171)

With Cache Components enabled in development, fetches without an
explicit cache config were incorrectly routed through
`createCachedPrerenderResponse` during the cache filling phase due to
the `serverComponentsHmrCache` condition. This function awaits the
entire response before returning to the caller, which prevents
infinitely streaming responses from ever being read or aborted.

This is a regression that was introduced in #84088.

These fetches should be treated as dynamic and use
`createCachedDynamicResponse` instead, which returns immediately and
caches in the background with proper error handling. The fix adds
`isCacheableRevalidate` to the condition so only explicitly cacheable
fetches use the blocking path.

Additionally, the "Failed to set fetch cache" warning is now suppressed
when the request's abort signal was triggered, since aborting is
intentional.
unstubbable added a commit that referenced this pull request Jan 29, 2026
…89171)

With Cache Components enabled in development, fetches without an
explicit cache config were incorrectly routed through
`createCachedPrerenderResponse` during the cache filling phase due to
the `serverComponentsHmrCache` condition. This function awaits the
entire response before returning to the caller, which prevents
infinitely streaming responses from ever being read or aborted.

This is a regression that was introduced in #84088.

These fetches should be treated as dynamic and use
`createCachedDynamicResponse` instead, which returns immediately and
caches in the background with proper error handling. The fix adds
`isCacheableRevalidate` to the condition so only explicitly cacheable
fetches use the blocking path.

Additionally, the "Failed to set fetch cache" warning is now suppressed
when the request's abort signal was triggered, since aborting is
intentional.
unstubbable added a commit that referenced this pull request Jan 29, 2026
@github-actions github-actions bot locked as resolved and limited conversation to collaborators Feb 12, 2026
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.

Projects

None yet

Development

Successfully merging this pull request may close these issues.

3 participants