SequenceDiagram/bin/server.js

180 lines
5.3 KiB
JavaScript
Executable File

#!/usr/bin/env -S node --disable-proto delete --disallow-code-generation-from-strings
const {Server} = require('./server/Server');
const {StaticRequestHandler} = require('./server/StaticRequestHandler');
const {RenderRequestHandler} = require('./handlers/RenderRequestHandler');
const {PreviewRequestHandler} = require('./handlers/PreviewRequestHandler');
const path = require('path');
const DEV = process.argv.includes('dev');
const HOSTNAME = '127.0.0.1';
const BASEDIR = path.join(__dirname, '..') + '/';
let PORT = Number.parseInt(process.argv[2], 10);
if(Number.isNaN(PORT)) {
PORT = 8080;
}
function devMapper(file, type, data) {
if(!type.includes('text/html')) {
return data;
}
const code = data.toString('utf8');
if(DEV) {
return code
.replace(/<!--* *DEV *-*>?([^]*?)(?:<!)?-* *\/DEV *-->/g, '$1')
.replace(/<!--* *LIVE[^]*? *\/LIVE *-->/g, '');
} else {
return code
.replace(/<!--* *LIVE *-*>?([^]*?)(?:<!)?-* *\/LIVE *-->/g, '$1')
.replace(/<!--* *DEV[^]*? *\/DEV *-->/g, '');
}
}
const MINUTE = 60;
const HOUR = MINUTE * 60;
const DAY = HOUR * 24;
const YEAR = DAY * 365;
const STATIC_CACHE = {
maxAgeSeconds: 10 * MINUTE,
staleIfErrorSeconds: YEAR,
staleWhileRevalidateSeconds: YEAR,
};
const RENDER_CACHE = {
immutable: true,
maxAgeSeconds: 30 * DAY,
staleIfErrorSeconds: YEAR,
staleWhileRevalidateSeconds: YEAR,
};
const PREVIEW_CACHE = {
maxAgeSeconds: HOUR,
staleIfErrorSeconds: DAY,
staleWhileRevalidateSeconds: DAY,
};
const SKETCH_CSS_SHA = 'sha256-s7UPtBgvov5WNF9C1DlTZDpqwLgEmfiWha5a5p/Zn7E=';
const PERMISSIONS_POLICY = [
'accelerometer=()',
'autoplay=()',
'camera=()',
'geolocation=()',
'gyroscope=()',
'interest-cohort=()',
'magnetometer=()',
'microphone=()',
'payment=()',
'sync-xhr=()',
'usb=()',
].join(', ');
const statics = new StaticRequestHandler('')
.setCache(DEV ? {} : STATIC_CACHE)
.addHeader('Content-Security-Policy', [
'base-uri \'self\'',
'default-src \'none\'',
'script-src \'self\' https://unpkg.com',
// Using fonts.googleapis.com for library.htm only
`style-src 'self' https://fonts.googleapis.com '${SKETCH_CSS_SHA}'`,
'connect-src \'self\'',
// Using fonts.gstatic.com for library.htm only
'font-src \'self\' data: https://fonts.gstatic.com',
'img-src \'self\' blob:',
'form-action \'self\'',
'frame-ancestors \'self\'',
'frame-src \'self\'',
].join('; '))
.addHeader('Cross-Origin-Embedder-Policy', 'require-corp')
.addHeader('Cross-Origin-Opener-Policy', 'same-origin')
.addHeader('Cross-Origin-Resource-Policy', 'same-origin')
.addHeader('Permissions-Policy', PERMISSIONS_POLICY)
.addHeader('Referrer-Policy', 'no-referrer')
.addHeader('X-Content-Type-Options', 'nosniff')
.addHeader('X-Frame-Options', 'DENY')
.addHeader('X-XSS-Protection', '1; mode=block')
.addMimeType('txt', 'text/plain; charset=utf-8')
.addMimeType('htm', 'text/html; charset=utf-8')
.addMimeType('html', 'text/html; charset=utf-8')
.addMimeType('js', 'application/javascript; charset=utf-8')
.addMimeType('mjs', 'application/javascript; charset=utf-8')
.addMimeType('css', 'text/css; charset=utf-8')
.addMimeType('png', 'image/png')
.addMimeType('svg', 'image/svg+xml; charset=utf-8');
statics
.add('/robots.txt', '')
.add('/ads.txt', [
'# Deny inclusion in any advertising system\n',
'placeholder.example.com, placeholder, DIRECT, placeholder\n',
].join(''))
.add('/.well-known/security.txt', [
'Contact: https://github.com/davidje13/SequenceDiagram/issues\n',
'Preferred-Languages: en\n',
'Expires: 3000-01-01T00:00:00Z\n',
].join(''))
.addResources('/', BASEDIR, [
'index.html',
'library.htm',
'lib',
'web/lib',
'web/resources',
'web/styles',
], devMapper);
if(DEV) {
statics.addResources('/', BASEDIR, [
'node_modules/requirejs/require.js',
'node_modules/codemirror/lib',
'node_modules/codemirror/addon',
'node_modules/codemirror/mode',
'scripts',
'web/scripts',
]);
statics.setFileWatch(true);
}
const render = new RenderRequestHandler('/render')
.setCache(DEV ? {} : RENDER_CACHE)
.setCrossOrigin(true)
.addHeader('Content-Security-Policy', [
'base-uri \'self\'',
'default-src \'none\'',
`style-src '${SKETCH_CSS_SHA}'`,
'connect-src \'none\'',
'font-src data:',
'form-action \'none\'',
].join('; '))
.addHeader('Cross-Origin-Embedder-Policy', 'require-corp')
.addHeader('Cross-Origin-Opener-Policy', 'unsafe-none')
.addHeader('Cross-Origin-Resource-Policy', 'cross-origin')
.addHeader('Permissions-Policy', PERMISSIONS_POLICY)
.addHeader('Referrer-Policy', 'no-referrer')
.addHeader('X-Content-Type-Options', 'nosniff');
const preview = new PreviewRequestHandler('/preview')
.setCache(DEV ? {} : PREVIEW_CACHE)
.addHeader('Content-Security-Policy', [
'base-uri \'self\'',
'default-src \'none\'',
'style-src \'self\'',
'connect-src \'none\'',
'img-src \'self\'',
'form-action \'none\'',
'frame-ancestors \'self\'',
'frame-src \'self\'',
].join('; '))
.addHeader('Cross-Origin-Embedder-Policy', 'require-corp')
.addHeader('Cross-Origin-Opener-Policy', 'same-origin')
.addHeader('Cross-Origin-Resource-Policy', 'same-origin')
.addHeader('Permissions-Policy', PERMISSIONS_POLICY)
.addHeader('Referrer-Policy', 'no-referrer')
.addHeader('X-Content-Type-Options', 'nosniff');
new Server()
.addHandler(render)
.addHandler(preview)
.addHandler(statics)
.listen(PORT, HOSTNAME)
.then((server) => server.printListeningInfo(process.stdout));