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

double invoke test #1

Closed
wants to merge 44 commits into from
Closed
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
44 commits
Select commit Hold shift + click to select a range
fecc288
[react devtools] Device storage support (#25452)
rbalicki2 Oct 25, 2022
1720405
[Float] fix coordination of resource identity and hydration (#25569)
gnoff Oct 27, 2022
09def59
[Float] handle noscript context for Resources (#25559)
gnoff Oct 27, 2022
28a574e
Try assigning fetch to globalThis if global assignment fails (#25571)
sebmarkbage Oct 27, 2022
cf3932b
Remove old react-fetch, react-fs and react-pg libraries (#25577)
sebmarkbage Oct 27, 2022
d2a0176
Detect and warn if use(promise) is wrapped with try/catch block (#25543)
acdlite Oct 28, 2022
5450dd4
Strict Mode: Reuse memoized result from first pass (#25583)
acdlite Oct 28, 2022
d2c0ab1
In work loop, add enum of reasons for suspending
acdlite Oct 24, 2022
952dfff
Split suspended work loop logic into separate functions
acdlite Oct 24, 2022
bdd3d08
Extract logic for detecting bad fallback to helper
acdlite Oct 24, 2022
2ac77aa
Clean up vestige of useOpaqueIdentifier (#25587)
acdlite Oct 30, 2022
765805b
Fix type check for null (#25595)
sebmarkbage Oct 31, 2022
ab075a2
Do not unmount layout effects on initial Offscreen mount (#25592)
sammy-SC Nov 1, 2022
6883d79
[ServerRenderer] Setup for adding data attributes streaming format (#…
mofeiZ Nov 1, 2022
36426e6
Allow uncached IO to stablize (#25561)
acdlite Nov 1, 2022
5f7ef8c
[Float] handle resource Resource creation inside svg context (#25599)
gnoff Nov 1, 2022
8e69bc4
Make host context use null as empty and only error in dev (#25609)
sebmarkbage Nov 1, 2022
4ea063b
refactor isHostResourceType to not receive the context from reconcile…
gnoff Nov 2, 2022
1a90262
Unwrap sync resolved thenables without suspending (#25615)
acdlite Nov 2, 2022
1a08f14
[ServerRenderer] Move fizz external runtime implementation to react-d…
mofeiZ Nov 3, 2022
df61e70
Remove check in renderDidSuspendDelayIfPossible (#25630)
acdlite Nov 3, 2022
4bd245e
Do not unmount layout effects if ancestor Offscreen is hidden (#25628)
sammy-SC Nov 4, 2022
18dff79
[DevTools] add support for HostSingleton & HostResource (#25616)
mondaychen Nov 7, 2022
1e3e30d
Fix useSyncExternalStore dropped update when state is dispatched in r…
pandaiolo Nov 8, 2022
d1e35c7
Don't disappear layout effects unnecessarily (#25660)
sammy-SC Nov 10, 2022
c54e354
[DevTools] bug fix for Hydrating fibers (#25663)
mondaychen Nov 11, 2022
e1dd0a2
Remove recoverable error when a sync update flows into a dehydrated b…
sebmarkbage Nov 16, 2022
c343f80
[react-float] feature detect getRootNode (#25689)
kassens Nov 16, 2022
355dd7d
Bump loader-utils from 2.0.0 to 2.0.4 in /fixtures/flight (#25694)
dependabot[bot] Nov 16, 2022
db8a3fc
Bump loader-utils from 1.4.0 to 1.4.2 in /fixtures/fizz (#25680)
dependabot[bot] Nov 16, 2022
d65b88d
Eagerly initialize an mutable object for instance.refs (#25696)
sebmarkbage Nov 16, 2022
07f46ec
Turn on key spread warning in jsx-runtime for everyone (#25697)
sebmarkbage Nov 16, 2022
6fb8133
Turn on string ref deprecation warning for everybody (not codemoddabl…
eps1lon Nov 17, 2022
7b17f7b
Enable warning for defaultProps on function components for everyone (…
sebmarkbage Nov 17, 2022
44c4e6f
Force unwind work loop during selective hydration (#25695)
acdlite Nov 17, 2022
9dfbd9f
use: Don't suspend if there are pending updates
acdlite Nov 3, 2022
4a2d86b
Don't reset work loop until stack is unwound
acdlite Nov 2, 2022
5eb78d0
Pass ThenableState to replaySuspendedUnitOfWork
acdlite Nov 2, 2022
4387d75
Allow more hooks to be added when replaying mount
acdlite Nov 3, 2022
33e3d28
Reuse hooks when replaying a suspended component
acdlite Nov 3, 2022
6b4c031
Check thenable instead of thenableState
acdlite Nov 3, 2022
f284d9f
Track ThenableState alongside other hooks
acdlite Nov 3, 2022
f31005d
Add SyncHydrationLane (#25698)
tyao1 Nov 17, 2022
b0a6f66
Test for double invoke effect of useLayoutEffect
kassens Nov 17, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
[Float] fix coordination of resource identity and hydration (facebook…
…#25569)

there are a few bugs where dom representations from SSR aren't
identified as Resources when they should be.

There are 3 semantics

Resource -> hoist to head, deduping, etc...
hydratable Component -> SSR'd and hydrated in place
non-hydratable Component -> never SSR'd, never hydrated, always inserted
on the client

this last category is small
(non stylesheet) links with onLoad and/or onError
async scripts with onLoad and/or onError

The reason we have this distinction for now is we need every SSR'd async
script to be assumable to be a Resource. we don't currently encode
onLoad on the server and so we couldn't otherwise tell if an async
script is a Resource or is an async script with an onLoad which would
not be a resource. To avoid this ambiguity we never emit the scripts in
SSR and assume they need to be inserted on the client.

We can explore changes to these semantics in the future or possibly
encode some identifier when we want to opt out of resource semantics but
still SSR the link or script.
  • Loading branch information
gnoff authored Oct 27, 2022
commit 17204056d5ec0a53a74979321c519137d215cf64
4 changes: 2 additions & 2 deletions packages/react-dom-bindings/src/client/ReactDOMFloatClient.js
Original file line number Diff line number Diff line change
Expand Up @@ -1465,9 +1465,9 @@ export function isHostResourceType(type: string, props: Props): boolean {
}
return (async: any) && typeof src === 'string' && !onLoad && !onError;
}
case 'noscript':
case 'template':
case 'style':
case 'noscript': {
case 'style': {
if (__DEV__) {
if (resourceFormOnly) {
console.error(
Expand Down
52 changes: 33 additions & 19 deletions packages/react-dom-bindings/src/client/ReactDOMHostConfig.js
Original file line number Diff line number Diff line change
Expand Up @@ -802,7 +802,15 @@ export const supportsHydration = true;
// inserted without breaking hydration
export function isHydratable(type: string, props: Props): boolean {
if (enableFloat) {
if (type === 'script') {
if (type === 'link') {
if (
(props: any).rel === 'stylesheet' &&
typeof (props: any).precedence !== 'string'
) {
return true;
}
return false;
} else if (type === 'script') {
const {async, onLoad, onError} = (props: any);
return !(async && (onLoad || onError));
}
Expand Down Expand Up @@ -902,16 +910,25 @@ function getNextHydratable(node) {
if (nodeType === ELEMENT_NODE) {
const element: Element = (node: any);
switch (element.tagName) {
case 'TITLE':
case 'META':
case 'BASE':
case 'HTML':
case 'HEAD':
case 'BODY': {
continue;
}
case 'LINK': {
const linkEl: HTMLLinkElement = (element: any);
const rel = linkEl.rel;
// All links that are server rendered are resources except
// stylesheets that do not have a precedence
if (
rel === 'preload' ||
(rel === 'stylesheet' && linkEl.hasAttribute('data-precedence'))
linkEl.rel === 'stylesheet' &&
!linkEl.hasAttribute('data-precedence')
) {
continue;
break;
}
break;
continue;
}
case 'STYLE': {
const styleEl: HTMLStyleElement = (element: any);
Expand All @@ -927,12 +944,6 @@ function getNextHydratable(node) {
}
break;
}
case 'TITLE':
case 'HTML':
case 'HEAD':
case 'BODY': {
continue;
}
}
break;
} else if (nodeType === TEXT_NODE) {
Expand All @@ -942,18 +953,21 @@ function getNextHydratable(node) {
if (nodeType === ELEMENT_NODE) {
const element: Element = (node: any);
switch (element.tagName) {
case 'TITLE':
case 'META':
case 'BASE': {
continue;
}
case 'LINK': {
const linkEl: HTMLLinkElement = (element: any);
const rel = linkEl.rel;
// All links that are server rendered are resources except
// stylesheets that do not have a precedence
if (
rel === 'preload' ||
(rel === 'stylesheet' && linkEl.hasAttribute('data-precedence'))
linkEl.rel === 'stylesheet' &&
!linkEl.hasAttribute('data-precedence')
) {
continue;
break;
}
break;
}
case 'TITLE': {
continue;
}
case 'STYLE': {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -863,7 +863,7 @@ export function resourcesFromLink(props: Props): boolean {
}
}
if (props.onLoad || props.onError) {
return false;
return true;
}

const sizes = typeof props.sizes === 'string' ? props.sizes : '';
Expand Down
58 changes: 58 additions & 0 deletions packages/react-dom/src/__tests__/ReactDOMFloat-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -281,6 +281,64 @@ describe('ReactDOMFloat', () => {
});
}

// @gate enableFloat
it('can hydrate non Resources in head when Resources are also inserted there', async () => {
await actIntoEmptyDocument(() => {
const {pipe} = ReactDOMFizzServer.renderToPipeableStream(
<html>
<head>
<meta property="foo" content="bar" />
<link rel="foo" href="bar" onLoad={() => {}} />
<title>foo</title>
<base target="foo" href="bar" />
<script async={true} src="foo" onLoad={() => {}} />
</head>
<body>foo</body>
</html>,
);
pipe(writable);
});
expect(getMeaningfulChildren(document)).toEqual(
<html>
<head>
<base target="foo" href="bar" />
<link rel="preload" href="foo" as="script" />
<meta property="foo" content="bar" />
<title>foo</title>
</head>
<body>foo</body>
</html>,
);

ReactDOMClient.hydrateRoot(
document,
<html>
<head>
<meta property="foo" content="bar" />
<link rel="foo" href="bar" onLoad={() => {}} />
<title>foo</title>
<base target="foo" href="bar" />
<script async={true} src="foo" onLoad={() => {}} />
</head>
<body>foo</body>
</html>,
);
expect(Scheduler).toFlushWithoutYielding();
expect(getMeaningfulChildren(document)).toEqual(
<html>
<head>
<base target="foo" href="bar" />
<link rel="preload" href="foo" as="script" />
<meta property="foo" content="bar" />
<title>foo</title>
<link rel="foo" href="bar" />
<script async="" src="foo" />
</head>
<body>foo</body>
</html>,
);
});

// @gate enableFloat || !__DEV__
it('warns if you render resource-like elements above <head> or <body>', async () => {
const root = ReactDOMClient.createRoot(document);
Expand Down