mirror of
https://github.com/withastro/astro.git
synced 2025-03-31 23:31:30 -05:00
feat(underscore-redirects): update API to support new hook (#12924)
This commit is contained in:
parent
fd12a26ac6
commit
3caa337f0b
4 changed files with 72 additions and 25 deletions
5
.changeset/heavy-yaks-brake.md
Normal file
5
.changeset/heavy-yaks-brake.md
Normal file
|
@ -0,0 +1,5 @@
|
|||
---
|
||||
'@astrojs/underscore-redirects': minor
|
||||
---
|
||||
|
||||
Updates how the output is determined in `createRedirectsFromAstroRoutes`. Since `v0.5.0`, the output would use the `buildOutput` property and `config.output` as a fallback. It no longer uses this fallback.
|
40
.changeset/tiny-plums-crash.md
Normal file
40
.changeset/tiny-plums-crash.md
Normal file
|
@ -0,0 +1,40 @@
|
|||
---
|
||||
'@astrojs/underscore-redirects': minor
|
||||
---
|
||||
|
||||
Updates the input requirements of `createRedirectsFromAstroRoutes`:
|
||||
|
||||
- `routeToDynamicTargetMap` keys are `IntegrationResolvedRoute` instead of `IntegrationRouteData` (obtained from the `astro:routes:resolved` hook)
|
||||
- There's a new `assets` property, that can be obtained from the `astro:build:done` hook
|
||||
|
||||
```js
|
||||
function myIntegration() {
|
||||
let routes
|
||||
let buildOutput
|
||||
let config
|
||||
|
||||
return {
|
||||
name: "my-integration",
|
||||
hooks: {
|
||||
"astro:routes:resolved": (params) => {
|
||||
routes = params.routes
|
||||
},
|
||||
"astro:config:done": (params) => {
|
||||
buildOutput = params.buildOutput
|
||||
config = params.config
|
||||
},
|
||||
"astro:build:done": (params) => {
|
||||
const redirects = createRedirectsFromAstroRoutes({
|
||||
config,
|
||||
buildOutput,
|
||||
routeToDynamicTargetMap: new Map(
|
||||
routes.map(route => [route, ''])
|
||||
),
|
||||
dir: params.dir,
|
||||
assets: params.assets
|
||||
})
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
|
@ -1,10 +1,15 @@
|
|||
import { posix } from 'node:path';
|
||||
import type { AstroConfig, IntegrationRouteData, ValidRedirectStatus } from 'astro';
|
||||
import type {
|
||||
AstroConfig,
|
||||
HookParameters,
|
||||
IntegrationResolvedRoute,
|
||||
ValidRedirectStatus,
|
||||
} from 'astro';
|
||||
import { Redirects } from './redirects.js';
|
||||
|
||||
const pathJoin = posix.join;
|
||||
|
||||
function getRedirectStatus(route: IntegrationRouteData): ValidRedirectStatus {
|
||||
function getRedirectStatus(route: IntegrationResolvedRoute): ValidRedirectStatus {
|
||||
if (typeof route.redirect === 'object') {
|
||||
return route.redirect.status;
|
||||
}
|
||||
|
@ -16,9 +21,10 @@ interface CreateRedirectsFromAstroRoutesParams {
|
|||
/**
|
||||
* Maps a `RouteData` to a dynamic target
|
||||
*/
|
||||
routeToDynamicTargetMap: Map<IntegrationRouteData, string>;
|
||||
routeToDynamicTargetMap: Map<IntegrationResolvedRoute, string>;
|
||||
dir: URL;
|
||||
buildOutput: 'static' | 'server';
|
||||
assets: HookParameters<'astro:build:done'>['assets'];
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -29,6 +35,7 @@ export function createRedirectsFromAstroRoutes({
|
|||
routeToDynamicTargetMap,
|
||||
dir,
|
||||
buildOutput,
|
||||
assets,
|
||||
}: CreateRedirectsFromAstroRoutesParams) {
|
||||
const base =
|
||||
config.base && config.base !== '/'
|
||||
|
@ -36,11 +43,10 @@ export function createRedirectsFromAstroRoutes({
|
|||
? config.base.slice(0, -1)
|
||||
: config.base
|
||||
: '';
|
||||
// TODO: the use of `config.output` is deprecated. We need to update the adapters that use this package to pass the new buildOutput
|
||||
const output = buildOutput ?? config.output;
|
||||
const _redirects = new Redirects();
|
||||
|
||||
for (const [route, dynamicTarget = ''] of routeToDynamicTargetMap) {
|
||||
const distURL = assets.get(route.pattern);
|
||||
// A route with a `pathname` is as static route.
|
||||
if (route.pathname) {
|
||||
if (route.redirect) {
|
||||
|
@ -57,13 +63,13 @@ export function createRedirectsFromAstroRoutes({
|
|||
}
|
||||
|
||||
// If this is a static build we don't want to add redirects to the HTML file.
|
||||
if (output === 'static') {
|
||||
if (buildOutput === 'static') {
|
||||
continue;
|
||||
} else if (route.distURL) {
|
||||
} else if (distURL) {
|
||||
_redirects.add({
|
||||
dynamic: false,
|
||||
input: `${base}${route.pathname}`,
|
||||
target: prependForwardSlash(route.distURL.toString().replace(dir.toString(), '')),
|
||||
target: prependForwardSlash(distURL.toString().replace(dir.toString(), '')),
|
||||
status: 200,
|
||||
weight: 2,
|
||||
});
|
||||
|
@ -76,7 +82,7 @@ export function createRedirectsFromAstroRoutes({
|
|||
weight: 2,
|
||||
});
|
||||
|
||||
if (route.route === '/404') {
|
||||
if (route.pattern === '/404') {
|
||||
_redirects.add({
|
||||
dynamic: true,
|
||||
input: '/*',
|
||||
|
@ -92,7 +98,7 @@ export function createRedirectsFromAstroRoutes({
|
|||
const pattern = generateDynamicPattern(route);
|
||||
|
||||
// This route was prerendered and should be forwarded to the HTML file.
|
||||
if (route.distURL) {
|
||||
if (distURL) {
|
||||
const targetRoute = route.redirectRoute ?? route;
|
||||
const targetPattern = generateDynamicPattern(targetRoute);
|
||||
let target = targetPattern;
|
||||
|
@ -128,7 +134,7 @@ export function createRedirectsFromAstroRoutes({
|
|||
* /team/articles/*
|
||||
* With stars replacing spread and :id syntax replacing [id]
|
||||
*/
|
||||
function generateDynamicPattern(route: IntegrationRouteData) {
|
||||
function generateDynamicPattern(route: IntegrationResolvedRoute) {
|
||||
const pattern =
|
||||
'/' +
|
||||
route.segments
|
||||
|
|
|
@ -3,28 +3,24 @@ import { describe, it } from 'node:test';
|
|||
import { createRedirectsFromAstroRoutes } from '../dist/index.js';
|
||||
|
||||
describe('Astro', () => {
|
||||
const serverConfig = {
|
||||
output: 'server',
|
||||
build: { format: 'directory' },
|
||||
};
|
||||
|
||||
it('Creates a Redirects object from routes', () => {
|
||||
const routeToDynamicTargetMap = new Map(
|
||||
Array.from([
|
||||
[
|
||||
{ pathname: '/', distURL: new URL('./index.html', import.meta.url), segments: [] },
|
||||
'./.adapter/dist/entry.mjs',
|
||||
],
|
||||
[
|
||||
{ pathname: '/one', distURL: new URL('./one/index.html', import.meta.url), segments: [] },
|
||||
'./.adapter/dist/entry.mjs',
|
||||
],
|
||||
[{ pattern: '/', pathname: '/', segments: [] }, './.adapter/dist/entry.mjs'],
|
||||
[{ pattern: '/one', pathname: '/one', segments: [] }, './.adapter/dist/entry.mjs'],
|
||||
]),
|
||||
);
|
||||
const _redirects = createRedirectsFromAstroRoutes({
|
||||
config: serverConfig,
|
||||
config: {
|
||||
build: { format: 'directory' },
|
||||
},
|
||||
routeToDynamicTargetMap,
|
||||
dir: new URL(import.meta.url),
|
||||
buildOutput: 'server',
|
||||
assets: new Map([
|
||||
['/', new URL('./index.html', import.meta.url)],
|
||||
['/one', new URL('./one/index.html', import.meta.url)],
|
||||
]),
|
||||
});
|
||||
|
||||
assert.equal(_redirects.definitions.length, 2);
|
||||
|
|
Loading…
Add table
Reference in a new issue