mirror of
https://github.com/withastro/astro.git
synced 2025-01-20 22:12:38 -05:00
fix(i18n): fallback should create consistent directories (#9142)
This commit is contained in:
parent
306781795d
commit
7d55cf68d8
7 changed files with 252 additions and 228 deletions
5
.changeset/curvy-sheep-lick.md
Normal file
5
.changeset/curvy-sheep-lick.md
Normal file
|
@ -0,0 +1,5 @@
|
|||
---
|
||||
'astro': patch
|
||||
---
|
||||
|
||||
Consistely emit fallback routes in the correct folders.
|
|
@ -13,6 +13,7 @@ import type {
|
|||
MiddlewareEndpointHandler,
|
||||
RouteData,
|
||||
RouteType,
|
||||
SSRElement,
|
||||
SSRError,
|
||||
SSRLoadedRenderer,
|
||||
SSRManifest,
|
||||
|
@ -261,21 +262,49 @@ async function generatePage(
|
|||
builtPaths: Set<string>,
|
||||
pipeline: BuildPipeline
|
||||
) {
|
||||
let timeStart = performance.now();
|
||||
// prepare information we need
|
||||
const logger = pipeline.getLogger();
|
||||
const config = pipeline.getConfig();
|
||||
const manifest = pipeline.getManifest();
|
||||
const pageModulePromise = ssrEntry.page;
|
||||
const onRequest = ssrEntry.onRequest;
|
||||
const pageInfo = getPageDataByComponent(pipeline.getInternals(), pageData.route.component);
|
||||
|
||||
// may be used in the future for handling rel=modulepreload, rel=icon, rel=manifest etc.
|
||||
const linkIds: [] = [];
|
||||
const scripts = pageInfo?.hoistedScript ?? null;
|
||||
const styles = pageData.styles
|
||||
// Calculate information of the page, like scripts, links and styles
|
||||
const hoistedScripts = pageInfo?.hoistedScript ?? null;
|
||||
const moduleStyles = pageData.styles
|
||||
.sort(cssOrder)
|
||||
.map(({ sheet }) => sheet)
|
||||
.reduce(mergeInlineCss, []);
|
||||
|
||||
const pageModulePromise = ssrEntry.page;
|
||||
const onRequest = ssrEntry.onRequest;
|
||||
// may be used in the future for handling rel=modulepreload, rel=icon, rel=manifest etc.
|
||||
const links = new Set<never>();
|
||||
const styles = createStylesheetElementSet(moduleStyles, manifest.base, manifest.assetsPrefix);
|
||||
const scripts = createModuleScriptsSet(
|
||||
hoistedScripts ? [hoistedScripts] : [],
|
||||
manifest.base,
|
||||
manifest.assetsPrefix
|
||||
);
|
||||
if (pipeline.getSettings().scripts.some((script) => script.stage === 'page')) {
|
||||
const hashedFilePath = pipeline.getInternals().entrySpecifierToBundleMap.get(PAGE_SCRIPT_ID);
|
||||
if (typeof hashedFilePath !== 'string') {
|
||||
throw new Error(`Cannot find the built path for ${PAGE_SCRIPT_ID}`);
|
||||
}
|
||||
const src = createAssetLink(hashedFilePath, manifest.base, manifest.assetsPrefix);
|
||||
scripts.add({
|
||||
props: { type: 'module', src },
|
||||
children: '',
|
||||
});
|
||||
}
|
||||
// Add all injected scripts to the page.
|
||||
for (const script of pipeline.getSettings().scripts) {
|
||||
if (script.stage === 'head-inline') {
|
||||
scripts.add({
|
||||
props: {},
|
||||
children: script.content,
|
||||
});
|
||||
}
|
||||
}
|
||||
// prepare the middleware
|
||||
const i18nMiddleware = createI18nMiddleware(
|
||||
pipeline.getManifest().i18n,
|
||||
pipeline.getManifest().base,
|
||||
|
@ -309,43 +338,24 @@ async function generatePage(
|
|||
return;
|
||||
}
|
||||
|
||||
const generationOptions: Readonly<GeneratePathOptions> = {
|
||||
pageData,
|
||||
linkIds,
|
||||
scripts,
|
||||
styles,
|
||||
mod: pageModule,
|
||||
};
|
||||
|
||||
const icon =
|
||||
pageData.route.type === 'page' ||
|
||||
pageData.route.type === 'redirect' ||
|
||||
pageData.route.type === 'fallback'
|
||||
? green('▶')
|
||||
: magenta('λ');
|
||||
if (isRelativePath(pageData.route.component)) {
|
||||
logger.info(null, `${icon} ${pageData.route.route}`);
|
||||
for (const fallbackRoute of pageData.route.fallbackRoutes) {
|
||||
logger.info(null, `${icon} ${fallbackRoute.route}`);
|
||||
// Now we explode the routes. A route render itself, and it can render its fallbacks (i18n routing)
|
||||
for (const route of eachRouteInRouteData(pageData)) {
|
||||
// Get paths for the route, calling getStaticPaths if needed.
|
||||
const paths = await getPathsForRoute(route, pageModule, pipeline, builtPaths);
|
||||
let timeStart = performance.now();
|
||||
let prevTimeEnd = timeStart;
|
||||
for (let i = 0; i < paths.length; i++) {
|
||||
const path = paths[i];
|
||||
pipeline.getEnvironment().logger.debug('build', `Generating: ${path}`);
|
||||
await generatePath(path, pipeline, route, links, scripts, styles, pageModule);
|
||||
const timeEnd = performance.now();
|
||||
const timeChange = getTimeStat(prevTimeEnd, timeEnd);
|
||||
const timeIncrease = `(+${timeChange})`;
|
||||
const filePath = getOutputFilename(pipeline.getConfig(), path, pageData.route.type);
|
||||
const lineIcon = i === paths.length - 1 ? '└─' : '├─';
|
||||
logger.info(null, ` ${cyan(lineIcon)} ${dim(filePath)} ${dim(timeIncrease)}`);
|
||||
prevTimeEnd = timeEnd;
|
||||
}
|
||||
} else {
|
||||
logger.info(null, `${icon} ${pageData.route.component}`);
|
||||
}
|
||||
|
||||
// Get paths for the route, calling getStaticPaths if needed.
|
||||
const paths = await getPathsForRoute(pageData, pageModule, pipeline, builtPaths);
|
||||
|
||||
let prevTimeEnd = timeStart;
|
||||
for (let i = 0; i < paths.length; i++) {
|
||||
const path = paths[i];
|
||||
await generatePath(path, generationOptions, pipeline);
|
||||
const timeEnd = performance.now();
|
||||
const timeChange = getTimeStat(prevTimeEnd, timeEnd);
|
||||
const timeIncrease = `(+${timeChange})`;
|
||||
const filePath = getOutputFilename(pipeline.getConfig(), path, pageData.route.type);
|
||||
const lineIcon = i === paths.length - 1 ? '└─' : '├─';
|
||||
logger.info(null, ` ${cyan(lineIcon)} ${dim(filePath)} ${dim(timeIncrease)}`);
|
||||
prevTimeEnd = timeEnd;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -357,7 +367,7 @@ function* eachRouteInRouteData(data: PageBuildData) {
|
|||
}
|
||||
|
||||
async function getPathsForRoute(
|
||||
pageData: PageBuildData,
|
||||
route: RouteData,
|
||||
mod: ComponentInstance,
|
||||
pipeline: BuildPipeline,
|
||||
builtPaths: Set<string>
|
||||
|
@ -365,68 +375,64 @@ async function getPathsForRoute(
|
|||
const opts = pipeline.getStaticBuildOptions();
|
||||
const logger = pipeline.getLogger();
|
||||
let paths: Array<string> = [];
|
||||
if (pageData.route.pathname) {
|
||||
paths.push(pageData.route.pathname);
|
||||
builtPaths.add(pageData.route.pathname);
|
||||
for (const virtualRoute of pageData.route.fallbackRoutes) {
|
||||
if (route.pathname) {
|
||||
paths.push(route.pathname);
|
||||
builtPaths.add(route.pathname);
|
||||
for (const virtualRoute of route.fallbackRoutes) {
|
||||
if (virtualRoute.pathname) {
|
||||
paths.push(virtualRoute.pathname);
|
||||
builtPaths.add(virtualRoute.pathname);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
for (const route of eachRouteInRouteData(pageData)) {
|
||||
const staticPaths = await callGetStaticPaths({
|
||||
mod,
|
||||
route,
|
||||
routeCache: opts.routeCache,
|
||||
logger,
|
||||
ssr: isServerLikeOutput(opts.settings.config),
|
||||
}).catch((err) => {
|
||||
logger.debug('build', `├── ${colors.bold(colors.red('✗'))} ${route.component}`);
|
||||
throw err;
|
||||
const staticPaths = await callGetStaticPaths({
|
||||
mod,
|
||||
route,
|
||||
routeCache: opts.routeCache,
|
||||
logger,
|
||||
ssr: isServerLikeOutput(opts.settings.config),
|
||||
}).catch((err) => {
|
||||
logger.debug('build', `├── ${colors.bold(colors.red('✗'))} ${route.component}`);
|
||||
throw err;
|
||||
});
|
||||
|
||||
const label = staticPaths.length === 1 ? 'page' : 'pages';
|
||||
logger.debug(
|
||||
'build',
|
||||
`├── ${colors.bold(colors.green('✔'))} ${route.component} → ${colors.magenta(
|
||||
`[${staticPaths.length} ${label}]`
|
||||
)}`
|
||||
);
|
||||
|
||||
paths = staticPaths
|
||||
.map((staticPath) => {
|
||||
try {
|
||||
return route.generate(staticPath.params);
|
||||
} catch (e) {
|
||||
if (e instanceof TypeError) {
|
||||
throw getInvalidRouteSegmentError(e, route, staticPath);
|
||||
}
|
||||
throw e;
|
||||
}
|
||||
})
|
||||
.filter((staticPath) => {
|
||||
// The path hasn't been built yet, include it
|
||||
if (!builtPaths.has(removeTrailingForwardSlash(staticPath))) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// The path was already built once. Check the manifest to see if
|
||||
// this route takes priority for the final URL.
|
||||
// NOTE: The same URL may match multiple routes in the manifest.
|
||||
// Routing priority needs to be verified here for any duplicate
|
||||
// paths to ensure routing priority rules are enforced in the final build.
|
||||
const matchedRoute = matchRoute(staticPath, opts.manifest);
|
||||
return matchedRoute === route;
|
||||
});
|
||||
|
||||
const label = staticPaths.length === 1 ? 'page' : 'pages';
|
||||
logger.debug(
|
||||
'build',
|
||||
`├── ${colors.bold(colors.green('✔'))} ${route.component} → ${colors.magenta(
|
||||
`[${staticPaths.length} ${label}]`
|
||||
)}`
|
||||
);
|
||||
|
||||
paths.push(
|
||||
...staticPaths
|
||||
.map((staticPath) => {
|
||||
try {
|
||||
return route.generate(staticPath.params);
|
||||
} catch (e) {
|
||||
if (e instanceof TypeError) {
|
||||
throw getInvalidRouteSegmentError(e, route, staticPath);
|
||||
}
|
||||
throw e;
|
||||
}
|
||||
})
|
||||
.filter((staticPath) => {
|
||||
// The path hasn't been built yet, include it
|
||||
if (!builtPaths.has(removeTrailingForwardSlash(staticPath))) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// The path was already built once. Check the manifest to see if
|
||||
// this route takes priority for the final URL.
|
||||
// NOTE: The same URL may match multiple routes in the manifest.
|
||||
// Routing priority needs to be verified here for any duplicate
|
||||
// paths to ensure routing priority rules are enforced in the final build.
|
||||
const matchedRoute = matchRoute(staticPath, opts.manifest);
|
||||
return matchedRoute === route;
|
||||
})
|
||||
);
|
||||
|
||||
// Add each path to the builtPaths set, to avoid building it again later.
|
||||
for (const staticPath of paths) {
|
||||
builtPaths.add(removeTrailingForwardSlash(staticPath));
|
||||
}
|
||||
// Add each path to the builtPaths set, to avoid building it again later.
|
||||
for (const staticPath of paths) {
|
||||
builtPaths.add(removeTrailingForwardSlash(staticPath));
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -509,106 +515,92 @@ function getUrlForPath(
|
|||
return url;
|
||||
}
|
||||
|
||||
async function generatePath(pathname: string, gopts: GeneratePathOptions, pipeline: BuildPipeline) {
|
||||
async function generatePath(
|
||||
pathname: string,
|
||||
pipeline: BuildPipeline,
|
||||
route: RouteData,
|
||||
links: Set<never>,
|
||||
scripts: Set<SSRElement>,
|
||||
styles: Set<SSRElement>,
|
||||
mod: ComponentInstance
|
||||
) {
|
||||
const manifest = pipeline.getManifest();
|
||||
const { mod, scripts: hoistedScripts, styles: _styles, pageData } = gopts;
|
||||
const logger = pipeline.getLogger();
|
||||
pipeline.getEnvironment().logger.debug('build', `Generating: ${pathname}`);
|
||||
|
||||
for (const route of eachRouteInRouteData(pageData)) {
|
||||
// This adds the page name to the array so it can be shown as part of stats.
|
||||
if (route.type === 'page') {
|
||||
addPageName(pathname, pipeline.getStaticBuildOptions());
|
||||
const icon =
|
||||
route.type === 'page' || route.type === 'redirect' || route.type === 'fallback'
|
||||
? green('▶')
|
||||
: magenta('λ');
|
||||
if (isRelativePath(route.component)) {
|
||||
logger.info(null, `${icon} ${route.route}`);
|
||||
} else {
|
||||
logger.info(null, `${icon} ${route.component}`);
|
||||
}
|
||||
|
||||
// This adds the page name to the array so it can be shown as part of stats.
|
||||
if (route.type === 'page') {
|
||||
addPageName(pathname, pipeline.getStaticBuildOptions());
|
||||
}
|
||||
|
||||
const ssr = isServerLikeOutput(pipeline.getConfig());
|
||||
const url = getUrlForPath(
|
||||
pathname,
|
||||
pipeline.getConfig().base,
|
||||
pipeline.getStaticBuildOptions().origin,
|
||||
pipeline.getConfig().build.format,
|
||||
route.type
|
||||
);
|
||||
|
||||
const request = createRequest({
|
||||
url,
|
||||
headers: new Headers(),
|
||||
logger: pipeline.getLogger(),
|
||||
ssr,
|
||||
});
|
||||
const i18n = pipeline.getConfig().experimental.i18n;
|
||||
|
||||
const renderContext = await createRenderContext({
|
||||
pathname,
|
||||
request,
|
||||
componentMetadata: manifest.componentMetadata,
|
||||
scripts,
|
||||
styles,
|
||||
links,
|
||||
route,
|
||||
env: pipeline.getEnvironment(),
|
||||
mod,
|
||||
locales: i18n?.locales,
|
||||
routingStrategy: i18n?.routingStrategy,
|
||||
defaultLocale: i18n?.defaultLocale,
|
||||
});
|
||||
|
||||
let body: string | Uint8Array;
|
||||
let encoding: BufferEncoding | undefined;
|
||||
|
||||
let response: Response;
|
||||
try {
|
||||
response = await pipeline.renderRoute(renderContext, mod);
|
||||
} catch (err) {
|
||||
if (!AstroError.is(err) && !(err as SSRError).id && typeof err === 'object') {
|
||||
(err as SSRError).id = route.component;
|
||||
}
|
||||
throw err;
|
||||
}
|
||||
|
||||
pipeline.getEnvironment().logger.debug('build', `Generating: ${pathname}`);
|
||||
|
||||
// may be used in the future for handling rel=modulepreload, rel=icon, rel=manifest etc.
|
||||
const links = new Set<never>();
|
||||
const scripts = createModuleScriptsSet(
|
||||
hoistedScripts ? [hoistedScripts] : [],
|
||||
manifest.base,
|
||||
manifest.assetsPrefix
|
||||
);
|
||||
const styles = createStylesheetElementSet(_styles, manifest.base, manifest.assetsPrefix);
|
||||
|
||||
if (pipeline.getSettings().scripts.some((script) => script.stage === 'page')) {
|
||||
const hashedFilePath = pipeline.getInternals().entrySpecifierToBundleMap.get(PAGE_SCRIPT_ID);
|
||||
if (typeof hashedFilePath !== 'string') {
|
||||
throw new Error(`Cannot find the built path for ${PAGE_SCRIPT_ID}`);
|
||||
}
|
||||
const src = createAssetLink(hashedFilePath, manifest.base, manifest.assetsPrefix);
|
||||
scripts.add({
|
||||
props: { type: 'module', src },
|
||||
children: '',
|
||||
});
|
||||
if (response.status >= 300 && response.status < 400) {
|
||||
// If redirects is set to false, don't output the HTML
|
||||
if (!pipeline.getConfig().build.redirects) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Add all injected scripts to the page.
|
||||
for (const script of pipeline.getSettings().scripts) {
|
||||
if (script.stage === 'head-inline') {
|
||||
scripts.add({
|
||||
props: {},
|
||||
children: script.content,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
const ssr = isServerLikeOutput(pipeline.getConfig());
|
||||
const url = getUrlForPath(
|
||||
pathname,
|
||||
pipeline.getConfig().base,
|
||||
pipeline.getStaticBuildOptions().origin,
|
||||
pipeline.getConfig().build.format,
|
||||
route.type
|
||||
);
|
||||
|
||||
const request = createRequest({
|
||||
url,
|
||||
headers: new Headers(),
|
||||
logger: pipeline.getLogger(),
|
||||
ssr,
|
||||
});
|
||||
const i18n = pipeline.getConfig().experimental.i18n;
|
||||
const renderContext = await createRenderContext({
|
||||
pathname,
|
||||
request,
|
||||
componentMetadata: manifest.componentMetadata,
|
||||
scripts,
|
||||
styles,
|
||||
links,
|
||||
route,
|
||||
env: pipeline.getEnvironment(),
|
||||
mod,
|
||||
locales: i18n?.locales,
|
||||
routingStrategy: i18n?.routingStrategy,
|
||||
defaultLocale: i18n?.defaultLocale,
|
||||
});
|
||||
|
||||
let body: string | Uint8Array;
|
||||
let encoding: BufferEncoding | undefined;
|
||||
|
||||
let response: Response;
|
||||
try {
|
||||
response = await pipeline.renderRoute(renderContext, mod);
|
||||
} catch (err) {
|
||||
if (!AstroError.is(err) && !(err as SSRError).id && typeof err === 'object') {
|
||||
(err as SSRError).id = pageData.component;
|
||||
}
|
||||
throw err;
|
||||
}
|
||||
|
||||
if (response.status >= 300 && response.status < 400) {
|
||||
// If redirects is set to false, don't output the HTML
|
||||
if (!pipeline.getConfig().build.redirects) {
|
||||
return;
|
||||
}
|
||||
const locationSite = getRedirectLocationOrThrow(response.headers);
|
||||
const siteURL = pipeline.getConfig().site;
|
||||
const location = siteURL ? new URL(locationSite, siteURL) : locationSite;
|
||||
const fromPath = new URL(renderContext.request.url).pathname;
|
||||
// A short delay causes Google to interpret the redirect as temporary.
|
||||
// https://developers.google.com/search/docs/crawling-indexing/301-redirects#metarefresh
|
||||
const delay = response.status === 302 ? 2 : 0;
|
||||
body = `<!doctype html>
|
||||
const locationSite = getRedirectLocationOrThrow(response.headers);
|
||||
const siteURL = pipeline.getConfig().site;
|
||||
const location = siteURL ? new URL(locationSite, siteURL) : locationSite;
|
||||
const fromPath = new URL(renderContext.request.url).pathname;
|
||||
// A short delay causes Google to interpret the redirect as temporary.
|
||||
// https://developers.google.com/search/docs/crawling-indexing/301-redirects#metarefresh
|
||||
const delay = response.status === 302 ? 2 : 0;
|
||||
body = `<!doctype html>
|
||||
<title>Redirecting to: ${location}</title>
|
||||
<meta http-equiv="refresh" content="${delay};url=${location}">
|
||||
<meta name="robots" content="noindex">
|
||||
|
@ -616,27 +608,26 @@ async function generatePath(pathname: string, gopts: GeneratePathOptions, pipeli
|
|||
<body>
|
||||
<a href="${location}">Redirecting from <code>${fromPath}</code> to <code>${location}</code></a>
|
||||
</body>`;
|
||||
if (pipeline.getConfig().compressHTML === true) {
|
||||
body = body.replaceAll('\n', '');
|
||||
}
|
||||
// A dynamic redirect, set the location so that integrations know about it.
|
||||
if (route.type !== 'redirect') {
|
||||
route.redirect = location.toString();
|
||||
}
|
||||
} else {
|
||||
// If there's no body, do nothing
|
||||
if (!response.body) return;
|
||||
body = Buffer.from(await response.arrayBuffer());
|
||||
encoding = (response.headers.get('X-Astro-Encoding') as BufferEncoding | null) ?? 'utf-8';
|
||||
if (pipeline.getConfig().compressHTML === true) {
|
||||
body = body.replaceAll('\n', '');
|
||||
}
|
||||
|
||||
const outFolder = getOutFolder(pipeline.getConfig(), pathname, route.type);
|
||||
const outFile = getOutFile(pipeline.getConfig(), outFolder, pathname, route.type);
|
||||
route.distURL = outFile;
|
||||
|
||||
await fs.promises.mkdir(outFolder, { recursive: true });
|
||||
await fs.promises.writeFile(outFile, body, encoding);
|
||||
// A dynamic redirect, set the location so that integrations know about it.
|
||||
if (route.type !== 'redirect') {
|
||||
route.redirect = location.toString();
|
||||
}
|
||||
} else {
|
||||
// If there's no body, do nothing
|
||||
if (!response.body) return;
|
||||
body = Buffer.from(await response.arrayBuffer());
|
||||
encoding = (response.headers.get('X-Astro-Encoding') as BufferEncoding | null) ?? 'utf-8';
|
||||
}
|
||||
|
||||
const outFolder = getOutFolder(pipeline.getConfig(), pathname, route.type);
|
||||
const outFile = getOutFile(pipeline.getConfig(), outFolder, pathname, route.type);
|
||||
route.distURL = outFile;
|
||||
|
||||
await fs.promises.mkdir(outFolder, { recursive: true });
|
||||
await fs.promises.writeFile(outFile, body, encoding);
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -603,22 +603,22 @@ export function createRouteManifest(
|
|||
if (!hasRoute) {
|
||||
let pathname: string | undefined;
|
||||
let route: string;
|
||||
if (fallbackToLocale === i18n.defaultLocale) {
|
||||
if (
|
||||
fallbackToLocale === i18n.defaultLocale &&
|
||||
i18n.routingStrategy === 'prefix-other-locales'
|
||||
) {
|
||||
if (fallbackToRoute.pathname) {
|
||||
pathname = `/${fallbackFromLocale}${fallbackToRoute.pathname}`;
|
||||
}
|
||||
route = `/${fallbackFromLocale}${fallbackToRoute.route}`;
|
||||
} else {
|
||||
pathname = fallbackToRoute.pathname?.replace(
|
||||
`/${fallbackToLocale}`,
|
||||
`/${fallbackFromLocale}`
|
||||
);
|
||||
route = fallbackToRoute.route.replace(
|
||||
`/${fallbackToLocale}`,
|
||||
`/${fallbackFromLocale}`
|
||||
);
|
||||
pathname = fallbackToRoute.pathname
|
||||
?.replace(`/${fallbackToLocale}/`, `/${fallbackFromLocale}/`)
|
||||
.replace(`/${fallbackToLocale}`, `/${fallbackFromLocale}`);
|
||||
route = fallbackToRoute.route
|
||||
.replace(`/${fallbackToLocale}`, `/${fallbackFromLocale}`)
|
||||
.replace(`/${fallbackToLocale}/`, `/${fallbackFromLocale}/`);
|
||||
}
|
||||
|
||||
const segments = removeLeadingForwardSlash(route)
|
||||
.split(path.posix.sep)
|
||||
.filter(Boolean)
|
||||
|
@ -626,7 +626,7 @@ export function createRouteManifest(
|
|||
validateSegment(s);
|
||||
return getParts(s, route);
|
||||
});
|
||||
|
||||
const generate = getRouteGenerator(segments, config.trailingSlash);
|
||||
const index = routes.findIndex((r) => r === fallbackToRoute);
|
||||
if (index) {
|
||||
const fallbackRoute: RouteData = {
|
||||
|
@ -634,6 +634,7 @@ export function createRouteManifest(
|
|||
pathname,
|
||||
route,
|
||||
segments,
|
||||
generate,
|
||||
pattern: getPattern(segments, config, config.trailingSlash),
|
||||
type: 'fallback',
|
||||
fallbackRoutes: [],
|
||||
|
|
|
@ -31,7 +31,7 @@ export function getParams(array: string[]) {
|
|||
export function stringifyParams(params: GetStaticPathsItem['params'], route: RouteData) {
|
||||
// validate parameter values then stringify each value
|
||||
const validatedParams = Object.entries(params).reduce((acc, next) => {
|
||||
validateGetStaticPathsParameter(next, route.component);
|
||||
validateGetStaticPathsParameter(next, route.route);
|
||||
const [key, value] = next;
|
||||
if (value !== undefined) {
|
||||
acc[key] = typeof value === 'string' ? trimSlashes(value) : value.toString();
|
||||
|
|
|
@ -41,7 +41,7 @@ export function createI18nMiddleware(
|
|||
}
|
||||
|
||||
const url = context.url;
|
||||
const { locales, defaultLocale, fallback } = i18n;
|
||||
const { locales, defaultLocale, fallback, routingStrategy } = i18n;
|
||||
const response = await next();
|
||||
|
||||
if (response instanceof Response) {
|
||||
|
@ -82,7 +82,7 @@ export function createI18nMiddleware(
|
|||
let newPathname: string;
|
||||
// If a locale falls back to the default locale, we want to **remove** the locale because
|
||||
// the default locale doesn't have a prefix
|
||||
if (fallbackLocale === defaultLocale) {
|
||||
if (fallbackLocale === defaultLocale && routingStrategy === 'prefix-other-locales') {
|
||||
newPathname = url.pathname.replace(`/${urlLocale}`, ``);
|
||||
} else {
|
||||
newPathname = url.pathname.replace(`/${urlLocale}`, `/${fallbackLocale}`);
|
||||
|
|
|
@ -646,6 +646,34 @@ describe('[SSG] i18n routing', () => {
|
|||
expect($('script').text()).includes('console.log("this is a script")');
|
||||
});
|
||||
});
|
||||
|
||||
describe('i18n routing with fallback and [prefix-always]', () => {
|
||||
/** @type {import('./test-utils').Fixture} */
|
||||
let fixture;
|
||||
|
||||
before(async () => {
|
||||
fixture = await loadFixture({
|
||||
root: './fixtures/i18n-routing-prefix-always/',
|
||||
experimental: {
|
||||
i18n: {
|
||||
defaultLocale: 'en',
|
||||
locales: ['en', 'pt', 'it'],
|
||||
fallback: {
|
||||
it: 'en',
|
||||
},
|
||||
routingStrategy: 'prefix-always',
|
||||
},
|
||||
},
|
||||
});
|
||||
await fixture.build();
|
||||
});
|
||||
|
||||
it('should render the en locale', async () => {
|
||||
let html = await fixture.readFile('/it/start/index.html');
|
||||
expect(html).to.include('http-equiv="refresh');
|
||||
expect(html).to.include('url=/new-site/en/start');
|
||||
});
|
||||
});
|
||||
});
|
||||
describe('[SSR] i18n routing', () => {
|
||||
let app;
|
||||
|
|
|
@ -109,7 +109,6 @@ describe('astro:ssr-manifest, split', () => {
|
|||
const request = new Request('http://example.com/');
|
||||
const response = await app.render(request);
|
||||
const html = await response.text();
|
||||
console.log(html);
|
||||
expect(html.includes('<title>Testing</title>')).to.be.true;
|
||||
});
|
||||
});
|
||||
|
|
Loading…
Add table
Reference in a new issue