-
Notifications
You must be signed in to change notification settings - Fork 27.3k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
refactor: added more strict app segment config parsing
- Loading branch information
Showing
11 changed files
with
564 additions
and
312 deletions.
There are no files selected for viewing
110 changes: 110 additions & 0 deletions
110
packages/next/src/build/app-segments/app-segment-config.test.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,110 @@ | ||
import { AppSegmentConfigSchema } from './app-segment-config' | ||
|
||
describe('AppConfigSchema', () => { | ||
it('should only support zero, a positive number or false for revalidate', () => { | ||
const valid = [0, 1, 100, false] | ||
|
||
for (const value of valid) { | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ revalidate: value }).success | ||
).toBe(true) | ||
} | ||
|
||
const invalid = [-1, -100, true] | ||
|
||
for (const value of invalid) { | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ revalidate: value }).success | ||
).toBe(false) | ||
} | ||
}) | ||
|
||
it('should support an empty config', () => { | ||
expect(AppSegmentConfigSchema.safeParse({}).success).toBe(true) | ||
}) | ||
|
||
it('should support a boolean for dynamicParams', () => { | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ dynamicParams: true }).success | ||
).toBe(true) | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ dynamicParams: false }).success | ||
).toBe(true) | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ dynamicParams: 'foo' }).success | ||
).toBe(false) | ||
}) | ||
|
||
it('should support "auto" | "force-dynamic" | "error" | "force-static" for dynamic', () => { | ||
expect(AppSegmentConfigSchema.safeParse({ dynamic: 'auto' }).success).toBe( | ||
true | ||
) | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ dynamic: 'force-dynamic' }).success | ||
).toBe(true) | ||
expect(AppSegmentConfigSchema.safeParse({ dynamic: 'error' }).success).toBe( | ||
true | ||
) | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ dynamic: 'force-static' }).success | ||
).toBe(true) | ||
}) | ||
|
||
it('should support "edge" | "nodejs" for runtime', () => { | ||
expect(AppSegmentConfigSchema.safeParse({ runtime: 'edge' }).success).toBe( | ||
true | ||
) | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ runtime: 'nodejs' }).success | ||
).toBe(true) | ||
expect(AppSegmentConfigSchema.safeParse({ runtime: 'foo' }).success).toBe( | ||
false | ||
) | ||
}) | ||
|
||
it('should support a positive number or zero for maxDuration', () => { | ||
expect(AppSegmentConfigSchema.safeParse({ maxDuration: 0 }).success).toBe( | ||
true | ||
) | ||
expect(AppSegmentConfigSchema.safeParse({ maxDuration: 100 }).success).toBe( | ||
true | ||
) | ||
expect(AppSegmentConfigSchema.safeParse({ maxDuration: -1 }).success).toBe( | ||
false | ||
) | ||
}) | ||
|
||
it('should support "force-cache" | "only-cache" for fetchCache', () => { | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ fetchCache: 'force-cache' }).success | ||
).toBe(true) | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ fetchCache: 'only-cache' }).success | ||
).toBe(true) | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ fetchCache: 'foo' }).success | ||
).toBe(false) | ||
}) | ||
|
||
it('should support a string or an array of strings for preferredRegion', () => { | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ preferredRegion: 'foo' }).success | ||
).toBe(true) | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ preferredRegion: ['foo', 'bar'] }) | ||
.success | ||
).toBe(true) | ||
}) | ||
|
||
it('should support a boolean for experimental_ppr', () => { | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ experimental_ppr: true }).success | ||
).toBe(true) | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ experimental_ppr: false }).success | ||
).toBe(true) | ||
expect( | ||
AppSegmentConfigSchema.safeParse({ experimental_ppr: 'foo' }).success | ||
).toBe(false) | ||
}) | ||
}) |
77 changes: 77 additions & 0 deletions
77
packages/next/src/build/app-segments/app-segment-config.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,77 @@ | ||
import { z } from 'next/dist/compiled/zod' | ||
|
||
/** | ||
* The schema for the dynamic behavior of a page. | ||
*/ | ||
export const AppSegmentConfigDynamicSchema = z.enum([ | ||
'auto', | ||
'error', | ||
'force-static', | ||
'force-dynamic', | ||
]) | ||
|
||
/** | ||
* The dynamic behavior of the page. | ||
*/ | ||
export type AppSegmentConfigDynamic = z.infer< | ||
typeof AppSegmentConfigDynamicSchema | ||
> | ||
|
||
/** | ||
* The schema for configuration for a page. | ||
*/ | ||
export const AppSegmentConfigSchema = z.object({ | ||
/** | ||
* The number of seconds to revalidate the page or false to disable revalidation. | ||
*/ | ||
revalidate: z | ||
.union([z.number().int().nonnegative(), z.literal(false)]) | ||
.optional(), | ||
|
||
/** | ||
* Whether the page supports dynamic parameters. | ||
*/ | ||
dynamicParams: z.boolean().optional(), | ||
|
||
/** | ||
* The dynamic behavior of the page. | ||
*/ | ||
dynamic: AppSegmentConfigDynamicSchema.optional(), | ||
|
||
/** | ||
* The caching behavior of the page. | ||
*/ | ||
fetchCache: z.enum(['force-cache', 'only-cache']).optional(), | ||
|
||
/** | ||
* The preferred region for the page. | ||
*/ | ||
preferredRegion: z.union([z.string(), z.array(z.string())]).optional(), | ||
|
||
/** | ||
* Whether the page supports partial prerendering. When true, the page will be | ||
* served using partial prerendering. This setting will only take affect if | ||
* it's enabled via the `experimental.ppr = "incremental"` option. | ||
*/ | ||
experimental_ppr: z.boolean().optional(), | ||
|
||
/** | ||
* The runtime to use for the page. | ||
*/ | ||
runtime: z.enum(['edge', 'nodejs']).optional(), | ||
|
||
/** | ||
* The maximum duration for the page in seconds. | ||
*/ | ||
maxDuration: z.number().int().nonnegative().optional(), | ||
}) | ||
|
||
/** | ||
* The configuration for a page. | ||
*/ | ||
export type AppSegmentConfig = z.infer<typeof AppSegmentConfigSchema> | ||
|
||
/** | ||
* The keys of the configuration for a page. | ||
*/ | ||
export const AppSegmentConfigSchemaKeys = AppSegmentConfigSchema.keyof().options |
182 changes: 182 additions & 0 deletions
182
packages/next/src/build/app-segments/collect-app-segments.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,182 @@ | ||
import type { LoadComponentsReturnType } from '../../server/load-components' | ||
import type { Params } from '../../server/request/params' | ||
import type { | ||
AppPageRouteModule, | ||
AppPageModule, | ||
} from '../../server/route-modules/app-page/module.compiled' | ||
import type { | ||
AppRouteRouteModule, | ||
AppRouteModule, | ||
} from '../../server/route-modules/app-route/module.compiled' | ||
import { | ||
type AppSegmentConfig, | ||
AppSegmentConfigSchema, | ||
} from './app-segment-config' | ||
|
||
import { InvariantError } from '../../shared/lib/invariant-error' | ||
import { | ||
isAppRouteRouteModule, | ||
isAppPageRouteModule, | ||
} from '../../server/route-modules/checks' | ||
import { isClientReference } from '../../lib/client-reference' | ||
import { getSegmentParam } from '../../server/app-render/get-segment-param' | ||
import { getLayoutOrPageModule } from '../../server/lib/app-dir-module' | ||
|
||
type GenerateStaticParams = (options: { params?: Params }) => Promise<Params[]> | ||
|
||
/** | ||
* Filters out segments that don't contribute to static generation. | ||
* | ||
* @param segments the segments to filter | ||
* @returns the filtered segments | ||
*/ | ||
function filterSegments(segments: AppSegment[]) { | ||
return segments.filter((result) => { | ||
return ( | ||
result.config || result.generateStaticParams || result.isDynamicSegment | ||
) | ||
}) | ||
} | ||
|
||
/** | ||
* Parses the app config and attaches it to the segment. | ||
*/ | ||
function attach(segment: AppSegment, userland: unknown) { | ||
// If the userland is not an object, then we can't do anything with it. | ||
if (typeof userland !== 'object' || userland === null) { | ||
return | ||
} | ||
|
||
// Try to parse the application configuration. If there were any keys, attach | ||
// it to the segment. | ||
const config = AppSegmentConfigSchema.safeParse(userland) | ||
if (config.success && Object.keys(config.data).length > 0) { | ||
segment.config = config.data | ||
} | ||
|
||
if ( | ||
'generateStaticParams' in userland && | ||
typeof userland.generateStaticParams === 'function' | ||
) { | ||
segment.generateStaticParams = | ||
userland.generateStaticParams as GenerateStaticParams | ||
} | ||
} | ||
|
||
export type AppSegment = { | ||
name: string | ||
param: string | undefined | ||
filePath: string | undefined | ||
config: AppSegmentConfig | undefined | ||
isDynamicSegment: boolean | ||
generateStaticParams: GenerateStaticParams | undefined | ||
} | ||
|
||
/** | ||
* Walks the loader tree and collects the generate parameters for each segment. | ||
* | ||
* @param routeModule the app page route module | ||
* @returns the segments for the app page route module | ||
*/ | ||
async function collectAppPageSegments(routeModule: AppPageRouteModule) { | ||
const segments: AppSegment[] = [] | ||
|
||
let current = routeModule.userland.loaderTree | ||
while (current) { | ||
const [name, parallelRoutes] = current | ||
const { mod: userland, filePath } = await getLayoutOrPageModule(current) | ||
|
||
const isClientComponent: boolean = userland && isClientReference(userland) | ||
const isDynamicSegment = /^\[.*\]$/.test(name) | ||
const param = isDynamicSegment ? getSegmentParam(name)?.param : undefined | ||
|
||
const segment: AppSegment = { | ||
name, | ||
param, | ||
filePath, | ||
config: undefined, | ||
isDynamicSegment, | ||
generateStaticParams: undefined, | ||
} | ||
|
||
// Only server components can have app segment configurations. If this isn't | ||
// an object, then we should skip it. This can happen when parsing the | ||
// error components. | ||
if (!isClientComponent) { | ||
attach(segment, userland) | ||
} | ||
|
||
segments.push(segment) | ||
|
||
// Use this route's parallel route children as the next segment. | ||
current = parallelRoutes.children | ||
} | ||
|
||
return filterSegments(segments) | ||
} | ||
|
||
/** | ||
* Collects the segments for a given app route module. | ||
* | ||
* @param routeModule the app route module | ||
* @returns the segments for the app route module | ||
*/ | ||
function collectAppRouteSegments( | ||
routeModule: AppRouteRouteModule | ||
): AppSegment[] { | ||
// Get the pathname parts, slice off the first element (which is empty). | ||
const parts = routeModule.definition.pathname.split('/').slice(1) | ||
if (parts.length === 0) { | ||
throw new InvariantError('Expected at least one segment') | ||
} | ||
|
||
// Generate all the segments. | ||
const segments: AppSegment[] = parts.map((name) => { | ||
const isDynamicSegment = /^\[.*\]$/.test(name) | ||
const param = isDynamicSegment ? getSegmentParam(name)?.param : undefined | ||
|
||
return { | ||
name, | ||
param, | ||
filePath: undefined, | ||
isDynamicSegment, | ||
config: undefined, | ||
generateStaticParams: undefined, | ||
} | ||
}) | ||
|
||
// We know we have at least one, we verified this above. We should get the | ||
// last segment which represents the root route module. | ||
const segment = segments[segments.length - 1] | ||
|
||
segment.filePath = routeModule.definition.filename | ||
|
||
// Extract the segment config from the userland module. | ||
attach(segment, routeModule.userland) | ||
|
||
return filterSegments(segments) | ||
} | ||
|
||
/** | ||
* Collects the segments for a given route module. | ||
* | ||
* @param components the loaded components | ||
* @returns the segments for the route module | ||
*/ | ||
export function collectSegments({ | ||
routeModule, | ||
}: LoadComponentsReturnType<AppPageModule | AppRouteModule>): | ||
| Promise<AppSegment[]> | ||
| AppSegment[] { | ||
if (isAppRouteRouteModule(routeModule)) { | ||
return collectAppRouteSegments(routeModule) | ||
} | ||
|
||
if (isAppPageRouteModule(routeModule)) { | ||
return collectAppPageSegments(routeModule) | ||
} | ||
|
||
throw new InvariantError( | ||
'Expected a route module to be one of app route or page' | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.