Skip to content

ref(core): Remove redundant iframe check in supportsNativeFetch#19853

Closed
HazAT wants to merge 5 commits intodevelopfrom
bundle-size/supports-native-fetch
Closed

ref(core): Remove redundant iframe check in supportsNativeFetch#19853
HazAT wants to merge 5 commits intodevelopfrom
bundle-size/supports-native-fetch

Conversation

@HazAT
Copy link
Member

@HazAT HazAT commented Mar 17, 2026

Summary

Remove ~30 lines of iframe-based DOM manipulation from supportsNativeFetch(). Saves ~200 bytes gzipped — the single biggest win in this bundle size effort.

Problem

supportsNativeFetch() created a sandboxed iframe to check if fetch is natively implemented. This is identical logic to getNativeImplementation("fetch") in browser-utils, creating two separate iframe-based checks in the CDN bundle.

The function is only called behind a skipNativeFetchCheck guard:

if (skipNativeFetchCheck && !supportsNativeFetch()) { return; }

In the base CDN bundle, skipNativeFetchCheck is never set to true, making the iframe code dead weight. Terser cannot tree-shake it because it cannot prove the parameter is always falsy.

Solution

Simplify supportsNativeFetch() to just delegate to _isFetchSupported() (checks if window.fetch exists). The native-vs-polyfill distinction is handled by getNativeImplementation at call sites that need it.

Also simplifies isNativeFunction from /^function\s+\w+\(\)\s+\{\s+\[native code\]\s+\}$/ to /\[native code\]/ — more permissive across different JS engine toString() formats.

Part of #19833.

Co-Authored-By: Claude claude@anthropic.com

@github-actions
Copy link
Contributor

github-actions bot commented Mar 17, 2026

size-limit report 📦

⚠️ Warning: Base artifact is not the latest one, because the latest workflow run is not done yet. This may lead to incorrect results. Try to re-run all tests to get up to date results.

Path Size % Change Change
@sentry/browser 25.64 kB +0.02% +3 B 🔺
@sentry/browser - with treeshaking flags 24.16 kB +0.08% +19 B 🔺
@sentry/browser (incl. Tracing) 42.67 kB +0.12% +50 B 🔺
@sentry/browser (incl. Tracing, Profiling) 47.32 kB +0.09% +41 B 🔺
@sentry/browser (incl. Tracing, Replay) 81.43 kB +0.01% +7 B 🔺
@sentry/browser (incl. Tracing, Replay) - with treeshaking flags 71.07 kB +0.11% +75 B 🔺
@sentry/browser (incl. Tracing, Replay with Canvas) 86.11 kB -0.02% -10 B 🔽
@sentry/browser (incl. Tracing, Replay, Feedback) 98.38 kB +0.01% +2 B 🔺
@sentry/browser (incl. Feedback) 42.45 kB +0.01% +3 B 🔺
@sentry/browser (incl. sendFeedback) 30.31 kB +0.01% +2 B 🔺
@sentry/browser (incl. FeedbackAsync) 35.37 kB +0.02% +4 B 🔺
@sentry/browser (incl. Metrics) 26.93 kB +0.03% +6 B 🔺
@sentry/browser (incl. Logs) 27.07 kB +0.01% +2 B 🔺
@sentry/browser (incl. Metrics & Logs) 27.74 kB -0.01% -1 B 🔽
@sentry/react 27.41 kB +0.06% +16 B 🔺
@sentry/react (incl. Tracing) 44.97 kB +0.04% +16 B 🔺
@sentry/vue 30.07 kB -0.03% -9 B 🔽
@sentry/vue (incl. Tracing) 44.49 kB +0.03% +11 B 🔺
@sentry/svelte 25.66 kB -0.02% -4 B 🔽
CDN Bundle 28.35 kB +0.25% +70 B 🔺
CDN Bundle (incl. Tracing) 43.55 kB +0.1% +42 B 🔺
CDN Bundle (incl. Logs, Metrics) 29.2 kB +0.2% +58 B 🔺
CDN Bundle (incl. Tracing, Logs, Metrics) 44.42 kB +0.16% +67 B 🔺
CDN Bundle (incl. Replay, Logs, Metrics) 68.28 kB +0.1% +67 B 🔺
CDN Bundle (incl. Tracing, Replay) 80.44 kB +0.14% +111 B 🔺
CDN Bundle (incl. Tracing, Replay, Logs, Metrics) 81.35 kB +0.14% +113 B 🔺
CDN Bundle (incl. Tracing, Replay, Feedback) 85.96 kB +0.11% +93 B 🔺
CDN Bundle (incl. Tracing, Replay, Feedback, Logs, Metrics) 86.85 kB +0.09% +78 B 🔺
CDN Bundle - uncompressed 82.56 kB -0.08% -58 B 🔽
CDN Bundle (incl. Tracing) - uncompressed 128.49 kB -0.06% -75 B 🔽
CDN Bundle (incl. Logs, Metrics) - uncompressed 85.43 kB -0.07% -58 B 🔽
CDN Bundle (incl. Tracing, Logs, Metrics) - uncompressed 131.35 kB -0.06% -75 B 🔽
CDN Bundle (incl. Replay, Logs, Metrics) - uncompressed 209.08 kB -0.02% -37 B 🔽
CDN Bundle (incl. Tracing, Replay) - uncompressed 245.36 kB -0.02% -49 B 🔽
CDN Bundle (incl. Tracing, Replay, Logs, Metrics) - uncompressed 248.22 kB -0.02% -49 B 🔽
CDN Bundle (incl. Tracing, Replay, Feedback) - uncompressed 258.27 kB -0.02% -49 B 🔽
CDN Bundle (incl. Tracing, Replay, Feedback, Logs, Metrics) - uncompressed 261.12 kB -0.02% -49 B 🔽
@sentry/nextjs (client) 47.4 kB +0.07% +33 B 🔺
@sentry/sveltekit (client) 43.08 kB +0.03% +9 B 🔺
@sentry/node-core 56.42 kB +0.13% +73 B 🔺
@sentry/node 173.38 kB +0.13% +221 B 🔺
@sentry/node - without tracing 96.43 kB +0.1% +87 B 🔺
@sentry/aws-serverless 113.44 kB +0.09% +100 B 🔺

View base workflow run

@github-actions
Copy link
Contributor

github-actions bot commented Mar 17, 2026

node-overhead report 🧳

Note: This is a synthetic benchmark with a minimal express app and does not necessarily reflect the real-world performance impact in an application.

Scenario Requests/s % of Baseline Prev. Requests/s Change %
GET Baseline 8,646 - 8,555 +1%
GET With Sentry 1,523 18% 1,560 -2%
GET With Sentry (error only) 5,736 66% 5,893 -3%
POST Baseline 1,136 - 1,145 -1%
POST With Sentry 535 47% 553 -3%
POST With Sentry (error only) 996 88% 1,031 -3%
MYSQL Baseline 3,106 - 3,189 -3%
MYSQL With Sentry 327 11% 401 -18%
MYSQL With Sentry (error only) 2,558 82% 2,597 -2%

View base workflow run

@Lms24 Lms24 force-pushed the bundle-size/supports-native-fetch branch from e2fd3a8 to 39fcc12 Compare March 19, 2026 11:42
@Lms24 Lms24 requested review from a team as code owners March 19, 2026 11:42
@Lms24 Lms24 changed the title refactor(core): Remove redundant iframe check in supportsNativeFetch ref(core): Remove redundant iframe check in supportsNativeFetch Mar 19, 2026
@Lms24 Lms24 changed the title ref(core): Remove redundant iframe check in supportsNativeFetch ref(core): Remove redundant iframe check in supportsNativeFetch Mar 19, 2026
@Lms24 Lms24 self-assigned this Mar 19, 2026
@github-actions
Copy link
Contributor

github-actions bot commented Mar 19, 2026

Semver Impact of This PR

🟢 Patch (bug fixes)

📋 Changelog Preview

This is how your changes will appear in the changelog.
Entries from this PR are highlighted with a left border (blockquote style).


New Features ✨

Deps

  • Bump mongodb-memory-server-global from 10.1.4 to 11.0.1 by dependabot in #19888
  • Bump stacktrace-parser from 0.1.10 to 0.1.11 by dependabot in #19887

Bug Fixes 🐛

Core

  • Do not overwrite user provided conversation id in Vercel by nicohrubec in #19903
  • Return same value from startSpan as callback returns by s1gr1d in #19300

Other

  • (cloudflare) Forward ctx argument to Workflow.do user callback by Lms24 in #19891
  • (deps) Bump socket.io-parser to 4.2.6 to fix CVE-2026-33151 by chargome in #19880
  • (nestjs) Add node to nest metadata by chargome in #19875
  • (serverless) Add node to metadata by nicohrubec in #19878

Internal Changes 🔧

  • (astro) Re-enable server island tracing e2e test in Astro 6 by Lms24 in #19872
  • (ci) Fix "Gatbsy" typo in issue package label workflow by chargome in #19905
  • (core) Remove redundant iframe check in supportsNativeFetch by HazAT in #19853
  • (lint) Resolve oxlint warnings by isaacs in #19893
  • (node-integration-tests) Remove unnecessary file-type dependency by Lms24 in #19824
  • (sveltekit) Replace recast + @babel/parser with acorn by roli-lpci in #19533
  • Add external contributor to CHANGELOG.md by javascript-sdk-gitflow in #19909

🤖 This preview updates automatically when you update the PR.

@Lms24 Lms24 changed the base branch from autoresearch/browser-bundle-size-2026-03-17 to develop March 19, 2026 11:49
@Lms24 Lms24 force-pushed the bundle-size/supports-native-fetch branch from 39fcc12 to dff6335 Compare March 19, 2026 12:13
// eslint-disable-next-line @typescript-eslint/ban-types
export function isNativeFunction(func: Function): boolean {
return func && /^function\s+\w+\(\)\s+\{\s+\[native code\]\s+\}$/.test(func.toString());
return func && /\[native code\]/.test(func.toString());
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll revert this. It saves a few bytes but drastically reduces specificity. We have no unit tests that cover this function :(

Comment on lines -111 to -141
if (!_isFetchSupported()) {
return false;
}

// Fast path to avoid DOM I/O
// eslint-disable-next-line @typescript-eslint/unbound-method
if (isNativeFunction(WINDOW.fetch)) {
return true;
}

// window.fetch is implemented, but is polyfilled or already wrapped (e.g: by a chrome extension)
// so create a "pure" iframe to see if that has native fetch
let result = false;
const doc = WINDOW.document;
// eslint-disable-next-line deprecation/deprecation
if (doc && typeof (doc.createElement as unknown) === 'function') {
try {
const sandbox = doc.createElement('iframe');
sandbox.hidden = true;
doc.head.appendChild(sandbox);
if (sandbox.contentWindow?.fetch) {
// eslint-disable-next-line @typescript-eslint/unbound-method
result = isNativeFunction(sandbox.contentWindow.fetch);
}
doc.head.removeChild(sandbox);
} catch (err) {
DEBUG_BUILD && debug.warn('Could not create sandbox iframe for pure fetch check, bailing to window.fetch: ', err);
}
}

return result;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Unfortunately, this is a behaviour change to the original function. Supporting a fetch implementation is not the same as supporting/having the native one. I'm going to experiment with moving getNativeImplementation to core so that we can actually reuse the iframe lookup but I'm not sure how much bundle size this still saves.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Saving old size check for comparison:

size-limit report 📦

⚠️ Warning: Base artifact is not the latest one, because the latest workflow run is not done yet. This may lead to incorrect results. Try to re-run all tests to get up to date results.

Path Size % Change Change
@sentry/browser 25.4 kB -0.92% -234 B 🔽
@sentry/browser - with treeshaking flags 23.94 kB -0.84% -201 B 🔽
@sentry/browser (incl. Tracing) 42.4 kB -0.52% -220 B 🔽
@sentry/browser (incl. Tracing, Profiling) 47.06 kB -0.48% -223 B 🔽
@sentry/browser (incl. Tracing, Replay) 81.18 kB -0.31% -245 B 🔽
@sentry/browser (incl. Tracing, Replay) - with treeshaking flags 70.78 kB -0.3% -211 B 🔽
@sentry/browser (incl. Tracing, Replay with Canvas) 85.88 kB -0.29% -245 B 🔽
@sentry/browser (incl. Tracing, Replay, Feedback) 98.14 kB -0.24% -235 B 🔽
@sentry/browser (incl. Feedback) 42.22 kB -0.53% -221 B 🔽
@sentry/browser (incl. sendFeedback) 30.07 kB -0.78% -234 B 🔽
@sentry/browser (incl. FeedbackAsync) 35.13 kB -0.66% -232 B 🔽
@sentry/browser (incl. Metrics) 26.68 kB -0.92% -246 B 🔽
@sentry/browser (incl. Logs) 26.82 kB -0.91% -245 B 🔽
@sentry/browser (incl. Metrics & Logs) 27.51 kB -0.83% -228 B 🔽
@sentry/react 27.19 kB -0.75% -203 B 🔽
@sentry/react (incl. Tracing) 44.73 kB -0.5% -224 B 🔽
@sentry/vue 29.86 kB -0.73% -218 B 🔽
@sentry/vue (incl. Tracing) 44.26 kB -0.51% -226 B 🔽
@sentry/svelte 25.43 kB -0.93% -238 B 🔽
CDN Bundle 28.11 kB -0.61% -172 B 🔽
CDN Bundle (incl. Tracing) 43.29 kB -0.49% -213 B 🔽
CDN Bundle (incl. Logs, Metrics) 28.97 kB -0.62% -178 B 🔽
CDN Bundle (incl. Tracing, Logs, Metrics) 44.16 kB -0.44% -195 B 🔽
CDN Bundle (incl. Replay, Logs, Metrics) 68.03 kB -0.27% -184 B 🔽
CDN Bundle (incl. Tracing, Replay) 80.15 kB -0.23% -179 B 🔽
CDN Bundle (incl. Tracing, Replay, Logs, Metrics) 81.04 kB -0.24% -193 B 🔽
CDN Bundle (incl. Tracing, Replay, Feedback) 85.66 kB -0.25% -211 B 🔽
CDN Bundle (incl. Tracing, Replay, Feedback, Logs, Metrics) 86.55 kB -0.25% -215 B 🔽
CDN Bundle - uncompressed 81.97 kB -0.8% -655 B 🔽
CDN Bundle (incl. Tracing) - uncompressed 127.89 kB -0.52% -666 B 🔽
CDN Bundle (incl. Logs, Metrics) - uncompressed 84.83 kB -0.77% -655 B 🔽
CDN Bundle (incl. Tracing, Logs, Metrics) - uncompressed 130.76 kB -0.51% -666 B 🔽
CDN Bundle (incl. Replay, Logs, Metrics) - uncompressed 208.47 kB -0.32% -655 B 🔽
CDN Bundle (incl. Tracing, Replay) - uncompressed 244.75 kB -0.28% -666 B 🔽
CDN Bundle (incl. Tracing, Replay, Logs, Metrics) - uncompressed 247.6 kB -0.27% -666 B 🔽
CDN Bundle (incl. Tracing, Replay, Feedback) - uncompressed 257.66 kB -0.26% -666 B 🔽
CDN Bundle (incl. Tracing, Replay, Feedback, Logs, Metrics) - uncompressed 260.5 kB -0.26% -666 B 🔽
@sentry/nextjs (client) 47.16 kB -0.44% -204 B 🔽
@sentry/sveltekit (client) 42.82 kB -0.6% -255 B 🔽
@sentry/node-core 56.28 kB -0.12% -63 B 🔽
@sentry/node 173.21 kB +0.03% +51 B 🔺
@sentry/node - without tracing 96.27 kB -0.08% -76 B 🔽
@sentry/aws-serverless 113.28 kB -0.06% -62 B 🔽

View base workflow run

@Lms24
Copy link
Member

Lms24 commented Mar 19, 2026

trying one more approach here to only extract the iframe stuff

Copy link

@cursor cursor bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Cursor Bugbot has reviewed your changes and found 1 potential issue.

Fix All in Cursor

Bugbot Autofix is OFF. To automatically fix reported issues with cloud agents, enable autofix in the Cursor dashboard.

Comment on lines +48 to +61
// This is a really weird fallback but here's what's going on:
// We're just being extra careful here. According to types, windowImpl is _always_ defined.
// However, in some very rare cases (for example test environments), it may in fact not be defined.
// In exactly this case, if we fail to get an iframeImpl, AND no windowImpl either,
// we skip caching and just return effectively undefined (despite types saying it's always defined)
// This basically tricks TS into thinking this function never returns `undefined` which
// for the most part is true.
if (!windowImpl) {
return windowImpl; // but actually return undefined
}

return (cachedImplementations[name] = impl.bind(WINDOW) as CacheableImplementations[T]);
// If _only_ iframeImpl is undefined and windowImpl is defined and not not native, we end up here
// In this case, we deliberately cache the windowImpl.
return cacheAndReturn(windowImpl);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Writing this down in case we ever wonder why the hell this code path exists:

This PR DID NOT change any prior behavior. I think this function was already flawed but for the scope of this PR, I'll not fix it right away. Two sketchy things:

  • We have a code path where we effectively return undefined, despite TS assuming we always return a defined implementation (see 1st comment why)
  • We return and cache a non-native, potentially patched version of an implementation in a function that's called getNativeImplementation.

This PR just more clearly call out this behaviour as it was quite hidden away in the prior implementation. I'll investigate if we can fix at least the latter flaw by returning undefined.

HazAT and others added 5 commits March 20, 2026 13:24
supportsNativeFetch() created a sandboxed iframe to check if the Fetch API
is natively implemented — identical logic to getNativeImplementation('fetch')
in browser-utils. The iframe approach adds ~30 lines of DOM manipulation code.

The function is only called behind a `skipNativeFetchCheck` guard that is
never set to true in the base CDN bundle, making the iframe code dead weight
that cannot be tree-shaken by terser (it can't prove the parameter is always
falsy across the program).

Simplify to delegate to `_isFetchSupported()` which checks if `window.fetch`
exists. The actual native-vs-polyfill distinction is already handled by
`getNativeImplementation` at the call sites that need it.

Also simplifies the `isNativeFunction` regex from an exact whitespace pattern
to a simpler `/[native code]/` check, which is more permissive across
different JS engine toString() formats.

Saves ~200 bytes gzipped on the base browser bundle.

Co-Authored-By: Claude claude@anthropic.com
@Lms24 Lms24 force-pushed the bundle-size/supports-native-fetch branch from 416056a to f7c95ac Compare March 20, 2026 12:24
@Lms24
Copy link
Member

Lms24 commented Mar 20, 2026

lol, so after making far to many changes to still get this in without behaviour changes but less duplication, I ended up with a larger bundle size than before. Code is arguably cleaner now but honestly not worth pursuing further. Closing this experiment 😅

@Lms24 Lms24 closed this Mar 20, 2026
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants