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

Add lazy route property API, use for loading middleware #13294

Merged
merged 16 commits into from
Apr 1, 2025

Conversation

markdalgleish
Copy link
Member

@markdalgleish markdalgleish commented Mar 24, 2025

This removes route.unstable_lazyMiddleware in favour of a new object-based lazy API:

route.lazy = {
  unstable_middleware: () => { /* ... */ },
  ...etc
}

Lazy property functions can return null or undefined if the property is not available. This allows us to define lazy property functions even in cases where we don't know if the property is exported from the route module.

TODO:

  • Add initial implementation
  • Duplicate and modify all lazy tests to cover new API
  • Update middleware tests to new API
  • Add tests to cover undefined / null return values from lazy property functions
  • Update framework to use new lazy property API
  • Factor out shared logic for loading individual properties, share between client middleware case and route module case.
  • Refactor .then chaining in route.lazy() code path to use the new let promise = (async () => {})() pattern
  • Ensure lazy object keys are type checked correctly (e.g. you can't define route.lazy.lazy) and that unsupported keys aren't provided in autocomplete
  • Add docs
  • Add changeset

Copy link

changeset-bot bot commented Mar 24, 2025

🦋 Changeset detected

Latest commit: 8c9c4ad

The changes in this PR will be included in the next version bump.

This PR includes changesets to release 11 packages
Name Type
react-router Major
@react-router/architect Major
@react-router/cloudflare Major
@react-router/dev Major
react-router-dom Major
@react-router/express Major
@react-router/node Major
@react-router/serve Major
@react-router/fs-routes Major
@react-router/remix-routes-option-adapter Major
create-react-router Major

Not sure what this means? Click here to learn what changesets are.

Click here if you're a maintainer who wants to add another changeset to this PR

@markdalgleish markdalgleish changed the title WIP: Add lazy route property API Add lazy route property API, use for loading middleware Mar 27, 2025
@markdalgleish markdalgleish marked this pull request as ready for review March 27, 2025 04:25
@@ -726,70 +734,6 @@ describe("context/middleware", () => {
"Route property unstable_middleware is not a supported property to be returned from a lazy route function. This property will be ignored."
);
});

it("ignores lazy middleware returned from route.lazy", async () => {
Copy link
Member Author

Choose a reason for hiding this comment

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

This test was removed because route.unstable_lazyMiddleware was removed.

invariant(
clientMiddlewareModule?.unstable_clientMiddleware,
"No `unstable_clientMiddleware` export in chunk"
let lazyRoutePromise:
Copy link
Member Author

Choose a reason for hiding this comment

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

This allows us to dedupe calls to getLazyRoute across the different lazy property loaders. This is only required because we have additional logic beyond calling await import('...').

Comment on lines +4919 to 4926
if (typeof routeToUpdate.lazy === "object") {
routeToUpdate.lazy[key as keyof typeof routeToUpdate.lazy] = undefined;
if (
!isPropertyStaticallyDefined &&
!unsupportedLazyRouteFunctionKeys.has(
lazyRouteProperty as UnsupportedLazyRouteFunctionKey
)
Object.values(routeToUpdate.lazy).every((value) => value === undefined)
) {
routeUpdates[lazyRouteProperty] =
lazyRoute[lazyRouteProperty as keyof typeof lazyRoute];
routeToUpdate.lazy = undefined;
}
}
Copy link
Member Author

Choose a reason for hiding this comment

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

We perform this cleanup because the existing route.lazy logic uses the presence of a route.lazy value to decide whether to wait on a lazy call or not.

Comment on lines +4995 to +5010
if (isUnsupported) {
warning(
!isUnsupported,
"Route property " +
lazyRouteProperty +
" is not a supported property to be returned from a lazy route function. This property will be ignored."
);
} else if (isStaticallyDefined) {
warning(
!isStaticallyDefined,
`Route "${routeToUpdate.id}" has a static property "${lazyRouteProperty}" ` +
`defined but its lazy function is also returning a value for this property. ` +
`The lazy route property "${lazyRouteProperty}" will be ignored.`
);
} else {
routeUpdates[lazyRouteProperty] = lazyValue;
Copy link
Member Author

@markdalgleish markdalgleish Mar 30, 2025

Choose a reason for hiding this comment

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

This change was made to align the approach I landed on for loading lazy route properties. This ensures that you only get a single error message with the most relevant information. Most notably, if you have an unsupported key in your route.lazy object (e.g. id) then it's irrelevant that it's also statically defined since we'll never use the lazy value anyway.

export type LazyRouteObject<R extends AgnosticRouteObject> = {
[K in keyof R as K extends UnsupportedLazyRouteObjectKey
? never
: K]?: () => Promise<R[K] | null | undefined>;
Copy link
Member Author

Choose a reason for hiding this comment

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

Just calling this out as a discussion point since it's public API — lazy property functions are always async and can return null or undefined.

Copy link
Contributor

@brophdawg11 brophdawg11 left a comment

Choose a reason for hiding this comment

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

Code looks good! Gonna pull and do some testing now...

action: async () =>
(await import("./show.action.js")).action,
Component: async () =>
(await import("./show.component.js")).Component,
Copy link
Contributor

Choose a reason for hiding this comment

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

huh - I didn't realize our docs showed this type of granular file splitting. I feel like the more common use case is lazy: () => import("./route") so I would actually vote to document that version with a note on how to get more granular if/when needed

Copy link
Member Author

Choose a reason for hiding this comment

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

I feel this is the better API to document because it handles all properties, whereas the lazy function can't handle middleware. I also feel like this version is a bit clearer in terms of what's going on.

loader:
dataRoute.loader || !route.hasClientLoader
? undefined
: async () => {
Copy link
Contributor

Choose a reason for hiding this comment

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

I think I expected to see the if (route.clientLoaderModule) logic above moving down into route.lazy.loader as well? I think it would be another split in this conditional (using if/else for readability now that it has 3 branches). It might nicely colocate all of our loader initialization whereas we used to have 2 spots (outside/inside of lazy)

      // assume `dataRoute.lazy` is initialized as an empty object above - may not 
      // be the way we want to do it but it might read nicely as we build up the 
      // route object and let us colocate initialization logic per-field...
      if (!route.hasClientLoader) {
        // No `clientLoader` exists, use the `loader` to load styles and call the
        // server `loader` (if it exists) in parallel with `route.lazy` execution
        dataRoute.loader = (_: LoaderFunctionArgs, singleFetch?: unknown) =>
          prefetchStylesAndCallHandler(() => {
            return fetchServerLoader(singleFetch);
          });
      } else if (route.clientLoaderModule) {
        // A `clientLoader` module exists, load it with route.lazy.loader
        dataRoute.lazy.loader = async () => {
          invariant(route.clientLoaderModule);
          let { clientLoader } = await import(
            /* @vite-ignore */
            /* webpackIgnore: true */
            route.clientLoaderModule
          );
          return (args: LoaderFunctionArgs, singleFetch?: unknown) =>
            clientLoader({
              ...args,
              async serverLoader() {
                preventInvalidServerHandlerCall("loader", route);
                return fetchServerLoader(singleFetch);
              },
            });
        };
      } else {
        // No `clientLoader` module exists, load the `clientLoader` via the
        // full route module
        dataRoute.lazy.loader = async () => {
          let { clientLoader } = await getLazyRoute();
          invariant(clientLoader, "No `clientLoader` export found");
          // This below is the same as above so may be able to be shared.  
          // All that differs between this and the conditional branch above is 
          // where we get `clientLoader` from...
          return (args: LoaderFunctionArgs, singleFetch?: unknown) =>
            clientLoader({
              ...args,
              async serverLoader() {
                preventInvalidServerHandlerCall("loader", route);
                return fetchServerLoader(singleFetch);
              },
            });
        };
      }

Copy link
Member Author

Choose a reason for hiding this comment

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

Thanks for catching this, I've pushed a refactor that cleans this up.

m.route.lazy
? loadLazyRouteModule(m.route, mapRouteProperties, manifest)
? loadLazyRoute(m.route, manifest, mapRouteProperties)
Copy link
Contributor

Choose a reason for hiding this comment

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

This will run all lazy.* functions here right? I think that's the right approach to start - but curious if you were thinking we should take it farther/more granular and do the type of stuff we do in framework mode for library mode where we skip the hydrate fallback if we're not hydrating?

Copy link
Member Author

Choose a reason for hiding this comment

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

Yep, I wanted to do that as a follow-up.

@markdalgleish markdalgleish merged commit d07cefe into dev Apr 1, 2025
5 checks passed
@markdalgleish markdalgleish deleted the markdalgleish/lazy-route-properties branch April 1, 2025 02:48
Copy link
Contributor

github-actions bot commented Apr 2, 2025

🤖 Hello there,

We just published version 7.5.0-pre.0 which includes this pull request. If you'd like to take it for a test run please try it out and let us know what you think!

Thanks!

wilcoxmd added a commit to wilcoxmd/react-router that referenced this pull request Apr 3, 2025
…d-route-typegen

* upstream/dev: (56 commits)
  chore: format
  feat: unstable_subResourceIntegrity (remix-run#13163)
  Add lazy route property API, use for loading middleware (remix-run#13294)
  Add `rolldown-vite` playground (remix-run#13333)
  chore: Update version for release (remix-run#13322)
  Exit prerelease mode
  Update release notes
  Remove `react-router:override-optimize-deps` plugin (remix-run#13317)
  chore: format
  Add Wrangler v4 to `@react-router/dev` peer dep range (remix-run#13258)
  update changelog
  Handle custom base for critical CSS with Vite Env API (remix-run#13305)
  chore: Update version for release (pre) (remix-run#13312)
  Enter prerelease mode
  chore: format
  host parse (remix-run#13309)
  Fix middleware return types (remix-run#13311)
  more data docs
  fix Location link (remix-run#13291)
  Also fix the meta link
  ...
Copy link
Contributor

github-actions bot commented Apr 4, 2025

🤖 Hello there,

We just published version 7.5.0 which includes this pull request. If you'd like to take it for a test run please try it out and let us know what you think!

Thanks!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants