-
Notifications
You must be signed in to change notification settings - Fork 12
/
vite.config.ts
154 lines (145 loc) · 4.9 KB
/
vite.config.ts
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
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
/*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, you can obtain one at https://mozilla.org/MPL/2.0/.
*
* Copyright Oxide Computer Company
*/
import { randomBytes } from 'crypto'
import { resolve } from 'path'
import basicSsl from '@vitejs/plugin-basic-ssl'
import react from '@vitejs/plugin-react-swc'
import { defineConfig } from 'vite'
import { createHtmlPlugin } from 'vite-plugin-html'
import tsconfigPaths from 'vite-tsconfig-paths'
import { z } from 'zod'
import vercelConfig from './vercel.json'
const ApiMode = z.enum(['msw', 'dogfood', 'nexus'])
const apiModeResult = ApiMode.default('nexus').safeParse(process.env.API_MODE)
if (!apiModeResult.success) {
const options = ApiMode.options.join(', ')
console.error(`Error: API_MODE must be one of: [${options}]. If unset, default is "msw".`)
process.exit(1)
}
/**
* What API are we talking to? Only relevant in development mode.
*
* - `msw` (default): Mock Service Worker
* - `dogfood`: Dogfood rack at oxide.sys.rack2.eng.oxide.computer. Requires VPN.
* - `nexus`: Builds for production, assumes Nexus at localhost:12220 in dev mode only
*/
const apiMode = apiModeResult.data
// if you want a different host you can override it with EXT_HOST
const DOGFOOD_HOST = process.env.EXT_HOST || 'oxide.sys.rack2.eng.oxide.computer'
const previewAnalyticsTag = {
injectTo: 'head' as const,
tag: 'script',
attrs: {
'data-domain':
process.env.VERCEL_ENV === 'production'
? 'oxide-console-preview.vercel.app'
: // not a real domain. we're only using it to distinguish prod
// from preview traffic in plausible
'console-pr-preview.vercel.app',
defer: true,
src: '/viewscript.js',
},
}
const previewMetaTag = [
{
injectTo: 'head' as const,
tag: 'meta',
attrs: {
property: 'og:image',
content: '/assets/og-preview-image.webp',
},
},
{
injectTo: 'head' as const,
tag: 'meta',
attrs: {
property: 'og:description',
content: 'Preview of the Oxide web console with in-browser mock API',
},
},
]
// vercel config is source of truth for headers
const vercelHeaders = vercelConfig.headers[0].headers
const headers = Object.fromEntries(vercelHeaders.map((h) => [h.key, h.value]))
// This is only needed for local dev to avoid breaking Vite's script injection.
// Rather than use unsafe-inline all the time, the nonce approach is much more
// narrowly scoped and lets us make sure everything *else* works fine without
// unsafe-inline.
const cspNonce = randomBytes(8).toString('hex')
const csp = headers['content-security-policy']
const devHeaders = {
...headers,
'content-security-policy': `${csp}; script-src 'nonce-${cspNonce}' 'self'`,
}
// see https://vitejs.dev/config/
export default defineConfig(({ mode }) => ({
build: {
outDir: resolve(__dirname, 'dist'),
emptyOutDir: true,
sourcemap: true,
// minify: false, // uncomment for debugging
rollupOptions: {
input: {
app: 'index.html',
},
},
// prevent inlining assets as `data:`, which is not permitted by our Content-Security-Policy
assetsInlineLimit: 0,
},
define: {
'process.env.MSW': JSON.stringify(apiMode === 'msw'),
// we don't want to have to look at this banner all day
'process.env.MSW_BANNER': JSON.stringify(apiMode === 'msw' && mode === 'production'),
// used in production build to console.log the SHA at page load
'process.env.SHA': JSON.stringify(process.env.SHA),
// used by MSW — number for % likelihood of API request failure (decimals allowed)
'process.env.CHAOS': JSON.stringify(mode !== 'production' && process.env.CHAOS),
},
plugins: [
tsconfigPaths(),
createHtmlPlugin({
inject: {
tags: process.env.VERCEL ? [previewAnalyticsTag, ...previewMetaTag] : [],
},
}),
react(),
apiMode === 'dogfood' && basicSsl(),
],
html: {
// don't include a placeholder nonce in production.
// use a CSP nonce in dev to avoid needing to permit 'unsafe-inline'
cspNonce: mode === 'production' ? undefined : cspNonce,
},
server: {
port: 4000,
headers: devHeaders,
// these only get hit when MSW doesn't intercept the request
proxy: {
'/v1': {
target:
apiMode === 'dogfood' ? `https://${DOGFOOD_HOST}` : 'http://localhost:12220',
changeOrigin: true,
},
'^/v1/instances/[^/]+/serial-console/stream': {
target:
// in msw mode, serial console is served by tools/deno/mock-serial-console.ts
apiMode === 'dogfood'
? `wss://${DOGFOOD_HOST}`
: 'ws://127.0.0.1:' + (apiMode === 'msw' ? 6036 : 12220),
changeOrigin: true,
ws: true,
},
},
},
preview: { headers },
test: {
environment: 'jsdom',
setupFiles: ['test/unit/setup.ts'],
includeSource: ['app/**/*.ts'],
},
}))