-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserver.js
98 lines (88 loc) · 2.55 KB
/
server.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
import * as crypto from "node:crypto";
import { createRequestHandler } from "@remix-run/express";
import helmet from "helmet";
import compression from "compression";
import express from "express";
import morgan from "morgan";
const isProduction = process.env.NODE_ENV === "production";
const viteDevServer = isProduction
? undefined
: await import("vite").then((vite) =>
vite.createServer({
server: { middlewareMode: true },
}),
);
const remixHandler = createRequestHandler({
getLoadContext: (req, res) => ({
cspNonce: res.locals.cspNonce,
}),
build: viteDevServer
? () => viteDevServer.ssrLoadModule("virtual:remix/server-build")
: await import("./build/server/index.js"),
});
const app = express();
app.use((req, res, next) => {
res.locals.cspNonce = crypto.randomBytes(16).toString("hex");
res.setHeader(
"Permissions-Policy",
"accelerometer=(), camera=(), geolocation=(), gyroscope=(), magnetometer=(), microphone=(), payment=(), usb=()",
);
next();
});
app.use(
helmet({
contentSecurityPolicy: {
directives: {
"script-src": [
"'self'",
"https://www.googletagmanager.com/",
"https://js.stripe.com/",
(req, res) => `'nonce-${res.locals.cspNonce}'`,
],
"frame-src": [
"'self'",
"https://js.stripe.com/",
"https://airtable.com/",
],
"img-src": [
"'self'",
"data:",
"https://widgets.guidestar.org/",
"https://www.googletagmanager.com",
],
"connect-src": [
"'self'",
"https://www.google-analytics.com/",
() => (isProduction ? "" : "ws://localhost:24678"),
],
},
},
crossOriginEmbedderPolicy: false,
}),
);
app.use(compression());
// handle asset requests
if (viteDevServer) {
app.use(viteDevServer.middlewares);
} else {
// Vite fingerprints its assets so we can cache forever.
app.use(
"/assets",
express.static("build/client/assets", { immutable: true, maxAge: "1y" }),
);
}
// Serve images with 1-year cache
app.use(
"/images",
express.static("public/images", { immutable: true, maxAge: "1y" }),
);
// Everything else (like favicon.ico) is cached for an hour. You may want to be
// more aggressive with this caching.
app.use(express.static("build/client", { maxAge: "1h" }));
app.use(morgan("tiny"));
// handle SSR requests
app.all("*", remixHandler);
const port = process.env.PORT || 3000;
app.listen(port, () =>
console.log(`Express server listening at http://localhost:${port}`),
);