78 lines
1.9 KiB
TypeScript
78 lines
1.9 KiB
TypeScript
|
import fs from 'fs';
|
||
|
import http from 'http';
|
||
|
import https from 'https';
|
||
|
import { fileURLToPath } from 'url';
|
||
|
import send from 'send';
|
||
|
|
||
|
interface CreateServerOptions {
|
||
|
client: URL;
|
||
|
port: number;
|
||
|
host: string | undefined;
|
||
|
}
|
||
|
|
||
|
export function createServer({ client, port, host }: CreateServerOptions, handler: http.RequestListener) {
|
||
|
const listener: http.RequestListener = (req, res) => {
|
||
|
if(req.url) {
|
||
|
const fileURL = new URL('.' + req.url, client);
|
||
|
|
||
|
const stream = send(req, fileURLToPath(fileURL), {
|
||
|
dotfiles: 'deny',
|
||
|
});
|
||
|
|
||
|
let forwardError = false;
|
||
|
|
||
|
stream.on('error', err => {
|
||
|
if(forwardError) {
|
||
|
// eslint-disable-next-line no-console
|
||
|
console.error(err.toString());
|
||
|
res.writeHead(500);
|
||
|
res.end('Internal server error');
|
||
|
return;
|
||
|
}
|
||
|
// File not found, forward to the SSR handler
|
||
|
handler(req, res);
|
||
|
});
|
||
|
|
||
|
stream.on('file', () => {
|
||
|
forwardError = true;
|
||
|
});
|
||
|
stream.pipe(res);
|
||
|
} else {
|
||
|
handler(req, res);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
let httpServer: http.Server<typeof http.IncomingMessage, typeof http.ServerResponse> |
|
||
|
https.Server<typeof http.IncomingMessage, typeof http.ServerResponse>;
|
||
|
|
||
|
if(process.env.SERVER_CERT_PATH && process.env.SERVER_KEY_PATH) {
|
||
|
httpServer = https.createServer({
|
||
|
key: fs.readFileSync(process.env.SERVER_KEY_PATH),
|
||
|
cert: fs.readFileSync(process.env.SERVER_CERT_PATH),
|
||
|
}, listener);
|
||
|
} else {
|
||
|
httpServer = http.createServer(listener);
|
||
|
}
|
||
|
httpServer.listen(port, host);
|
||
|
|
||
|
// Resolves once the server is closed
|
||
|
const closed = new Promise<void>((resolve, reject) => {
|
||
|
httpServer.addListener('close', resolve);
|
||
|
httpServer.addListener('error', reject);
|
||
|
});
|
||
|
|
||
|
return {
|
||
|
host,
|
||
|
port,
|
||
|
closed() {
|
||
|
return closed;
|
||
|
},
|
||
|
server: httpServer,
|
||
|
stop: async () => {
|
||
|
await new Promise((resolve, reject) => {
|
||
|
httpServer.close((err) => (err ? reject(err) : resolve(undefined)));
|
||
|
});
|
||
|
},
|
||
|
};
|
||
|
}
|