2022-06-06 16:47:28 +00:00
|
|
|
// Normal Imports
|
2022-03-30 12:42:19 +00:00
|
|
|
import type { SSRManifest } from 'astro';
|
|
|
|
import { App } from 'astro/app';
|
2023-02-16 15:16:07 +00:00
|
|
|
|
2023-02-17 15:58:53 +00:00
|
|
|
// @ts-ignore
|
2023-02-17 18:19:19 +00:00
|
|
|
import { Server, serveFile, fromFileUrl } from '@astrojs/deno/__deno_imports.js';
|
2023-02-17 15:58:53 +00:00
|
|
|
|
2022-03-30 12:42:19 +00:00
|
|
|
interface Options {
|
|
|
|
port?: number;
|
|
|
|
hostname?: string;
|
|
|
|
start?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
let _server: Server | undefined = undefined;
|
|
|
|
let _startPromise: Promise<void> | undefined = undefined;
|
|
|
|
|
2023-02-17 18:19:19 +00:00
|
|
|
async function* getPrerenderedFiles(clientRoot: URL): AsyncGenerator<URL> {
|
|
|
|
// @ts-ignore
|
|
|
|
for await (const ent of Deno.readDir(clientRoot)) {
|
|
|
|
if (ent.isDirectory) {
|
|
|
|
yield* getPrerenderedFiles(new URL(`./${ent.name}/`, clientRoot))
|
|
|
|
} else if (ent.name.endsWith('.html')) {
|
|
|
|
yield new URL(`./${ent.name}`, clientRoot)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-30 12:42:19 +00:00
|
|
|
export function start(manifest: SSRManifest, options: Options) {
|
2022-03-30 12:43:13 +00:00
|
|
|
if (options.start === false) {
|
2022-03-30 12:42:19 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-04-15 21:01:33 +00:00
|
|
|
const clientRoot = new URL('../client/', import.meta.url);
|
2022-03-30 12:42:19 +00:00
|
|
|
const app = new App(manifest);
|
2022-07-19 20:10:15 +00:00
|
|
|
const handler = async (request: Request, connInfo: any) => {
|
2022-04-15 21:02:19 +00:00
|
|
|
if (app.match(request)) {
|
2022-07-19 20:10:15 +00:00
|
|
|
let ip = connInfo?.remoteAddr?.hostname;
|
|
|
|
Reflect.set(request, Symbol.for('astro.clientAddress'), ip);
|
2022-09-28 20:55:27 +00:00
|
|
|
const response = await app.render(request);
|
2022-09-28 20:57:35 +00:00
|
|
|
if (app.setCookieHeaders) {
|
|
|
|
for (const setCookieHeader of app.setCookieHeaders(response)) {
|
2022-09-28 20:55:27 +00:00
|
|
|
response.headers.append('Set-Cookie', setCookieHeader);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return response;
|
2022-04-15 21:01:33 +00:00
|
|
|
}
|
2022-08-31 20:14:25 +00:00
|
|
|
|
2022-08-31 20:11:46 +00:00
|
|
|
// If the request path wasn't found in astro,
|
|
|
|
// try to fetch a static file instead
|
2022-04-15 21:01:33 +00:00
|
|
|
const url = new URL(request.url);
|
2022-11-07 15:05:12 +00:00
|
|
|
const localPath = new URL('./' + app.removeBase(url.pathname), clientRoot);
|
2023-02-17 18:19:19 +00:00
|
|
|
|
|
|
|
let fileResp = await serveFile(request, fromFileUrl(localPath));
|
|
|
|
|
|
|
|
// Attempt to serve `index.html` if 404
|
|
|
|
if (fileResp.status == 404) {
|
|
|
|
let fallback;
|
|
|
|
for await (const file of getPrerenderedFiles(clientRoot)) {
|
|
|
|
const pathname = file.pathname.replace(/\/(index)?\.html$/, '');
|
|
|
|
if (localPath.pathname.endsWith(pathname)) {
|
|
|
|
fallback = file;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (fallback) {
|
|
|
|
fileResp = await serveFile(request, fromFileUrl(fallback));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-16 15:18:31 +00:00
|
|
|
|
2022-08-31 20:11:46 +00:00
|
|
|
// If the static file can't be found
|
|
|
|
if (fileResp.status == 404) {
|
2022-08-31 20:14:25 +00:00
|
|
|
// Render the astro custom 404 page
|
2022-09-28 20:55:27 +00:00
|
|
|
const response = await app.render(request);
|
|
|
|
|
2022-09-28 20:57:35 +00:00
|
|
|
if (app.setCookieHeaders) {
|
|
|
|
for (const setCookieHeader of app.setCookieHeaders(response)) {
|
2022-09-28 20:55:27 +00:00
|
|
|
response.headers.append('Set-Cookie', setCookieHeader);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return response;
|
2022-08-31 20:14:25 +00:00
|
|
|
|
|
|
|
// If the static file is found
|
2022-08-31 20:11:46 +00:00
|
|
|
} else {
|
2022-08-31 20:14:25 +00:00
|
|
|
return fileResp;
|
2022-08-31 20:11:46 +00:00
|
|
|
}
|
2022-03-30 12:42:19 +00:00
|
|
|
};
|
|
|
|
|
2022-06-06 16:02:13 +00:00
|
|
|
const port = options.port ?? 8085;
|
2022-03-30 12:42:19 +00:00
|
|
|
_server = new Server({
|
2022-06-06 16:02:13 +00:00
|
|
|
port,
|
2022-03-30 12:43:13 +00:00
|
|
|
hostname: options.hostname ?? '0.0.0.0',
|
|
|
|
handler,
|
|
|
|
});
|
2022-03-30 12:42:19 +00:00
|
|
|
|
2022-06-06 16:02:13 +00:00
|
|
|
_startPromise = Promise.resolve(_server.listenAndServe());
|
2022-06-22 15:59:49 +00:00
|
|
|
// eslint-disable-next-line no-console
|
2022-06-06 16:02:13 +00:00
|
|
|
console.error(`Server running on port ${port}`);
|
2022-03-30 12:42:19 +00:00
|
|
|
}
|
|
|
|
|
2022-04-15 21:01:33 +00:00
|
|
|
export function createExports(manifest: SSRManifest, options: Options) {
|
2022-03-30 12:42:19 +00:00
|
|
|
const app = new App(manifest);
|
|
|
|
return {
|
|
|
|
async stop() {
|
2022-03-30 12:43:13 +00:00
|
|
|
if (_server) {
|
|
|
|
_server.close();
|
2022-04-15 21:01:33 +00:00
|
|
|
_server = undefined;
|
2022-03-30 12:42:19 +00:00
|
|
|
}
|
|
|
|
await Promise.resolve(_startPromise);
|
|
|
|
},
|
2022-04-15 21:01:33 +00:00
|
|
|
running() {
|
|
|
|
return _server !== undefined;
|
|
|
|
},
|
|
|
|
async start() {
|
|
|
|
return start(manifest, options);
|
|
|
|
},
|
2022-03-30 12:42:19 +00:00
|
|
|
async handle(request: Request) {
|
|
|
|
return app.render(request);
|
2022-03-30 12:43:13 +00:00
|
|
|
},
|
|
|
|
};
|
2022-03-30 12:42:19 +00:00
|
|
|
}
|