2022-10-12 21:25:51 +00:00
import type { AstroAdapter , AstroConfig , AstroIntegration } from 'astro' ;
2022-06-16 14:12:25 +00:00
import esbuild from 'esbuild' ;
import * as fs from 'fs' ;
2022-11-21 13:31:21 +00:00
import * as os from 'os' ;
import glob from 'tiny-glob' ;
2023-01-30 19:50:44 +00:00
import { fileURLToPath , pathToFileURL } from 'url' ;
2022-06-16 14:12:25 +00:00
2022-08-08 17:10:48 +00:00
type Options = {
mode : 'directory' | 'advanced' ;
} ;
2022-10-12 21:25:51 +00:00
interface BuildConfig {
server : URL ;
client : URL ;
serverEntry : string ;
}
2022-08-08 17:10:48 +00:00
export function getAdapter ( isModeDirectory : boolean ) : AstroAdapter {
return isModeDirectory
? {
name : '@astrojs/cloudflare' ,
serverEntrypoint : '@astrojs/cloudflare/server.directory.js' ,
exports : [ 'onRequest' ] ,
}
: {
name : '@astrojs/cloudflare' ,
serverEntrypoint : '@astrojs/cloudflare/server.advanced.js' ,
exports : [ 'default' ] ,
} ;
2022-06-16 14:12:25 +00:00
}
2022-07-27 20:15:19 +00:00
const SHIM = ` globalThis.process = {
argv : [ ] ,
env : { } ,
} ; ` ;
2022-11-21 13:31:21 +00:00
const SERVER_BUILD_FOLDER = '/$server_build/' ;
2022-08-08 17:10:48 +00:00
export default function createIntegration ( args? : Options ) : AstroIntegration {
2022-06-16 14:12:25 +00:00
let _config : AstroConfig ;
let _buildConfig : BuildConfig ;
2022-08-08 17:10:48 +00:00
const isModeDirectory = args ? . mode === 'directory' ;
2022-06-16 14:12:25 +00:00
return {
name : '@astrojs/cloudflare' ,
hooks : {
2022-10-12 21:25:51 +00:00
'astro:config:setup' : ( { config , updateConfig } ) = > {
updateConfig ( {
build : {
2022-11-21 13:31:21 +00:00
client : new URL ( ` . ${ config . base } ` , config . outDir ) ,
server : new URL ( ` . ${ SERVER_BUILD_FOLDER } ` , config . outDir ) ,
2023-01-26 17:43:39 +00:00
serverEntry : '_worker.mjs' ,
2022-10-12 21:27:56 +00:00
} ,
2022-10-12 21:25:51 +00:00
} ) ;
} ,
2022-06-16 14:12:25 +00:00
'astro:config:done' : ( { setAdapter , config } ) = > {
2022-08-08 17:10:48 +00:00
setAdapter ( getAdapter ( isModeDirectory ) ) ;
2022-06-16 14:12:25 +00:00
_config = config ;
2022-10-12 21:25:51 +00:00
_buildConfig = config . build ;
2022-07-25 04:18:02 +00:00
2022-07-25 04:20:38 +00:00
if ( config . output === 'static' ) {
2022-07-27 15:50:48 +00:00
throw new Error ( `
[ @astrojs / cloudflare ] \ ` output: "server" \` is required to use this adapter. Otherwise, this adapter is not necessary to deploy a static site to Cloudflare.
` );
2022-07-25 04:18:02 +00:00
}
2022-11-21 13:31:21 +00:00
if ( config . base === SERVER_BUILD_FOLDER ) {
throw new Error ( `
[ @astrojs / cloudflare ] \ ` base: " ${ SERVER_BUILD_FOLDER } " \` is not allowed. Please change your \` base \` config to something else. ` ) ;
}
2022-06-16 14:12:25 +00:00
} ,
'astro:build:setup' : ( { vite , target } ) = > {
if ( target === 'server' ) {
vite . resolve = vite . resolve || { } ;
vite . resolve . alias = vite . resolve . alias || { } ;
const aliases = [ { find : 'react-dom/server' , replacement : 'react-dom/server.browser' } ] ;
if ( Array . isArray ( vite . resolve . alias ) ) {
vite . resolve . alias = [ . . . vite . resolve . alias , . . . aliases ] ;
} else {
for ( const alias of aliases ) {
( vite . resolve . alias as Record < string , string > ) [ alias . find ] = alias . replacement ;
}
}
2022-10-06 14:28:47 +00:00
vite . ssr = vite . ssr || { } ;
vite . ssr . target = vite . ssr . target || 'webworker' ;
2022-06-16 14:12:25 +00:00
}
} ,
2023-01-26 17:43:39 +00:00
'astro:build:done' : async ( { pages } ) = > {
2023-01-30 19:50:44 +00:00
const entryPath = fileURLToPath ( new URL ( _buildConfig . serverEntry , _buildConfig . server ) ) ;
const entryUrl = new URL ( _buildConfig . serverEntry , _config . outDir ) ;
const buildPath = fileURLToPath ( entryUrl ) ;
// A URL for the final build path after renaming
const finalBuildUrl = pathToFileURL ( buildPath . replace ( /\.mjs$/ , '.js' ) ) ;
2023-01-26 17:43:39 +00:00
2022-06-16 14:12:25 +00:00
await esbuild . build ( {
target : 'es2020' ,
2022-10-06 14:28:47 +00:00
platform : 'browser' ,
2022-11-21 13:31:21 +00:00
entryPoints : [ entryPath ] ,
outfile : buildPath ,
2022-06-16 14:12:25 +00:00
allowOverwrite : true ,
format : 'esm' ,
bundle : true ,
minify : true ,
2022-07-27 20:15:19 +00:00
banner : {
2022-07-27 20:17:38 +00:00
js : SHIM ,
} ,
2022-06-16 14:12:25 +00:00
} ) ;
2023-01-26 17:43:39 +00:00
// Rename to worker.js
2023-01-30 19:50:44 +00:00
await fs . promises . rename ( buildPath , finalBuildUrl ) ;
2023-01-26 17:43:39 +00:00
2022-06-16 14:12:25 +00:00
// throw the server folder in the bin
2022-11-21 13:31:21 +00:00
const serverUrl = new URL ( _buildConfig . server ) ;
await fs . promises . rm ( serverUrl , { recursive : true , force : true } ) ;
// move cloudflare specific files to the root
const cloudflareSpecialFiles = [ '_headers' , '_redirects' , '_routes.json' ] ;
if ( _config . base !== '/' ) {
for ( const file of cloudflareSpecialFiles ) {
try {
await fs . promises . rename (
new URL ( file , _buildConfig . client ) ,
new URL ( file , _config . outDir )
) ;
} catch ( e ) {
// ignore
}
}
}
const routesExists = await fs . promises
. stat ( new URL ( './_routes.json' , _config . outDir ) )
. then ( ( stat ) = > stat . isFile ( ) )
. catch ( ( ) = > false ) ;
// this creates a _routes.json, in case there is none present to enable
// cloudflare to handle static files and support _redirects configuration
// (without calling the function)
if ( ! routesExists ) {
const staticPathList : Array < string > = (
await glob ( ` ${ fileURLToPath ( _buildConfig . client ) } /**/* ` , {
cwd : fileURLToPath ( _config . outDir ) ,
filesOnly : true ,
} )
)
. filter ( ( file : string ) = > cloudflareSpecialFiles . indexOf ( file ) < 0 )
. map ( ( file : string ) = > ` / ${ file } ` ) ;
2023-01-26 17:45:39 +00:00
for ( let page of pages ) {
2023-01-26 17:43:39 +00:00
staticPathList . push ( prependForwardSlash ( page . pathname ) ) ;
}
2022-11-21 13:31:21 +00:00
const redirectsExists = await fs . promises
. stat ( new URL ( './_redirects' , _config . outDir ) )
. then ( ( stat ) = > stat . isFile ( ) )
. catch ( ( ) = > false ) ;
2022-11-21 13:33:45 +00:00
// convert all redirect source paths into a list of routes
// and add them to the static path
2022-11-21 13:31:21 +00:00
if ( redirectsExists ) {
const redirects = (
await fs . promises . readFile ( new URL ( './_redirects' , _config . outDir ) , 'utf-8' )
)
. split ( os . EOL )
. map ( ( line ) = > {
const parts = line . split ( ' ' ) ;
if ( parts . length < 2 ) {
return null ;
} else {
// convert /products/:id to /products/*
return (
parts [ 0 ]
. replace ( /\/:.*?(?=\/|$)/g , '/*' )
// remove query params as they are not supported by cloudflare
. replace ( /\?.*$/ , '' )
) ;
}
} )
. filter (
( line , index , arr ) = > line !== null && arr . indexOf ( line ) === index
) as string [ ] ;
if ( redirects . length > 0 ) {
staticPathList . push ( . . . redirects ) ;
}
}
await fs . promises . writeFile (
new URL ( './_routes.json' , _config . outDir ) ,
JSON . stringify (
{
version : 1 ,
include : [ '/*' ] ,
exclude : staticPathList ,
} ,
null ,
2
)
) ;
}
2022-08-08 17:10:48 +00:00
if ( isModeDirectory ) {
2023-02-13 01:19:38 +00:00
const functionsUrl = new URL ( 'functions/' , _config . root ) ;
2022-08-08 17:10:48 +00:00
await fs . promises . mkdir ( functionsUrl , { recursive : true } ) ;
2023-02-13 01:19:38 +00:00
2022-08-08 17:10:48 +00:00
const directoryUrl = new URL ( '[[path]].js' , functionsUrl ) ;
2023-01-30 19:50:44 +00:00
await fs . promises . rename ( finalBuildUrl , directoryUrl ) ;
2022-08-08 17:10:48 +00:00
}
2022-06-16 14:12:25 +00:00
} ,
} ,
} ;
}
2023-01-26 17:43:39 +00:00
function prependForwardSlash ( path : string ) {
return path [ 0 ] === '/' ? path : '/' + path ;
}