-
-
Notifications
You must be signed in to change notification settings - Fork 10.5k
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
Conversation
🦋 Changeset detectedLatest commit: 8c9c4ad The changes in this PR will be included in the next version bump. This PR includes changesets to release 11 packages
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 |
@@ -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 () => { |
There was a problem hiding this comment.
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: |
There was a problem hiding this comment.
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('...')
.
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; | ||
} | ||
} |
There was a problem hiding this comment.
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.
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; |
There was a problem hiding this comment.
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>; |
There was a problem hiding this comment.
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
.
There was a problem hiding this 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, |
There was a problem hiding this comment.
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
There was a problem hiding this comment.
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 () => { |
There was a problem hiding this comment.
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);
},
});
};
}
There was a problem hiding this comment.
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) |
There was a problem hiding this comment.
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?
There was a problem hiding this comment.
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.
🤖 Hello there, We just published version Thanks! |
…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 ...
🤖 Hello there, We just published version Thanks! |
This removes
route.unstable_lazyMiddleware
in favour of a new object-based lazy API:Lazy property functions can return
null
orundefined
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:
undefined
/null
return values from lazy property functions.then
chaining inroute.lazy()
code path to use the newlet promise = (async () => {})()
patternroute.lazy.lazy
) and that unsupported keys aren't provided in autocomplete