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

Warn if addTransitionType is called when there are no pending Actions #32793

Merged
merged 1 commit into from
Apr 1, 2025

Conversation

sebmarkbage
Copy link
Collaborator

Stacked on #32792.

It's tricky to associate a specific addTransitionType call to a specific startTransition call because we don't have AsyncContext in browsers yet. However, we can keep track if there are any async transitions running at all, and if not, warn. This should cover most cases.

This also errors when inside a React render which might be a legit way to associate a Transition Type to a specific render (e.g. based on props changing) but we want to be a more conservative about allowing that yet. If we wanted to support calling it in render, we might want to set which Transition object is currently rendering but it's still tricky if the render has async function components. So it might at least be restricted to sync components (like Hooks).

@react-sizebot
Copy link

react-sizebot commented Mar 30, 2025

Comparing: 0b1a9e9...9a6e020

Critical size changes

Includes critical production bundles, as well as any change greater than 2%:

Name +/- Base Current +/- gzip Base gzip Current gzip
oss-stable/react-dom/cjs/react-dom.production.js = 6.68 kB 6.68 kB = 1.83 kB 1.83 kB
oss-stable/react-dom/cjs/react-dom-client.production.js = 515.28 kB 515.28 kB = 91.81 kB 91.81 kB
oss-experimental/react-dom/cjs/react-dom.production.js = 6.69 kB 6.69 kB +0.05% 1.83 kB 1.83 kB
oss-experimental/react-dom/cjs/react-dom-client.production.js = 614.48 kB 614.48 kB = 108.71 kB 108.71 kB
facebook-www/ReactDOM-prod.classic.js = 648.21 kB 648.21 kB = 114.54 kB 114.54 kB
facebook-www/ReactDOM-prod.modern.js = 638.49 kB 638.49 kB = 112.95 kB 112.95 kB

Significant size changes

Includes any change greater than 0.2%:

Expand to show
Name +/- Base Current +/- gzip Base gzip Current gzip
oss-experimental/react/cjs/react.development.js +1.16% 48.40 kB 48.96 kB +1.06% 10.86 kB 10.97 kB
facebook-www/React-dev.modern.js +0.99% 55.02 kB 55.57 kB +0.96% 11.98 kB 12.10 kB
facebook-www/React-dev.classic.js +0.99% 55.02 kB 55.57 kB +0.96% 11.99 kB 12.10 kB
oss-experimental/react/cjs/react.react-server.development.js +0.62% 35.76 kB 35.98 kB +0.53% 8.53 kB 8.57 kB
oss-stable-semver/react/cjs/react.development.js +0.56% 44.81 kB 45.06 kB +0.48% 10.22 kB 10.27 kB
oss-stable/react/cjs/react.development.js +0.56% 44.84 kB 45.09 kB +0.51% 10.25 kB 10.30 kB
facebook-react-native/react/cjs/React-dev.js +0.49% 50.76 kB 51.01 kB +0.50% 11.29 kB 11.35 kB
test_utils/ReactAllWarnings.js +0.46% 64.11 kB 64.40 kB +0.27% 16.09 kB 16.13 kB

Generated by 🚫 dangerJS against 9a6e020

Copy link
Member

@rickhanlonii rickhanlonii left a comment

Choose a reason for hiding this comment

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

Nice f57175d

@jackpope can we add this to Troublshooting?

If it's zero, we can warn if you try to call addTransitionType.
@sebmarkbage sebmarkbage force-pushed the addtransitiontypeswarning branch from 6177366 to 9a6e020 Compare April 1, 2025 16:10
@sebmarkbage sebmarkbage merged commit deca965 into facebook:main Apr 1, 2025
2 of 4 checks passed
sebmarkbage added a commit that referenced this pull request Apr 1, 2025
Stacked on #32793.

This is meant to model the intended semantics of `addTransitionType`
better. The previous hack just consumed all transition types when any
root committed so it could steal them from other roots. Really each root
should get its own set. Really each transition lane should get its own
set.

We can't implement the full ideal semantics yet because 1) we currently
entangle transition lanes 2) we lack `AsyncContext` on the client so for
async actions we can't associate a `addTransitionType` call to a
specific `startTransition`.

This starts by modeling Transition Types to be stored on the Transition
instance. Conceptually they belong to the Transition instance of that
`startTransition` they belong to. That instance is otherwise mostly just
used for Transition Tracing but it makes sense that those would be able
to be passed the Transition Types for that specific instance.

Nested `startTransition` need to get entangled. So that this
`addTransitionType` can be associated with the `setState`:

```js
startTransition(() => {
  startTransition(() => {
    addTransitionType(...)
  });
  setState(...);
});
```

Ideally we'd probably just use the same Transition instance itself since
these are conceptually all part of one entangled one. But transition
tracing uses multiple names and start times. Unclear what we want to do
with that. So I kept separate instances but shared `types` set.

Next I collect the types added during a `startTransition` to any root
scheduled with a Transition. This should really be collected one set per
Transition lane in a `LaneMap`. In fact, the information would already
be there if Transition Tracing was always enabled because it tracks all
Transition instances per lane. For now I just keep track of one set for
all Transition lanes. Maybe we should only add it if a `setState` was
done on this root in this particular `startTransition` call rather
having already scheduled any Transition earlier.

While async transitions are entangled, we don't know if there will be a
startTransition+setState on a new root in the future. Therefore, we
collect all transition types while this is happening and if a new root
gets startTransition+setState they get added to that root.

```js
startTransition(async () => {
  addTransitionType(...)
  await ...;
  setState(...);
});
```
github-actions bot pushed a commit that referenced this pull request Apr 1, 2025
…#32793)

Stacked on #32792.

It's tricky to associate a specific `addTransitionType` call to a
specific `startTransition` call because we don't have `AsyncContext` in
browsers yet. However, we can keep track if there are any async
transitions running at all, and if not, warn. This should cover most
cases.

This also errors when inside a React render which might be a legit way
to associate a Transition Type to a specific render (e.g. based on props
changing) but we want to be a more conservative about allowing that yet.
If we wanted to support calling it in render, we might want to set which
Transition object is currently rendering but it's still tricky if the
render has `async function` components. So it might at least be
restricted to sync components (like Hooks).

DiffTrain build for [deca965](deca965)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
CLA Signed React Core Team Opened by a member of the React Core Team
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants