agent-message 0.1.3 → 0.1.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +97 -22
- package/codex-message/Cargo.lock +1867 -0
- package/codex-message/Cargo.toml +17 -0
- package/codex-message/README.md +29 -0
- package/codex-message/src/agent_message.rs +205 -0
- package/codex-message/src/app.rs +857 -0
- package/codex-message/src/codex.rs +278 -0
- package/codex-message/src/main.rs +101 -0
- package/codex-message/src/render.rs +125 -0
- package/npm/bin/agent-message.mjs +245 -13
- package/npm/bin/codex-message.mjs +41 -0
- package/npm/runtime/agent_gateway.mjs +5 -2
- package/npm/runtime/bin/agent-message-cli-darwin-amd64 +0 -0
- package/npm/runtime/bin/agent-message-cli-darwin-arm64 +0 -0
- package/npm/runtime/bin/agent-message-server-darwin-amd64 +0 -0
- package/npm/runtime/bin/agent-message-server-darwin-arm64 +0 -0
- package/npm/runtime/web-dist/assets/index-AfnEJAni.css +1 -0
- package/npm/runtime/web-dist/assets/index-QUUfdfoN.js +182 -0
- package/npm/runtime/web-dist/index.html +2 -2
- package/npm/runtime/web-dist/sw.js +2 -1
- package/package.json +4 -2
- package/npm/runtime/web-dist/assets/index-4VmoBZF3.js +0 -182
- package/npm/runtime/web-dist/assets/index-D_RPU5JN.css +0 -1
- package/npm/runtime/web-dist/workbox-8c29f6e4.js +0 -1
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
|
|
3
3
|
import { spawn, spawnSync } from 'node:child_process'
|
|
4
|
+
import { generateKeyPairSync } from 'node:crypto'
|
|
4
5
|
import { closeSync, existsSync, mkdirSync, openSync, readFileSync, rmSync, writeFileSync } from 'node:fs'
|
|
5
6
|
import { access, constants } from 'node:fs/promises'
|
|
6
7
|
import os from 'node:os'
|
|
@@ -15,6 +16,8 @@ const DEFAULT_WEB_PORT = 45788
|
|
|
15
16
|
const STARTUP_ATTEMPTS = 40
|
|
16
17
|
const STARTUP_DELAY_MS = 500
|
|
17
18
|
const PROCESS_STOP_DELAY_MS = 1000
|
|
19
|
+
const DEFAULT_WEB_PUSH_SUBJECT = 'mailto:agent-message@local.invalid'
|
|
20
|
+
const DEFAULT_TUNNEL_WEB_PUSH_SUBJECT = 'https://agent-message.namjaeyoun.com'
|
|
18
21
|
|
|
19
22
|
const scriptDir = dirname(fileURLToPath(import.meta.url))
|
|
20
23
|
const packageRoot = resolve(scriptDir, '..', '..')
|
|
@@ -22,6 +25,12 @@ const bundleRoot = resolve(packageRoot, 'npm', 'runtime')
|
|
|
22
25
|
const bundleBinDir = join(bundleRoot, 'bin')
|
|
23
26
|
const bundleGatewayPath = join(bundleRoot, 'agent_gateway.mjs')
|
|
24
27
|
const bundleWebDistDir = join(bundleRoot, 'web-dist')
|
|
28
|
+
const sourceServerDir = resolve(packageRoot, 'server')
|
|
29
|
+
const sourceWebDir = resolve(packageRoot, 'web')
|
|
30
|
+
const sourceGatewayPath = resolve(packageRoot, 'deploy', 'agent_gateway.mjs')
|
|
31
|
+
const sourceWebDistDir = resolve(sourceWebDir, 'dist')
|
|
32
|
+
const tunnelConfigPath = resolve(packageRoot, 'deploy', 'agent_tunnel_config.yml')
|
|
33
|
+
const tunnelName = 'agent-namjaeyoun-com'
|
|
25
34
|
|
|
26
35
|
const lifecycleCommands = new Set(['start', 'stop', 'status'])
|
|
27
36
|
|
|
@@ -39,7 +48,9 @@ async function main() {
|
|
|
39
48
|
|
|
40
49
|
if (lifecycleCommands.has(command)) {
|
|
41
50
|
const options = parseLifecycleOptions(rest)
|
|
42
|
-
|
|
51
|
+
if (!options.dev) {
|
|
52
|
+
await ensureBundleReady()
|
|
53
|
+
}
|
|
43
54
|
|
|
44
55
|
if (command === 'start') {
|
|
45
56
|
await startStack(options)
|
|
@@ -59,14 +70,16 @@ async function main() {
|
|
|
59
70
|
|
|
60
71
|
function printRootUsage() {
|
|
61
72
|
console.error(`Usage:
|
|
62
|
-
agent-message start [--runtime-dir <dir>] [--api-host <host>] [--api-port <port>] [--web-host <host>] [--web-port <port>]
|
|
63
|
-
agent-message stop [--runtime-dir <dir>]
|
|
64
|
-
agent-message status [--runtime-dir <dir>] [--api-host <host>] [--api-port <port>] [--web-host <host>] [--web-port <port>]
|
|
73
|
+
agent-message start [--dev] [--with-tunnel] [--runtime-dir <dir>] [--api-host <host>] [--api-port <port>] [--web-host <host>] [--web-port <port>]
|
|
74
|
+
agent-message stop [--dev] [--with-tunnel] [--runtime-dir <dir>]
|
|
75
|
+
agent-message status [--dev] [--runtime-dir <dir>] [--api-host <host>] [--api-port <port>] [--web-host <host>] [--web-port <port>]
|
|
65
76
|
agent-message <existing-cli-command> [...args]`)
|
|
66
77
|
}
|
|
67
78
|
|
|
68
79
|
function parseLifecycleOptions(args) {
|
|
69
80
|
const options = {
|
|
81
|
+
dev: false,
|
|
82
|
+
withTunnel: false,
|
|
70
83
|
runtimeDir: join(os.homedir(), '.agent-message'),
|
|
71
84
|
apiHost: DEFAULT_API_HOST,
|
|
72
85
|
apiPort: DEFAULT_API_PORT,
|
|
@@ -81,6 +94,14 @@ function parseLifecycleOptions(args) {
|
|
|
81
94
|
process.exit(0)
|
|
82
95
|
}
|
|
83
96
|
|
|
97
|
+
if (arg === '--dev') {
|
|
98
|
+
options.dev = true
|
|
99
|
+
continue
|
|
100
|
+
}
|
|
101
|
+
if (arg === '--with-tunnel' || arg === '--all') {
|
|
102
|
+
options.withTunnel = true
|
|
103
|
+
continue
|
|
104
|
+
}
|
|
84
105
|
if (arg === '--runtime-dir') {
|
|
85
106
|
options.runtimeDir = requireOptionValue(args, ++index, arg)
|
|
86
107
|
continue
|
|
@@ -164,20 +185,25 @@ function resolveBinaryPath(baseName) {
|
|
|
164
185
|
function runtimePaths(runtimeDir) {
|
|
165
186
|
return {
|
|
166
187
|
runtimeDir,
|
|
188
|
+
binDir: join(runtimeDir, 'bin'),
|
|
167
189
|
logDir: join(runtimeDir, 'logs'),
|
|
168
190
|
uploadDir: join(runtimeDir, 'uploads'),
|
|
169
191
|
serverLog: join(runtimeDir, 'logs', 'server.log'),
|
|
170
192
|
gatewayLog: join(runtimeDir, 'logs', 'gateway.log'),
|
|
193
|
+
tunnelLog: join(runtimeDir, 'logs', 'named-tunnel.log'),
|
|
171
194
|
serverPidfile: join(runtimeDir, 'server.pid'),
|
|
172
195
|
gatewayPidfile: join(runtimeDir, 'gateway.pid'),
|
|
196
|
+
tunnelPidfile: join(runtimeDir, 'named-tunnel.pid'),
|
|
173
197
|
stackMetadataPath: join(runtimeDir, 'stack.json'),
|
|
174
198
|
sqlitePath: join(runtimeDir, 'agent_message.sqlite'),
|
|
199
|
+
webPushConfigPath: join(runtimeDir, 'web-push.json'),
|
|
175
200
|
}
|
|
176
201
|
}
|
|
177
202
|
|
|
178
203
|
async function startStack(options) {
|
|
179
204
|
const paths = runtimePaths(options.runtimeDir)
|
|
180
205
|
mkdirSync(paths.runtimeDir, { recursive: true })
|
|
206
|
+
mkdirSync(paths.binDir, { recursive: true })
|
|
181
207
|
mkdirSync(paths.logDir, { recursive: true })
|
|
182
208
|
mkdirSync(paths.uploadDir, { recursive: true })
|
|
183
209
|
|
|
@@ -185,38 +211,55 @@ async function startStack(options) {
|
|
|
185
211
|
|
|
186
212
|
writeFileSync(paths.serverLog, '')
|
|
187
213
|
writeFileSync(paths.gatewayLog, '')
|
|
214
|
+
if (options.withTunnel) {
|
|
215
|
+
ensureTunnelTargetMatchesDefaults(options)
|
|
216
|
+
writeFileSync(paths.tunnelLog, '')
|
|
217
|
+
}
|
|
188
218
|
|
|
189
|
-
const
|
|
190
|
-
const gatewayChild = { pid: null }
|
|
219
|
+
const launchSpec = options.dev ? buildDevLaunchSpec(paths) : buildBundledLaunchSpec()
|
|
191
220
|
|
|
192
221
|
try {
|
|
193
|
-
const
|
|
222
|
+
const webPushConfig = ensureWebPushConfig(paths, options)
|
|
223
|
+
const serverChild = spawnDetached(launchSpec.serverCommand, launchSpec.serverArgs, {
|
|
194
224
|
...process.env,
|
|
195
225
|
SERVER_ADDR: `${options.apiHost}:${options.apiPort}`,
|
|
196
226
|
DB_DRIVER: 'sqlite',
|
|
197
227
|
SQLITE_DSN: paths.sqlitePath,
|
|
198
228
|
UPLOAD_DIR: paths.uploadDir,
|
|
199
229
|
CORS_ALLOWED_ORIGINS: '*',
|
|
230
|
+
WEB_PUSH_VAPID_PUBLIC_KEY: webPushConfig.publicKey,
|
|
231
|
+
WEB_PUSH_VAPID_PRIVATE_KEY: webPushConfig.privateKey,
|
|
232
|
+
WEB_PUSH_SUBJECT: webPushConfig.subject,
|
|
200
233
|
}, paths.serverLog)
|
|
201
234
|
writeFileSync(paths.serverPidfile, `${serverChild.pid}\n`)
|
|
202
235
|
|
|
203
236
|
await waitForHttp(`http://${options.apiHost}:${options.apiPort}/healthz`, 'API server')
|
|
204
237
|
|
|
205
|
-
gatewayChild
|
|
206
|
-
|
|
207
|
-
|
|
238
|
+
const gatewayChild = spawnDetached(
|
|
239
|
+
launchSpec.gatewayCommand,
|
|
240
|
+
launchSpec.gatewayArgs,
|
|
208
241
|
{
|
|
209
242
|
...process.env,
|
|
210
243
|
AGENT_GATEWAY_HOST: options.webHost,
|
|
211
244
|
AGENT_GATEWAY_PORT: String(options.webPort),
|
|
212
245
|
AGENT_API_ORIGIN: `http://${options.apiHost}:${options.apiPort}`,
|
|
213
|
-
AGENT_WEB_DIST:
|
|
246
|
+
AGENT_WEB_DIST: launchSpec.webDistDir,
|
|
214
247
|
},
|
|
215
248
|
paths.gatewayLog,
|
|
216
|
-
)
|
|
249
|
+
)
|
|
217
250
|
writeFileSync(paths.gatewayPidfile, `${gatewayChild.pid}\n`)
|
|
218
251
|
|
|
219
252
|
await waitForHttp(`http://${options.webHost}:${options.webPort}`, 'web gateway')
|
|
253
|
+
if (options.withTunnel) {
|
|
254
|
+
const tunnelChild = spawnDetached(
|
|
255
|
+
'cloudflared',
|
|
256
|
+
['tunnel', '--config', tunnelConfigPath, 'run', tunnelName],
|
|
257
|
+
process.env,
|
|
258
|
+
paths.tunnelLog,
|
|
259
|
+
)
|
|
260
|
+
writeFileSync(paths.tunnelPidfile, `${tunnelChild.pid}\n`)
|
|
261
|
+
await waitForLogMessage(paths.tunnelLog, 'Registered tunnel connection', 'named tunnel')
|
|
262
|
+
}
|
|
220
263
|
writeStackMetadata(paths.stackMetadataPath, options)
|
|
221
264
|
} catch (error) {
|
|
222
265
|
await stopStack(options, { quiet: true })
|
|
@@ -227,16 +270,122 @@ async function startStack(options) {
|
|
|
227
270
|
console.log(`API: http://${options.apiHost}:${options.apiPort}`)
|
|
228
271
|
console.log(`Web: http://${options.webHost}:${options.webPort}`)
|
|
229
272
|
console.log(`Logs: ${paths.serverLog} ${paths.gatewayLog}`)
|
|
273
|
+
console.log(`Web Push: ${paths.webPushConfigPath}`)
|
|
274
|
+
if (options.withTunnel) {
|
|
275
|
+
console.log(`Tunnel: https://agent.namjaeyoun.com`)
|
|
276
|
+
console.log(`Tunnel log: ${paths.tunnelLog}`)
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
function ensureWebPushConfig(paths, options) {
|
|
281
|
+
const envPublicKey = process.env.WEB_PUSH_VAPID_PUBLIC_KEY?.trim() ?? ''
|
|
282
|
+
const envPrivateKey = process.env.WEB_PUSH_VAPID_PRIVATE_KEY?.trim() ?? ''
|
|
283
|
+
const envSubject = process.env.WEB_PUSH_SUBJECT?.trim() ?? ''
|
|
284
|
+
const defaultSubject = resolveDefaultWebPushSubject(options)
|
|
285
|
+
|
|
286
|
+
if ((envPublicKey && !envPrivateKey) || (!envPublicKey && envPrivateKey)) {
|
|
287
|
+
throw new Error('WEB_PUSH_VAPID_PUBLIC_KEY and WEB_PUSH_VAPID_PRIVATE_KEY must be set together.')
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
if (envPublicKey && envPrivateKey) {
|
|
291
|
+
return {
|
|
292
|
+
publicKey: envPublicKey,
|
|
293
|
+
privateKey: envPrivateKey,
|
|
294
|
+
subject: envSubject || defaultSubject,
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
const stored = readStoredWebPushConfig(paths.webPushConfigPath)
|
|
299
|
+
if (stored) {
|
|
300
|
+
const storedSubject = stored.subject || ''
|
|
301
|
+
const shouldMigrateDefaultSubject =
|
|
302
|
+
storedSubject === '' ||
|
|
303
|
+
(storedSubject === DEFAULT_WEB_PUSH_SUBJECT && defaultSubject !== DEFAULT_WEB_PUSH_SUBJECT)
|
|
304
|
+
const resolved = {
|
|
305
|
+
publicKey: stored.publicKey,
|
|
306
|
+
privateKey: stored.privateKey,
|
|
307
|
+
subject: envSubject || (shouldMigrateDefaultSubject ? defaultSubject : storedSubject),
|
|
308
|
+
}
|
|
309
|
+
if (resolved.subject !== stored.subject) {
|
|
310
|
+
writeStoredWebPushConfig(paths.webPushConfigPath, resolved)
|
|
311
|
+
}
|
|
312
|
+
return resolved
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
const generated = generateWebPushConfig(envSubject || defaultSubject)
|
|
316
|
+
writeStoredWebPushConfig(paths.webPushConfigPath, generated)
|
|
317
|
+
return generated
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
function resolveDefaultWebPushSubject(options) {
|
|
321
|
+
if (options?.withTunnel) {
|
|
322
|
+
return DEFAULT_TUNNEL_WEB_PUSH_SUBJECT
|
|
323
|
+
}
|
|
324
|
+
return DEFAULT_WEB_PUSH_SUBJECT
|
|
325
|
+
}
|
|
326
|
+
|
|
327
|
+
function readStoredWebPushConfig(path) {
|
|
328
|
+
if (!existsSync(path)) {
|
|
329
|
+
return null
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
try {
|
|
333
|
+
const parsed = JSON.parse(readFileSync(path, 'utf8'))
|
|
334
|
+
const publicKey = typeof parsed.publicKey === 'string' ? parsed.publicKey.trim() : ''
|
|
335
|
+
const privateKey = typeof parsed.privateKey === 'string' ? parsed.privateKey.trim() : ''
|
|
336
|
+
const subject = typeof parsed.subject === 'string' ? parsed.subject.trim() : ''
|
|
337
|
+
if (!publicKey || !privateKey) {
|
|
338
|
+
return null
|
|
339
|
+
}
|
|
340
|
+
return {
|
|
341
|
+
publicKey,
|
|
342
|
+
privateKey,
|
|
343
|
+
subject: subject || DEFAULT_WEB_PUSH_SUBJECT,
|
|
344
|
+
}
|
|
345
|
+
} catch {
|
|
346
|
+
return null
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
function writeStoredWebPushConfig(path, config) {
|
|
351
|
+
writeFileSync(path, `${JSON.stringify(config, null, 2)}\n`)
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
function generateWebPushConfig(subject) {
|
|
355
|
+
const { privateKey, publicKey } = generateKeyPairSync('ec', {
|
|
356
|
+
namedCurve: 'prime256v1',
|
|
357
|
+
})
|
|
358
|
+
const privateJWK = privateKey.export({ format: 'jwk' })
|
|
359
|
+
const publicJWK = publicKey.export({ format: 'jwk' })
|
|
360
|
+
|
|
361
|
+
if (
|
|
362
|
+
typeof privateJWK.d !== 'string' ||
|
|
363
|
+
typeof publicJWK.x !== 'string' ||
|
|
364
|
+
typeof publicJWK.y !== 'string'
|
|
365
|
+
) {
|
|
366
|
+
throw new Error('failed to generate VAPID keys')
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
const x = Buffer.from(publicJWK.x, 'base64url')
|
|
370
|
+
const y = Buffer.from(publicJWK.y, 'base64url')
|
|
371
|
+
const publicKeyBytes = Buffer.concat([Buffer.from([0x04]), x, y])
|
|
372
|
+
|
|
373
|
+
return {
|
|
374
|
+
publicKey: publicKeyBytes.toString('base64url'),
|
|
375
|
+
privateKey: privateJWK.d,
|
|
376
|
+
subject,
|
|
377
|
+
}
|
|
230
378
|
}
|
|
231
379
|
|
|
232
380
|
async function stopStack(options, { quiet }) {
|
|
233
381
|
const paths = runtimePaths(options.runtimeDir)
|
|
234
382
|
const stoppedServer = await killFromPidfile(paths.serverPidfile)
|
|
235
383
|
const stoppedGateway = await killFromPidfile(paths.gatewayPidfile)
|
|
384
|
+
const stoppedTunnel = await killFromPidfile(paths.tunnelPidfile)
|
|
236
385
|
rmSync(paths.stackMetadataPath, { force: true })
|
|
237
386
|
|
|
238
387
|
if (!quiet) {
|
|
239
|
-
if (stoppedServer || stoppedGateway) {
|
|
388
|
+
if (stoppedServer || stoppedGateway || stoppedTunnel) {
|
|
240
389
|
console.log('Agent Message is stopped.')
|
|
241
390
|
} else {
|
|
242
391
|
console.log('Agent Message is not running.')
|
|
@@ -250,16 +399,68 @@ async function printStatus(options) {
|
|
|
250
399
|
const gatewayPid = readPidfile(paths.gatewayPidfile)
|
|
251
400
|
const serverRunning = serverPid !== null && isPidAlive(serverPid)
|
|
252
401
|
const gatewayRunning = gatewayPid !== null && isPidAlive(gatewayPid)
|
|
402
|
+
const tunnelPid = readPidfile(paths.tunnelPidfile)
|
|
403
|
+
const tunnelRunning = tunnelPid !== null && isPidAlive(tunnelPid)
|
|
253
404
|
const apiHealthy = serverRunning ? await isHttpReady(`http://${options.apiHost}:${options.apiPort}/healthz`) : false
|
|
254
405
|
const webHealthy = gatewayRunning ? await isHttpReady(`http://${options.webHost}:${options.webPort}`) : false
|
|
255
406
|
|
|
256
407
|
console.log(`API server: ${serverRunning ? 'running' : 'stopped'}${apiHealthy ? ' (healthy)' : ''}`)
|
|
257
408
|
console.log(`Web gateway: ${gatewayRunning ? 'running' : 'stopped'}${webHealthy ? ' (healthy)' : ''}`)
|
|
409
|
+
if (tunnelPid !== null) {
|
|
410
|
+
console.log(`Named tunnel: ${tunnelRunning ? 'running' : 'stopped'}`)
|
|
411
|
+
}
|
|
258
412
|
console.log(`Runtime dir: ${paths.runtimeDir}`)
|
|
259
413
|
console.log(`API URL: http://${options.apiHost}:${options.apiPort}`)
|
|
260
414
|
console.log(`Web URL: http://${options.webHost}:${options.webPort}`)
|
|
261
415
|
}
|
|
262
416
|
|
|
417
|
+
function buildBundledLaunchSpec() {
|
|
418
|
+
return {
|
|
419
|
+
serverCommand: resolveBinaryPath('agent-message-server'),
|
|
420
|
+
serverArgs: [],
|
|
421
|
+
gatewayCommand: process.execPath,
|
|
422
|
+
gatewayArgs: [bundleGatewayPath],
|
|
423
|
+
webDistDir: bundleWebDistDir,
|
|
424
|
+
}
|
|
425
|
+
}
|
|
426
|
+
|
|
427
|
+
function buildDevLaunchSpec(paths) {
|
|
428
|
+
ensureDevSourcesReady()
|
|
429
|
+
|
|
430
|
+
if (!existsSync(join(sourceWebDir, 'node_modules'))) {
|
|
431
|
+
runForeground('npm', ['ci'], { cwd: sourceWebDir })
|
|
432
|
+
}
|
|
433
|
+
runForeground('npm', ['run', 'build'], { cwd: sourceWebDir })
|
|
434
|
+
|
|
435
|
+
const serverBinaryPath = join(paths.binDir, 'agent-message-server')
|
|
436
|
+
runForeground('go', ['build', '-o', serverBinaryPath, '.'], { cwd: sourceServerDir })
|
|
437
|
+
|
|
438
|
+
return {
|
|
439
|
+
serverCommand: serverBinaryPath,
|
|
440
|
+
serverArgs: [],
|
|
441
|
+
gatewayCommand: process.execPath,
|
|
442
|
+
gatewayArgs: [sourceGatewayPath],
|
|
443
|
+
webDistDir: sourceWebDistDir,
|
|
444
|
+
}
|
|
445
|
+
}
|
|
446
|
+
|
|
447
|
+
function ensureDevSourcesReady() {
|
|
448
|
+
const requiredPaths = [sourceServerDir, sourceWebDir, sourceGatewayPath]
|
|
449
|
+
for (const target of requiredPaths) {
|
|
450
|
+
if (!existsSync(target)) {
|
|
451
|
+
throw new Error(`development mode requires a local checkout with ${target}`)
|
|
452
|
+
}
|
|
453
|
+
}
|
|
454
|
+
}
|
|
455
|
+
|
|
456
|
+
function ensureTunnelTargetMatchesDefaults(options) {
|
|
457
|
+
if (options.webHost !== DEFAULT_WEB_HOST || options.webPort !== DEFAULT_WEB_PORT) {
|
|
458
|
+
throw new Error(
|
|
459
|
+
`--with-tunnel requires the default web listener ${DEFAULT_WEB_HOST}:${DEFAULT_WEB_PORT} to match ${tunnelConfigPath}.`,
|
|
460
|
+
)
|
|
461
|
+
}
|
|
462
|
+
}
|
|
463
|
+
|
|
263
464
|
function delegateToBundledCli(args) {
|
|
264
465
|
const cliBinary = resolveBinaryPath('agent-message-cli')
|
|
265
466
|
const delegatedArgs = buildDelegatedCliArgs(args)
|
|
@@ -355,6 +556,27 @@ function spawnDetached(command, args, env, logFile) {
|
|
|
355
556
|
}
|
|
356
557
|
}
|
|
357
558
|
|
|
559
|
+
function runForeground(command, args, { cwd }) {
|
|
560
|
+
const result = spawnSync(command, args, {
|
|
561
|
+
cwd,
|
|
562
|
+
env: process.env,
|
|
563
|
+
stdio: 'inherit',
|
|
564
|
+
})
|
|
565
|
+
|
|
566
|
+
if (result.error) {
|
|
567
|
+
throw result.error
|
|
568
|
+
}
|
|
569
|
+
|
|
570
|
+
if (result.signal) {
|
|
571
|
+
process.kill(process.pid, result.signal)
|
|
572
|
+
return
|
|
573
|
+
}
|
|
574
|
+
|
|
575
|
+
if ((result.status ?? 1) !== 0) {
|
|
576
|
+
throw new Error(`${command} ${args.join(' ')} failed with exit code ${result.status ?? 1}`)
|
|
577
|
+
}
|
|
578
|
+
}
|
|
579
|
+
|
|
358
580
|
function readPidfile(pidfile) {
|
|
359
581
|
if (!existsSync(pidfile)) {
|
|
360
582
|
return null
|
|
@@ -409,6 +631,16 @@ async function waitForHttp(url, label) {
|
|
|
409
631
|
throw new Error(`${label} did not become ready: ${url}`)
|
|
410
632
|
}
|
|
411
633
|
|
|
634
|
+
async function waitForLogMessage(logFile, message, label) {
|
|
635
|
+
for (let attempt = 0; attempt < STARTUP_ATTEMPTS; attempt += 1) {
|
|
636
|
+
if (existsSync(logFile) && readFileSync(logFile, 'utf8').includes(message)) {
|
|
637
|
+
return
|
|
638
|
+
}
|
|
639
|
+
await sleep(STARTUP_DELAY_MS)
|
|
640
|
+
}
|
|
641
|
+
throw new Error(`${label} did not become ready: ${message}`)
|
|
642
|
+
}
|
|
643
|
+
|
|
412
644
|
async function isHttpReady(url) {
|
|
413
645
|
try {
|
|
414
646
|
const controller = new AbortController()
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
import { spawnSync } from 'node:child_process'
|
|
4
|
+
import { existsSync } from 'node:fs'
|
|
5
|
+
import { dirname, resolve } from 'node:path'
|
|
6
|
+
import process from 'node:process'
|
|
7
|
+
import { fileURLToPath } from 'node:url'
|
|
8
|
+
|
|
9
|
+
const scriptDir = dirname(fileURLToPath(import.meta.url))
|
|
10
|
+
const packageRoot = resolve(scriptDir, '..', '..')
|
|
11
|
+
const crateDir = resolve(packageRoot, 'codex-message')
|
|
12
|
+
const manifestPath = resolve(crateDir, 'Cargo.toml')
|
|
13
|
+
const binaryPath = resolve(crateDir, 'target', 'debug', process.platform === 'win32' ? 'codex-message.exe' : 'codex-message')
|
|
14
|
+
|
|
15
|
+
if (!existsSync(manifestPath)) {
|
|
16
|
+
console.error(`codex-message sources were not found at ${crateDir}`)
|
|
17
|
+
process.exit(1)
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
if (!existsSync(binaryPath)) {
|
|
21
|
+
const build = spawnSync('cargo', ['build', '--manifest-path', manifestPath], {
|
|
22
|
+
stdio: 'inherit',
|
|
23
|
+
cwd: packageRoot,
|
|
24
|
+
})
|
|
25
|
+
if (build.status !== 0) {
|
|
26
|
+
process.exit(build.status ?? 1)
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
const child = spawnSync(binaryPath, process.argv.slice(2), {
|
|
31
|
+
stdio: 'inherit',
|
|
32
|
+
cwd: process.cwd(),
|
|
33
|
+
env: process.env,
|
|
34
|
+
})
|
|
35
|
+
|
|
36
|
+
if (child.error) {
|
|
37
|
+
console.error(child.error.message)
|
|
38
|
+
process.exit(1)
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
process.exit(child.status ?? 0)
|
|
@@ -4,8 +4,8 @@ import http from 'node:http'
|
|
|
4
4
|
import { basename, extname, join, normalize, resolve } from 'node:path'
|
|
5
5
|
|
|
6
6
|
const host = process.env.AGENT_GATEWAY_HOST ?? '127.0.0.1'
|
|
7
|
-
const port = Number(process.env.AGENT_GATEWAY_PORT ?? '
|
|
8
|
-
const apiOrigin = process.env.AGENT_API_ORIGIN ?? 'http://127.0.0.1:
|
|
7
|
+
const port = Number(process.env.AGENT_GATEWAY_PORT ?? '45788')
|
|
8
|
+
const apiOrigin = process.env.AGENT_API_ORIGIN ?? 'http://127.0.0.1:8080'
|
|
9
9
|
const distDir = resolve(process.env.AGENT_WEB_DIST ?? join(process.cwd(), 'web', 'dist'))
|
|
10
10
|
const indexPath = join(distDir, 'index.html')
|
|
11
11
|
|
|
@@ -87,6 +87,9 @@ async function proxyRequest(req, res) {
|
|
|
87
87
|
upstream.status,
|
|
88
88
|
Object.fromEntries(upstream.headers.entries()),
|
|
89
89
|
)
|
|
90
|
+
if (upstream.headers.get('content-type')?.startsWith('text/event-stream')) {
|
|
91
|
+
res.flushHeaders()
|
|
92
|
+
}
|
|
90
93
|
|
|
91
94
|
if (!upstream.body) {
|
|
92
95
|
res.end()
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
._page_ymysk_1{height:100%;min-height:0;padding:0;background:radial-gradient(circle at top,#f5f9ff,#ebf2ff 34%,#e3edff);color:#10213f;overflow:hidden}._shell_ymysk_11{width:100%;height:100%;min-height:0;padding:0;display:grid;grid-template-rows:auto 1fr;gap:.9rem;overflow:hidden}._header_ymysk_22{padding:calc(1rem + env(safe-area-inset-top)) 1rem 1rem;border-radius:0 0 22px 22px;background:linear-gradient(180deg,#1f4fbf,#2e67dc);color:#fff;box-shadow:0 18px 34px -24px #132b6099}._headerTop_ymysk_30{display:flex;align-items:flex-start;justify-content:space-between;gap:.8rem}._eyebrow_ymysk_37{margin:0 0 .28rem;font-size:.74rem;letter-spacing:.12em;text-transform:uppercase;opacity:.78}._brand_ymysk_45{margin:0;font-size:1.4rem;line-height:1.1}._headerMeta_ymysk_51{margin-top:.75rem;display:flex;align-items:center;justify-content:space-between;gap:.75rem}._currentUser_ymysk_59{margin:0;font-size:.92rem;color:#ffffffe0}._statusBadge_ymysk_65{display:inline-flex;align-items:center;min-height:30px;padding:.22rem .65rem;border-radius:999px;background:#ffffff24;color:#fff;font-size:.78rem;font-weight:700;letter-spacing:.02em}._logoutButton_ymysk_78{border:1px solid rgba(255,255,255,.32);border-radius:999px;background:#ffffff24;color:#fff;padding:.46rem .8rem;font-size:.82rem;font-weight:600;cursor:pointer}._logoutButton_ymysk_78:disabled{opacity:.65;cursor:not-allowed}._listSection_ymysk_94{background:#ffffffdb;border:1px solid rgba(195,211,244,.9);border-radius:22px;padding:.95rem;box-shadow:0 14px 30px -26px #10213f73;-webkit-backdrop-filter:blur(10px);backdrop-filter:blur(10px)}._listSection_ymysk_94{min-height:0;display:grid;grid-template-rows:auto 1fr;overflow:hidden}._sectionHeading_ymysk_110{margin-bottom:.7rem}._sectionTitle_ymysk_114{margin:0;font-size:.96rem;color:#1b3464}._sectionCopy_ymysk_120{margin:.28rem 0 0;font-size:.82rem;color:#4a6291}._conversationList_ymysk_126{min-height:0;display:grid;align-content:start;grid-auto-rows:max-content;gap:.55rem;overflow:auto;overscroll-behavior:contain;-webkit-overflow-scrolling:touch}._conversationItem_ymysk_137{display:grid;gap:.34rem;padding:.88rem .9rem;border-radius:18px;border:1px solid #d9e4fb;background:linear-gradient(180deg,#fff,#f6f9ff);text-decoration:none;color:inherit;box-shadow:0 10px 24px -24px #10213f80}._conversationItem_ymysk_137:hover{background:linear-gradient(180deg,#fff,#eef4ff)}._conversationItemActive_ymysk_153{border-color:#557fdd;background:linear-gradient(180deg,#eff5ff,#e8f1ff);box-shadow:inset 0 0 0 1px #557fdd}._conversationMeta_ymysk_159{display:flex;align-items:baseline;justify-content:space-between;gap:.75rem}._conversationName_ymysk_166{font-size:.96rem;font-weight:700}._conversationTime_ymysk_171{font-size:.75rem;color:#4a5f8c}._conversationPreview_ymysk_176{margin:0;font-size:.85rem;color:#314a79;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}._conversationId_ymysk_185{display:none}._statusText_ymysk_189{margin:.7rem 0 0;font-size:.82rem;color:#445e91}._errorMessage_ymysk_195{margin:.7rem 0 0;font-size:.82rem;color:#9b2035}._emptyState_ymysk_201{padding:1.1rem .4rem .5rem;text-align:center}._emptyTitle_ymysk_206{margin:0;color:#1b3464;font-weight:700}._emptyCopy_ymysk_212{margin:.42rem 0 0;color:#5a709d;font-size:.84rem}@media (min-width: 768px){._page_ymysk_1{height:100%;padding:1rem}._shell_ymysk_11{max-width:560px;height:100%;margin:0 auto;padding:0}._header_ymysk_22{border-radius:24px}}._root_ael8t_1{margin-top:.42rem;width:100%;max-width:100%;min-width:0;overflow-wrap:anywhere;word-break:break-word}._root_ael8t_1 [data-slot=card],._root_ael8t_1 [data-slot=card-header],._root_ael8t_1 [data-slot=card-content]{min-width:0;max-width:100%}._root_ael8t_1 [data-slot=card-title],._root_ael8t_1 [data-slot=card-description],._root_ael8t_1 p,._root_ael8t_1 code{overflow-wrap:anywhere;word-break:break-word}._fallback_ael8t_25{margin:0;font-size:.84rem;color:#4f6286}._page_12q5i_1{position:fixed;top:0;right:0;bottom:0;left:0;width:100%;min-height:0;background:radial-gradient(circle at top,#f5f9ff,#edf3ff 28%,#e4edff);padding:0;overflow:hidden;overscroll-behavior:none}._panel_12q5i_13{height:100%;min-height:0;display:grid;grid-template-rows:auto 1fr;overflow:hidden}._header_12q5i_21{position:sticky;top:0;z-index:5;padding:calc(.75rem + env(safe-area-inset-top)) 1rem .75rem;border-bottom:1px solid rgba(207,219,245,.9);background:#ffffffe0;-webkit-backdrop-filter:blur(14px);backdrop-filter:blur(14px)}._headerBar_12q5i_31{display:flex;align-items:center;gap:.6rem;min-width:0}._backButton_12q5i_38{border:1px solid #c9d8f8;border-radius:999px;background:#f7faff;color:#20406f;font-size:.78rem;font-weight:700;padding:.38rem .68rem;cursor:pointer;flex-shrink:0}._title_12q5i_50{margin:0;min-width:0;flex:1;font-size:1rem;color:#152b53;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}._headerStatusBadge_12q5i_61{flex-shrink:0;border:1px solid #c7d9fb;border-radius:999px;background:#eef4ff;color:#2b4f8a;font-size:.75rem;font-weight:700;line-height:1;padding:.34rem .52rem}._headerStatusBadgeError_12q5i_73{border-color:#f1c4cc;background:#fff4f6;color:#8c1d33}._muted_12q5i_79{margin:.42rem 0 0;color:#486293;font-size:.84rem}._error_12q5i_85{margin:.52rem 0 0;color:#98233a;font-size:.86rem}._timelineSection_12q5i_91{min-height:0;padding:0;display:grid;grid-template-rows:minmax(0,1fr) auto;gap:0;overflow:hidden}._timelineViewport_12q5i_100{min-height:0;position:relative;overflow:auto;overscroll-behavior:contain;-webkit-overflow-scrolling:touch;border:0;border-radius:0;background:transparent;padding:0;box-shadow:none}._timelineContent_12q5i_113{min-height:100%;padding:.8rem 1rem 1rem}._timelinePullStatus_12q5i_118{margin:0 0 .7rem;color:#486293;font-size:.78rem;text-align:center}._timelineList_12q5i_125{margin:0;padding:0;list-style:none;display:grid;gap:.62rem}._timelineItem_12q5i_133{display:flex;min-width:0}._timelineItemOwn_12q5i_138{justify-content:flex-end}._timelineItemOther_12q5i_142{justify-content:flex-start}._messageBubble_12q5i_146{width:min(100%,31rem);min-width:0;border:1px solid #d9e4fb;border-radius:18px;background:#fff;padding:.6rem .68rem;box-shadow:0 12px 22px -24px #10213f73}._timelineItemOwn_12q5i_138 ._messageBubble_12q5i_146{background:linear-gradient(180deg,#edf4ff,#e4efff);border-color:#c8dafc}._timelineMeta_12q5i_161{display:flex;align-items:baseline;justify-content:space-between;gap:.7rem}._timelineMetaRight_12q5i_168{display:inline-flex;align-items:center;gap:.36rem}._sender_12q5i_174{font-weight:700;color:#1e355f;font-size:.84rem}._timestamp_12q5i_180{color:#506998;font-size:.74rem}._editedBadge_12q5i_185{color:#395b96;font-size:.74rem}._messageActionsTrigger_12q5i_190{border:0;border-radius:8px;background:transparent;color:#345990;font-size:1rem;line-height:1;padding:.08rem .3rem;cursor:pointer}._messageActionsTrigger_12q5i_190:hover{background:#dbe8ff}._messageText_12q5i_205{margin:.42rem 0 0;font-size:.9rem;color:#1c335d;white-space:pre-wrap;word-break:break-word}._messageTextDeleted_12q5i_213{color:#5f6880;font-style:italic}._imageAttachmentLink_12q5i_218{margin-top:.5rem;display:inline-block}._imageAttachment_12q5i_218{max-width:min(100%,18rem);max-height:15rem;border:1px solid #cbd9f6;border-radius:14px;object-fit:cover;display:block}._fileAttachmentLink_12q5i_232{margin-top:.52rem;display:inline-flex;align-items:center;border:1px solid #bfceef;border-radius:999px;background:#f6f9ff;color:#1f3e73;text-decoration:none;font-size:.82rem;font-weight:700;padding:.38rem .62rem}._fileAttachmentLink_12q5i_232:hover{background:#eef4ff}._reactionSection_12q5i_250{margin-top:.52rem;display:grid;gap:.38rem}._reactionGroups_12q5i_256{display:flex;align-items:center;gap:.35rem;flex-wrap:wrap}._reactionChip_12q5i_263{border:1px solid #bfd0f2;border-radius:999px;background:#f5f9ff;color:#1e3b6d;display:inline-flex;align-items:center;gap:.24rem;font-size:.8rem;padding:.2rem .48rem}._reactionChipOwn_12q5i_275{border-color:#5a80d8;background:#e5efff;color:#1b3f7c}._unreadBanner_12q5i_281{position:absolute;left:50%;transform:translate(-50%);bottom:calc(100% + .5rem);z-index:8;border:1px solid #c5d6f7;border-radius:999px;background:#fff;color:#1e355f;font-size:.82rem;font-weight:700;padding:.44rem .9rem;cursor:pointer;box-shadow:0 6px 18px -6px #10213f59;white-space:nowrap;animation:_unreadSlideUp_12q5i_1 .2s ease-out}._unreadBanner_12q5i_281:hover{background:#eef4ff}@keyframes _unreadSlideUp_12q5i_1{0%{opacity:0;transform:translate(-50%) translateY(8px)}to{opacity:1;transform:translate(-50%) translateY(0)}}._composerForm_12q5i_315{position:relative;bottom:auto;z-index:0;border-top:1px solid #dce6fb;border-right:0;border-bottom:0;border-left:0;border-radius:0;background:#ffffffe6;padding:.8rem 1rem max(.45rem,calc(env(safe-area-inset-bottom) + .1rem));display:grid;gap:.55rem;-webkit-backdrop-filter:blur(12px);backdrop-filter:blur(12px);box-shadow:none}._editingBanner_12q5i_332{display:flex;align-items:center;justify-content:space-between;gap:.7rem;font-size:.82rem;color:#23457f}._editingCancelButton_12q5i_341{border:1px solid #b9ccec;border-radius:999px;background:#fff;color:#2a487f;font-size:.78rem;padding:.28rem .5rem;cursor:pointer}._composerInput_12q5i_351{width:100%;border:1px solid #c2d2f4;border-radius:16px;padding:.72rem .78rem;background:#fff;color:#122447;resize:vertical;min-height:72px}._composerInput_12q5i_351:focus{outline:2px solid #9ab3ef;border-color:#5d80db}._composerControls_12q5i_367{display:flex;align-items:center;justify-content:space-between;gap:.6rem;flex-wrap:wrap}._attachLabel_12q5i_375{border:1px solid #beceee;border-radius:999px;background:#f6f9ff;color:#1f3f78;font-size:.82rem;font-weight:700;padding:.4rem .62rem;cursor:pointer}._attachLabelDisabled_12q5i_386{opacity:.6;cursor:not-allowed}._attachInput_12q5i_391{display:none}._attachmentChip_12q5i_395{display:inline-flex;align-items:center;gap:.4rem;border:1px solid #c8d9f7;border-radius:999px;background:#fff;padding:.2rem .42rem .2rem .54rem}._attachmentName_12q5i_405{font-size:.78rem;color:#2b477b;max-width:13rem;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}._attachmentRemove_12q5i_414{border:0;border-radius:999px;background:#dce8ff;color:#2f4f89;font-size:.72rem;padding:.16rem .36rem;cursor:pointer}._submitButton_12q5i_424{border:0;border-radius:999px;background:linear-gradient(180deg,#2454d6,#163b9e);color:#fff;font-size:.86rem;font-weight:700;padding:.5rem .9rem;cursor:pointer}._submitButton_12q5i_424:disabled{opacity:.62;cursor:not-allowed}._contextMenu_12q5i_440{position:fixed;z-index:20;display:grid;gap:.22rem;min-width:11rem;padding:.36rem;border:1px solid #c5d6f7;border-radius:14px;background:#fff;box-shadow:0 18px 28px -18px #132f636b}._contextMenuAction_12q5i_453{border:0;border-radius:10px;text-align:left;background:#f4f8ff;color:#203d70;padding:.48rem .56rem;font-size:.84rem;cursor:pointer}._contextMenuAction_12q5i_453:hover{background:#eaf2ff}._contextMenuActionDanger_12q5i_468{color:#8c1d33;background:#fff4f6}._contextMenuActionDanger_12q5i_468:hover{background:#ffe7ec}._card_12q5i_477{width:min(100%,38rem);margin:1rem auto;padding:1.2rem;border:1px solid #d4e2fb;border-radius:18px;background:#fff;box-shadow:0 14px 28px -24px #10357959}@media (min-width: 768px){._page_12q5i_1{padding:1rem}._panel_12q5i_13{max-width:56rem;margin:0 auto;min-height:calc(100vh - 2rem);border:1px solid rgba(207,219,245,.9);border-radius:28px;overflow:hidden;background:#ffffff59;box-shadow:0 24px 44px -34px #10213f70}._messageBubble_12q5i_146{width:100%;max-width:none}}._wrapper_36fc3_1{width:min(100%,440px);margin:3rem auto;padding:1.5rem;border:1px solid #d6dff4;border-radius:14px;background:#fff;box-shadow:0 12px 30px -24px #1d3e8a}._title_36fc3_11{margin:0 0 .4rem}._subtitle_36fc3_15{margin:0 0 1rem;color:#344b74}._form_36fc3_20{display:grid;gap:.8rem}._field_36fc3_25{display:grid;gap:.35rem}._label_36fc3_30{font-size:.9rem;font-weight:600}._input_36fc3_35{border:1px solid #c4d2f1;border-radius:8px;padding:.55rem .7rem;font-size:.95rem}._input_36fc3_35:focus{outline:2px solid #a8bdf0;border-color:#5b7dd9}._submit_36fc3_47{border:0;border-radius:8px;padding:.65rem .9rem;background:#2454d6;color:#fff;font-weight:600;cursor:pointer}._submit_36fc3_47:disabled{opacity:.6;cursor:not-allowed}._hint_36fc3_62{margin:0;font-size:.85rem;color:#455c88}._error_36fc3_68{margin:0;font-size:.9rem;color:#9d1930}._success_36fc3_74{margin:0;font-size:.9rem;color:#1d6a3b}._card_1jmko_1{width:min(100%,680px);padding:1.5rem;border:1px solid #d6dff4;border-radius:14px;background:#fff;box-shadow:0 12px 30px -24px #1d3e8a}._card_1jmko_1 h2{margin-top:0;margin-bottom:.6rem}._card_1jmko_1 p{margin-top:0;margin-bottom:1rem;line-height:1.5}._link_1jmko_21{color:#2454d6;text-decoration:none;font-weight:600}._link_1jmko_21:hover{text-decoration:underline}._app_19urc_1{width:100%;height:100%;overflow:hidden}/*! tailwindcss v4.2.2 | MIT License | https://tailwindcss.com */@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-translate-x:0;--tw-translate-y:0;--tw-translate-z:0;--tw-rotate-x:initial;--tw-rotate-y:initial;--tw-rotate-z:initial;--tw-skew-x:initial;--tw-skew-y:initial;--tw-space-y-reverse:0;--tw-space-x-reverse:0;--tw-border-style:solid;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-outline-style:solid;--tw-blur:initial;--tw-brightness:initial;--tw-contrast:initial;--tw-grayscale:initial;--tw-hue-rotate:initial;--tw-invert:initial;--tw-opacity:initial;--tw-saturate:initial;--tw-sepia:initial;--tw-drop-shadow:initial;--tw-drop-shadow-color:initial;--tw-drop-shadow-alpha:100%;--tw-drop-shadow-size:initial;--tw-duration:initial;--tw-content:""}}}@layer theme{:root,:host{--font-sans:ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";--font-mono:ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--color-yellow-50:oklch(98.7% .026 102.212);--color-yellow-100:oklch(97.3% .071 103.193);--color-yellow-200:oklch(94.5% .129 101.54);--color-yellow-800:oklch(47.6% .114 61.907);--color-yellow-900:oklch(42.1% .095 57.708);--color-yellow-950:oklch(28.6% .066 53.813);--color-green-50:oklch(98.2% .018 155.826);--color-green-100:oklch(96.2% .044 156.743);--color-green-200:oklch(92.5% .084 155.995);--color-green-800:oklch(44.8% .119 151.328);--color-green-900:oklch(39.3% .095 152.535);--color-green-950:oklch(26.6% .065 152.934);--color-blue-50:oklch(97% .014 254.604);--color-blue-100:oklch(93.2% .032 255.585);--color-blue-200:oklch(88.2% .059 254.128);--color-blue-800:oklch(42.4% .199 265.638);--color-blue-900:oklch(37.9% .146 265.522);--color-blue-950:oklch(28.2% .091 267.935);--color-black:#000;--color-white:#fff;--spacing:.25rem;--container-xs:20rem;--container-sm:24rem;--container-md:28rem;--container-lg:32rem;--text-xs:.75rem;--text-xs--line-height:calc(1 / .75);--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--text-base:1rem;--text-base--line-height: 1.5 ;--text-lg:1.125rem;--text-lg--line-height:calc(1.75 / 1.125);--text-xl:1.25rem;--text-xl--line-height:calc(1.75 / 1.25);--text-2xl:1.5rem;--text-2xl--line-height:calc(2 / 1.5);--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--tracking-tight:-.025em;--tracking-widest:.1em;--leading-relaxed:1.625;--radius-xs:.125rem;--animate-spin:spin 1s linear infinite;--animate-pulse:pulse 2s cubic-bezier(.4, 0, .6, 1) infinite;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4, 0, .2, 1);--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono)}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;-moz-tab-size:4;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab,red,red)){::placeholder{color:color-mix(in oklab,currentcolor 50%,transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){-webkit-appearance:button;-moz-appearance:button;appearance:button}::file-selector-button{-webkit-appearance:button;-moz-appearance:button;appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.\@container\/card-header{container:card-header/inline-size}.pointer-events-none{pointer-events:none}.visible{visibility:visible}.sr-only{clip-path:inset(50%);white-space:nowrap;border-width:0;width:1px;height:1px;margin:-1px;padding:0;position:absolute;overflow:hidden}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.inset-0{inset:calc(var(--spacing) * 0)}.start{inset-inline-start:var(--spacing)}.end{inset-inline-end:var(--spacing)}.-top-12{top:calc(var(--spacing) * -12)}.top-1\/2{top:50%}.top-4{top:calc(var(--spacing) * 4)}.top-\[50\%\]{top:50%}.-right-12{right:calc(var(--spacing) * -12)}.right-2{right:calc(var(--spacing) * 2)}.right-4{right:calc(var(--spacing) * 4)}.-bottom-12{bottom:calc(var(--spacing) * -12)}.-left-12{left:calc(var(--spacing) * -12)}.left-1\/2{left:50%}.left-2{left:calc(var(--spacing) * 2)}.left-\[50\%\]{left:50%}.z-50{z-index:50}.col-start-2{grid-column-start:2}.container{width:100%}@media (min-width:40rem){.container{max-width:40rem}}@media (min-width:48rem){.container{max-width:48rem}}@media (min-width:64rem){.container{max-width:64rem}}@media (min-width:80rem){.container{max-width:80rem}}@media (min-width:96rem){.container{max-width:96rem}}.-mx-1{margin-inline:calc(var(--spacing) * -1)}.mx-2{margin-inline:calc(var(--spacing) * 2)}.mx-auto{margin-inline:auto}.my-1{margin-block:calc(var(--spacing) * 1)}.my-3{margin-block:calc(var(--spacing) * 3)}.-mt-4{margin-top:calc(var(--spacing) * -4)}.mt-4{margin-top:calc(var(--spacing) * 4)}.mt-auto{margin-top:auto}.mb-1{margin-bottom:calc(var(--spacing) * 1)}.-ml-4{margin-left:calc(var(--spacing) * -4)}.ml-auto{margin-left:auto}.line-clamp-1{-webkit-line-clamp:1;-webkit-box-orient:vertical;display:-webkit-box;overflow:hidden}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline-flex{display:inline-flex}.table{display:table}.table-caption{display:table-caption}.table-cell{display:table-cell}.table-row{display:table-row}.field-sizing-content{field-sizing:content}.aspect-square{aspect-ratio:1}.size-2{width:calc(var(--spacing) * 2);height:calc(var(--spacing) * 2)}.size-2\.5{width:calc(var(--spacing) * 2.5);height:calc(var(--spacing) * 2.5)}.size-3\.5{width:calc(var(--spacing) * 3.5);height:calc(var(--spacing) * 3.5)}.size-4{width:calc(var(--spacing) * 4);height:calc(var(--spacing) * 4)}.size-6{width:calc(var(--spacing) * 6);height:calc(var(--spacing) * 6)}.size-8{width:calc(var(--spacing) * 8);height:calc(var(--spacing) * 8)}.size-9{width:calc(var(--spacing) * 9);height:calc(var(--spacing) * 9)}.size-10{width:calc(var(--spacing) * 10);height:calc(var(--spacing) * 10)}.size-full{width:100%;height:100%}.h-2{height:calc(var(--spacing) * 2)}.h-4{height:calc(var(--spacing) * 4)}.h-6{height:calc(var(--spacing) * 6)}.h-8{height:calc(var(--spacing) * 8)}.h-9{height:calc(var(--spacing) * 9)}.h-10{height:calc(var(--spacing) * 10)}.h-12{height:calc(var(--spacing) * 12)}.h-\[calc\(100\%-1px\)\]{height:calc(100% - 1px)}.h-\[var\(--radix-select-trigger-height\)\]{height:var(--radix-select-trigger-height)}.h-auto{height:auto}.h-full{height:100%}.h-px{height:1px}.max-h-\(--radix-dropdown-menu-content-available-height\){max-height:var(--radix-dropdown-menu-content-available-height)}.max-h-\(--radix-select-content-available-height\){max-height:var(--radix-select-content-available-height)}.min-h-4{min-height:calc(var(--spacing) * 4)}.min-h-16{min-height:calc(var(--spacing) * 16)}.w-4{width:calc(var(--spacing) * 4)}.w-6{width:calc(var(--spacing) * 6)}.w-8{width:calc(var(--spacing) * 8)}.w-10{width:calc(var(--spacing) * 10)}.w-12{width:calc(var(--spacing) * 12)}.w-64{width:calc(var(--spacing) * 64)}.w-72{width:calc(var(--spacing) * 72)}.w-\[100px\]{width:100px}.w-auto{width:auto}.w-fit{width:fit-content}.w-full{width:100%}.max-w-\[calc\(100\%-2rem\)\]{max-width:calc(100% - 2rem)}.max-w-full{max-width:100%}.max-w-md{max-width:var(--container-md)}.max-w-sm{max-width:var(--container-sm)}.max-w-xs{max-width:var(--container-xs)}.min-w-0{min-width:calc(var(--spacing) * 0)}.min-w-8{min-width:calc(var(--spacing) * 8)}.min-w-9{min-width:calc(var(--spacing) * 9)}.min-w-10{min-width:calc(var(--spacing) * 10)}.min-w-\[8rem\]{min-width:8rem}.min-w-\[var\(--radix-select-trigger-width\)\]{min-width:var(--radix-select-trigger-width)}.flex-1{flex:1}.shrink-0{flex-shrink:0}.grow{flex-grow:1}.grow-0{flex-grow:0}.basis-3\/4{flex-basis:75%}.basis-full{flex-basis:100%}.caption-bottom{caption-side:bottom}.origin-\(--radix-dropdown-menu-content-transform-origin\){transform-origin:var(--radix-dropdown-menu-content-transform-origin)}.origin-\(--radix-popover-content-transform-origin\){transform-origin:var(--radix-popover-content-transform-origin)}.origin-\(--radix-select-content-transform-origin\){transform-origin:var(--radix-select-content-transform-origin)}.origin-\(--radix-tooltip-content-transform-origin\){transform-origin:var(--radix-tooltip-content-transform-origin)}.-translate-x-1\/2{--tw-translate-x: -50% ;translate:var(--tw-translate-x) var(--tw-translate-y)}.translate-x-\[-50\%\]{--tw-translate-x:-50%;translate:var(--tw-translate-x) var(--tw-translate-y)}.-translate-y-1\/2{--tw-translate-y: -50% ;translate:var(--tw-translate-x) var(--tw-translate-y)}.translate-y-0\.5{--tw-translate-y:calc(var(--spacing) * .5);translate:var(--tw-translate-x) var(--tw-translate-y)}.translate-y-\[-50\%\]{--tw-translate-y:-50%;translate:var(--tw-translate-x) var(--tw-translate-y)}.translate-y-\[calc\(-50\%_-_2px\)\]{--tw-translate-y: calc(-50% - 2px) ;translate:var(--tw-translate-x) var(--tw-translate-y)}.rotate-45{rotate:45deg}.rotate-90{rotate:90deg}.rotate-180{rotate:180deg}.transform{transform:var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,)}.animate-pulse{animation:var(--animate-pulse)}.animate-spin{animation:var(--animate-spin)}.cursor-default{cursor:default}.cursor-help{cursor:help}.cursor-pointer{cursor:pointer}.touch-none{touch-action:none}.scroll-my-1{scroll-margin-block:calc(var(--spacing) * 1)}.auto-rows-min{grid-auto-rows:min-content}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.grid-cols-6{grid-template-columns:repeat(6,minmax(0,1fr))}.grid-cols-\[0_1fr\]{grid-template-columns:0 1fr}.grid-rows-\[auto_auto\]{grid-template-rows:auto auto}.flex-col{flex-direction:column}.flex-col-reverse{flex-direction:column-reverse}.flex-row{flex-direction:row}.flex-wrap{flex-wrap:wrap}.place-content-center{place-content:center}.items-center{align-items:center}.items-end{align-items:flex-end}.items-start{align-items:flex-start}.items-stretch{align-items:stretch}.justify-around{justify-content:space-around}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.justify-end{justify-content:flex-end}.justify-items-start{justify-items:start}.gap-0{gap:calc(var(--spacing) * 0)}.gap-0\.5{gap:calc(var(--spacing) * .5)}.gap-1{gap:calc(var(--spacing) * 1)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-3{gap:calc(var(--spacing) * 3)}.gap-4{gap:calc(var(--spacing) * 4)}.gap-6{gap:calc(var(--spacing) * 6)}.gap-\[--spacing\(var\(--gap\)\)\]{gap:calc(var(--spacing) * var(--gap))}:where(.space-y-1>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 2) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 2) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-x-2>:not(:last-child)){--tw-space-x-reverse:0;margin-inline-start:calc(calc(var(--spacing) * 2) * var(--tw-space-x-reverse));margin-inline-end:calc(calc(var(--spacing) * 2) * calc(1 - var(--tw-space-x-reverse)))}.gap-y-0\.5{row-gap:calc(var(--spacing) * .5)}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-x-hidden{overflow-x:hidden}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-\[2px\]{border-radius:2px}.rounded-\[4px\]{border-radius:4px}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--radius)}.rounded-md{border-radius:calc(var(--radius) - 2px)}.rounded-sm{border-radius:calc(var(--radius) - 4px)}.rounded-xl{border-radius:calc(var(--radius) + 4px)}.rounded-xs{border-radius:var(--radius-xs)}.rounded-l-md{border-top-left-radius:calc(var(--radius) - 2px);border-bottom-left-radius:calc(var(--radius) - 2px)}.rounded-r-md{border-top-right-radius:calc(var(--radius) - 2px);border-bottom-right-radius:calc(var(--radius) - 2px)}.border{border-style:var(--tw-border-style);border-width:1px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-l{border-left-style:var(--tw-border-style);border-left-width:1px}.border-blue-200{border-color:var(--color-blue-200)}.border-border{border-color:var(--border)}.border-green-200{border-color:var(--color-green-200)}.border-input{border-color:var(--input)}.border-primary{border-color:var(--primary)}.border-transparent{border-color:#0000}.border-yellow-200{border-color:var(--color-yellow-200)}.bg-accent{background-color:var(--accent)}.bg-background{background-color:var(--background)}.bg-black\/50{background-color:#00000080}@supports (color:color-mix(in lab,red,red)){.bg-black\/50{background-color:color-mix(in oklab,var(--color-black) 50%,transparent)}}.bg-blue-50{background-color:var(--color-blue-50)}.bg-border{background-color:var(--border)}.bg-card{background-color:var(--card)}.bg-destructive{background-color:var(--destructive)}.bg-foreground{background-color:var(--foreground)}.bg-green-50{background-color:var(--color-green-50)}.bg-muted,.bg-muted\/50{background-color:var(--muted)}@supports (color:color-mix(in lab,red,red)){.bg-muted\/50{background-color:color-mix(in oklab,var(--muted) 50%,transparent)}}.bg-primary,.bg-primary\/20{background-color:var(--primary)}@supports (color:color-mix(in lab,red,red)){.bg-primary\/20{background-color:color-mix(in oklab,var(--primary) 20%,transparent)}}.bg-secondary{background-color:var(--secondary)}.bg-transparent{background-color:#0000}.bg-white{background-color:var(--color-white)}.bg-yellow-50{background-color:var(--color-yellow-50)}.fill-current{fill:currentColor}.fill-foreground{fill:var(--foreground)}.fill-primary{fill:var(--primary)}.p-1{padding:calc(var(--spacing) * 1)}.p-2{padding:calc(var(--spacing) * 2)}.p-4{padding:calc(var(--spacing) * 4)}.p-6{padding:calc(var(--spacing) * 6)}.p-\[3px\]{padding:3px}.px-1\.5{padding-inline:calc(var(--spacing) * 1.5)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-2\.5{padding-inline:calc(var(--spacing) * 2.5)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-4{padding-inline:calc(var(--spacing) * 4)}.px-6{padding-inline:calc(var(--spacing) * 6)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.py-1{padding-block:calc(var(--spacing) * 1)}.py-1\.5{padding-block:calc(var(--spacing) * 1.5)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-3{padding-block:calc(var(--spacing) * 3)}.py-4{padding-block:calc(var(--spacing) * 4)}.py-6{padding-block:calc(var(--spacing) * 6)}.pt-0{padding-top:calc(var(--spacing) * 0)}.pt-2{padding-top:calc(var(--spacing) * 2)}.pt-4{padding-top:calc(var(--spacing) * 4)}.pr-2{padding-right:calc(var(--spacing) * 2)}.pr-8{padding-right:calc(var(--spacing) * 8)}.pb-4{padding-bottom:calc(var(--spacing) * 4)}.pl-2{padding-left:calc(var(--spacing) * 2)}.pl-4{padding-left:calc(var(--spacing) * 4)}.pl-8{padding-left:calc(var(--spacing) * 8)}.text-center{text-align:center}.text-left{text-align:left}.align-middle{vertical-align:middle}.font-mono{font-family:var(--font-mono)}.text-2xl{font-size:var(--text-2xl);line-height:var(--tw-leading,var(--text-2xl--line-height))}.text-base{font-size:var(--text-base);line-height:var(--tw-leading,var(--text-base--line-height))}.text-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.leading-none{--tw-leading:1;line-height:1}.font-bold{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-tight{--tw-tracking:var(--tracking-tight);letter-spacing:var(--tracking-tight)}.tracking-widest{--tw-tracking:var(--tracking-widest);letter-spacing:var(--tracking-widest)}.text-balance{text-wrap:balance}.whitespace-nowrap{white-space:nowrap}.text-background{color:var(--background)}.text-blue-900{color:var(--color-blue-900)}.text-card-foreground{color:var(--card-foreground)}.text-current{color:currentColor}.text-destructive{color:var(--destructive)}.text-foreground,.text-foreground\/60{color:var(--foreground)}@supports (color:color-mix(in lab,red,red)){.text-foreground\/60{color:color-mix(in oklab,var(--foreground) 60%,transparent)}}.text-green-900{color:var(--color-green-900)}.text-muted-foreground{color:var(--muted-foreground)}.text-primary{color:var(--primary)}.text-primary-foreground{color:var(--primary-foreground)}.text-secondary-foreground{color:var(--secondary-foreground)}.text-white{color:var(--color-white)}.text-yellow-900{color:var(--color-yellow-900)}.underline{text-decoration-line:underline}.decoration-dotted{text-decoration-style:dotted}.underline-offset-4{text-underline-offset:4px}.opacity-25{opacity:.25}.opacity-50{opacity:.5}.opacity-70{opacity:.7}.opacity-75{opacity:.75}.shadow-lg{--tw-shadow:0 10px 15px -3px var(--tw-shadow-color,#0000001a), 0 4px 6px -4px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-md{--tw-shadow:0 4px 6px -1px var(--tw-shadow-color,#0000001a), 0 2px 4px -2px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-sm{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a), 0 1px 2px -1px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-xs{--tw-shadow:0 1px 2px 0 var(--tw-shadow-color,#0000000d);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.ring-0{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.ring-ring\/50{--tw-ring-color:var(--ring)}@supports (color:color-mix(in lab,red,red)){.ring-ring\/50{--tw-ring-color:color-mix(in oklab, var(--ring) 50%, transparent)}}.ring-offset-background{--tw-ring-offset-color:var(--background)}.outline-hidden{--tw-outline-style:none;outline-style:none}@media (forced-colors:active){.outline-hidden{outline-offset:2px;outline:2px solid #0000}}.outline{outline-style:var(--tw-outline-style);outline-width:1px}.blur{--tw-blur:blur(8px);filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.transition-\[color\,box-shadow\]{transition-property:color,box-shadow;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-colors{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-opacity{transition-property:opacity;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-shadow{transition-property:box-shadow;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-transform{transition-property:transform,translate,scale,rotate;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-none{transition-property:none}.duration-200{--tw-duration:.2s;transition-duration:.2s}.outline-none{--tw-outline-style:none;outline-style:none}.select-none{-webkit-user-select:none;user-select:none}.group-data-\[disabled\=true\]\:pointer-events-none:is(:where(.group)[data-disabled=true] *){pointer-events:none}.group-data-\[disabled\=true\]\:opacity-50:is(:where(.group)[data-disabled=true] *){opacity:.5}.group-data-\[orientation\=horizontal\]\/tabs\:h-9:is(:where(.group\/tabs)[data-orientation=horizontal] *){height:calc(var(--spacing) * 9)}.group-data-\[orientation\=vertical\]\/tabs\:h-fit:is(:where(.group\/tabs)[data-orientation=vertical] *){height:fit-content}.group-data-\[orientation\=vertical\]\/tabs\:w-full:is(:where(.group\/tabs)[data-orientation=vertical] *){width:100%}.group-data-\[orientation\=vertical\]\/tabs\:flex-col:is(:where(.group\/tabs)[data-orientation=vertical] *){flex-direction:column}.group-data-\[orientation\=vertical\]\/tabs\:justify-start:is(:where(.group\/tabs)[data-orientation=vertical] *){justify-content:flex-start}.group-data-\[size\=default\]\/switch\:size-4:is(:where(.group\/switch)[data-size=default] *){width:calc(var(--spacing) * 4);height:calc(var(--spacing) * 4)}.group-data-\[size\=sm\]\/switch\:size-3:is(:where(.group\/switch)[data-size=sm] *){width:calc(var(--spacing) * 3);height:calc(var(--spacing) * 3)}.group-data-\[variant\=line\]\/tabs-list\:bg-transparent:is(:where(.group\/tabs-list)[data-variant=line] *){background-color:#0000}.group-data-\[vaul-drawer-direction\=bottom\]\/drawer-content\:block:is(:where(.group\/drawer-content)[data-vaul-drawer-direction=bottom] *){display:block}.group-data-\[vaul-drawer-direction\=bottom\]\/drawer-content\:text-center:is(:where(.group\/drawer-content)[data-vaul-drawer-direction=bottom] *),.group-data-\[vaul-drawer-direction\=top\]\/drawer-content\:text-center:is(:where(.group\/drawer-content)[data-vaul-drawer-direction=top] *){text-align:center}.peer-disabled\:cursor-not-allowed:is(:where(.peer):disabled~*){cursor:not-allowed}.peer-disabled\:opacity-50:is(:where(.peer):disabled~*){opacity:.5}.selection\:bg-primary ::selection{background-color:var(--primary)}.selection\:bg-primary::selection{background-color:var(--primary)}.selection\:text-primary-foreground ::selection{color:var(--primary-foreground)}.selection\:text-primary-foreground::selection{color:var(--primary-foreground)}.file\:inline-flex::file-selector-button{display:inline-flex}.file\:h-7::file-selector-button{height:calc(var(--spacing) * 7)}.file\:border-0::file-selector-button{border-style:var(--tw-border-style);border-width:0}.file\:bg-transparent::file-selector-button{background-color:#0000}.file\:text-sm::file-selector-button{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.file\:font-medium::file-selector-button{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.file\:text-foreground::file-selector-button{color:var(--foreground)}.placeholder\:text-muted-foreground::placeholder{color:var(--muted-foreground)}.after\:absolute:after{content:var(--tw-content);position:absolute}.after\:bg-foreground:after{content:var(--tw-content);background-color:var(--foreground)}.after\:opacity-0:after{content:var(--tw-content);opacity:0}.after\:transition-opacity:after{content:var(--tw-content);transition-property:opacity;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.group-data-\[orientation\=horizontal\]\/tabs\:after\:inset-x-0:is(:where(.group\/tabs)[data-orientation=horizontal] *):after{content:var(--tw-content);inset-inline:calc(var(--spacing) * 0)}.group-data-\[orientation\=horizontal\]\/tabs\:after\:bottom-\[-5px\]:is(:where(.group\/tabs)[data-orientation=horizontal] *):after{content:var(--tw-content);bottom:-5px}.group-data-\[orientation\=horizontal\]\/tabs\:after\:h-0\.5:is(:where(.group\/tabs)[data-orientation=horizontal] *):after{content:var(--tw-content);height:calc(var(--spacing) * .5)}.group-data-\[orientation\=vertical\]\/tabs\:after\:inset-y-0:is(:where(.group\/tabs)[data-orientation=vertical] *):after{content:var(--tw-content);inset-block:calc(var(--spacing) * 0)}.group-data-\[orientation\=vertical\]\/tabs\:after\:-right-1:is(:where(.group\/tabs)[data-orientation=vertical] *):after{content:var(--tw-content);right:calc(var(--spacing) * -1)}.group-data-\[orientation\=vertical\]\/tabs\:after\:w-0\.5:is(:where(.group\/tabs)[data-orientation=vertical] *):after{content:var(--tw-content);width:calc(var(--spacing) * .5)}.last\:border-b-0:last-child{border-bottom-style:var(--tw-border-style);border-bottom-width:0}@media (hover:hover){.hover\:bg-accent:hover{background-color:var(--accent)}.hover\:bg-destructive\/90:hover{background-color:var(--destructive)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-destructive\/90:hover{background-color:color-mix(in oklab,var(--destructive) 90%,transparent)}}.hover\:bg-muted:hover,.hover\:bg-muted\/50:hover{background-color:var(--muted)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-muted\/50:hover{background-color:color-mix(in oklab,var(--muted) 50%,transparent)}}.hover\:bg-primary\/90:hover{background-color:var(--primary)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-primary\/90:hover{background-color:color-mix(in oklab,var(--primary) 90%,transparent)}}.hover\:bg-secondary\/80:hover{background-color:var(--secondary)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-secondary\/80:hover{background-color:color-mix(in oklab,var(--secondary) 80%,transparent)}}.hover\:text-accent-foreground:hover{color:var(--accent-foreground)}.hover\:text-foreground:hover{color:var(--foreground)}.hover\:text-muted-foreground:hover{color:var(--muted-foreground)}.hover\:underline:hover{text-decoration-line:underline}.hover\:opacity-100:hover{opacity:1}.hover\:ring-4:hover{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(4px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}}.focus\:z-10:focus{z-index:10}.focus\:bg-accent:focus{background-color:var(--accent)}.focus\:text-accent-foreground:focus{color:var(--accent-foreground)}.focus\:ring-2:focus{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.focus\:ring-ring:focus{--tw-ring-color:var(--ring)}.focus\:ring-offset-2:focus{--tw-ring-offset-width:2px;--tw-ring-offset-shadow:var(--tw-ring-inset,) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color)}.focus\:outline-hidden:focus{--tw-outline-style:none;outline-style:none}@media (forced-colors:active){.focus\:outline-hidden:focus{outline-offset:2px;outline:2px solid #0000}}.focus-visible\:z-10:focus-visible{z-index:10}.focus-visible\:border-ring:focus-visible{border-color:var(--ring)}.focus-visible\:ring-4:focus-visible{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(4px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.focus-visible\:ring-\[3px\]:focus-visible{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(3px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.focus-visible\:ring-destructive\/20:focus-visible{--tw-ring-color:var(--destructive)}@supports (color:color-mix(in lab,red,red)){.focus-visible\:ring-destructive\/20:focus-visible{--tw-ring-color:color-mix(in oklab, var(--destructive) 20%, transparent)}}.focus-visible\:ring-ring\/50:focus-visible{--tw-ring-color:var(--ring)}@supports (color:color-mix(in lab,red,red)){.focus-visible\:ring-ring\/50:focus-visible{--tw-ring-color:color-mix(in oklab, var(--ring) 50%, transparent)}}.focus-visible\:outline-hidden:focus-visible{--tw-outline-style:none;outline-style:none}@media (forced-colors:active){.focus-visible\:outline-hidden:focus-visible{outline-offset:2px;outline:2px solid #0000}}.focus-visible\:outline-1:focus-visible{outline-style:var(--tw-outline-style);outline-width:1px}.focus-visible\:outline-ring:focus-visible{outline-color:var(--ring)}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:cursor-not-allowed:disabled{cursor:not-allowed}.disabled\:opacity-50:disabled{opacity:.5}.has-data-\[slot\=card-action\]\:grid-cols-\[1fr_auto\]:has([data-slot=card-action]){grid-template-columns:1fr auto}.has-\[\>svg\]\:grid-cols-\[calc\(var\(--spacing\)\*4\)_1fr\]:has(>svg){grid-template-columns:calc(var(--spacing) * 4) 1fr}.has-\[\>svg\]\:gap-x-3:has(>svg){column-gap:calc(var(--spacing) * 3)}.has-\[\>svg\]\:px-1\.5:has(>svg){padding-inline:calc(var(--spacing) * 1.5)}.has-\[\>svg\]\:px-2\.5:has(>svg){padding-inline:calc(var(--spacing) * 2.5)}.has-\[\>svg\]\:px-3:has(>svg){padding-inline:calc(var(--spacing) * 3)}.has-\[\>svg\]\:px-4:has(>svg){padding-inline:calc(var(--spacing) * 4)}.aria-invalid\:border-destructive[aria-invalid=true]{border-color:var(--destructive)}.aria-invalid\:ring-destructive\/20[aria-invalid=true]{--tw-ring-color:var(--destructive)}@supports (color:color-mix(in lab,red,red)){.aria-invalid\:ring-destructive\/20[aria-invalid=true]{--tw-ring-color:color-mix(in oklab, var(--destructive) 20%, transparent)}}.data-\[disabled\]\:pointer-events-none[data-disabled]{pointer-events:none}.data-\[disabled\]\:opacity-50[data-disabled]{opacity:.5}.data-\[inset\]\:pl-8[data-inset]{padding-left:calc(var(--spacing) * 8)}.data-\[orientation\=horizontal\]\:h-1\.5[data-orientation=horizontal]{height:calc(var(--spacing) * 1.5)}.data-\[orientation\=horizontal\]\:h-full[data-orientation=horizontal]{height:100%}.data-\[orientation\=horizontal\]\:h-px[data-orientation=horizontal]{height:1px}.data-\[orientation\=horizontal\]\:w-full[data-orientation=horizontal]{width:100%}.data-\[orientation\=horizontal\]\:flex-col[data-orientation=horizontal]{flex-direction:column}.data-\[orientation\=vertical\]\:h-full[data-orientation=vertical]{height:100%}.data-\[orientation\=vertical\]\:min-h-44[data-orientation=vertical]{min-height:calc(var(--spacing) * 44)}.data-\[orientation\=vertical\]\:w-1\.5[data-orientation=vertical]{width:calc(var(--spacing) * 1.5)}.data-\[orientation\=vertical\]\:w-auto[data-orientation=vertical]{width:auto}.data-\[orientation\=vertical\]\:w-full[data-orientation=vertical]{width:100%}.data-\[orientation\=vertical\]\:w-px[data-orientation=vertical]{width:1px}.data-\[orientation\=vertical\]\:flex-col[data-orientation=vertical]{flex-direction:column}.data-\[placeholder\]\:text-muted-foreground[data-placeholder]{color:var(--muted-foreground)}.data-\[side\=bottom\]\:translate-y-1[data-side=bottom]{--tw-translate-y:calc(var(--spacing) * 1);translate:var(--tw-translate-x) var(--tw-translate-y)}.data-\[side\=left\]\:-translate-x-1[data-side=left]{--tw-translate-x:calc(var(--spacing) * -1);translate:var(--tw-translate-x) var(--tw-translate-y)}.data-\[side\=right\]\:translate-x-1[data-side=right]{--tw-translate-x:calc(var(--spacing) * 1);translate:var(--tw-translate-x) var(--tw-translate-y)}.data-\[side\=top\]\:-translate-y-1[data-side=top]{--tw-translate-y:calc(var(--spacing) * -1);translate:var(--tw-translate-x) var(--tw-translate-y)}.data-\[size\=default\]\:h-9[data-size=default]{height:calc(var(--spacing) * 9)}.data-\[size\=default\]\:h-\[1\.15rem\][data-size=default]{height:1.15rem}.data-\[size\=default\]\:w-8[data-size=default]{width:calc(var(--spacing) * 8)}.data-\[size\=sm\]\:h-3\.5[data-size=sm]{height:calc(var(--spacing) * 3.5)}.data-\[size\=sm\]\:h-8[data-size=sm]{height:calc(var(--spacing) * 8)}.data-\[size\=sm\]\:w-6[data-size=sm]{width:calc(var(--spacing) * 6)}:is(.\*\:data-\[slot\=alert-description\]\:text-destructive\/90>*)[data-slot=alert-description]{color:var(--destructive)}@supports (color:color-mix(in lab,red,red)){:is(.\*\:data-\[slot\=alert-description\]\:text-destructive\/90>*)[data-slot=alert-description]{color:color-mix(in oklab,var(--destructive) 90%,transparent)}}:is(.\*\:data-\[slot\=select-value\]\:line-clamp-1>*)[data-slot=select-value]{-webkit-line-clamp:1;-webkit-box-orient:vertical;display:-webkit-box;overflow:hidden}:is(.\*\:data-\[slot\=select-value\]\:flex>*)[data-slot=select-value]{display:flex}:is(.\*\:data-\[slot\=select-value\]\:items-center>*)[data-slot=select-value]{align-items:center}:is(.\*\:data-\[slot\=select-value\]\:gap-2>*)[data-slot=select-value]{gap:calc(var(--spacing) * 2)}.data-\[spacing\=0\]\:rounded-none[data-spacing="0"]{border-radius:0}.data-\[spacing\=0\]\:shadow-none[data-spacing="0"]{--tw-shadow:0 0 #0000;box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.data-\[spacing\=0\]\:first\:rounded-l-md[data-spacing="0"]:first-child{border-top-left-radius:calc(var(--radius) - 2px);border-bottom-left-radius:calc(var(--radius) - 2px)}.data-\[spacing\=0\]\:last\:rounded-r-md[data-spacing="0"]:last-child{border-top-right-radius:calc(var(--radius) - 2px);border-bottom-right-radius:calc(var(--radius) - 2px)}.data-\[state\=active\]\:bg-background[data-state=active]{background-color:var(--background)}.data-\[state\=active\]\:text-foreground[data-state=active]{color:var(--foreground)}.group-data-\[variant\=default\]\/tabs-list\:data-\[state\=active\]\:shadow-sm:is(:where(.group\/tabs-list)[data-variant=default] *)[data-state=active]{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a), 0 1px 2px -1px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.group-data-\[variant\=line\]\/tabs-list\:data-\[state\=active\]\:bg-transparent:is(:where(.group\/tabs-list)[data-variant=line] *)[data-state=active]{background-color:#0000}.group-data-\[variant\=line\]\/tabs-list\:data-\[state\=active\]\:shadow-none:is(:where(.group\/tabs-list)[data-variant=line] *)[data-state=active]{--tw-shadow:0 0 #0000;box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.group-data-\[variant\=line\]\/tabs-list\:data-\[state\=active\]\:after\:opacity-100:is(:where(.group\/tabs-list)[data-variant=line] *)[data-state=active]:after{content:var(--tw-content);opacity:1}.data-\[state\=checked\]\:translate-x-\[calc\(100\%-2px\)\][data-state=checked]{--tw-translate-x: calc(100% - 2px) ;translate:var(--tw-translate-x) var(--tw-translate-y)}.data-\[state\=checked\]\:border-primary[data-state=checked]{border-color:var(--primary)}.data-\[state\=checked\]\:bg-primary[data-state=checked]{background-color:var(--primary)}.data-\[state\=checked\]\:text-primary-foreground[data-state=checked]{color:var(--primary-foreground)}.data-\[state\=on\]\:bg-accent[data-state=on]{background-color:var(--accent)}.data-\[state\=on\]\:text-accent-foreground[data-state=on]{color:var(--accent-foreground)}.data-\[state\=open\]\:bg-accent[data-state=open]{background-color:var(--accent)}.data-\[state\=open\]\:text-accent-foreground[data-state=open]{color:var(--accent-foreground)}.data-\[state\=open\]\:text-muted-foreground[data-state=open]{color:var(--muted-foreground)}.data-\[state\=selected\]\:bg-muted[data-state=selected]{background-color:var(--muted)}.data-\[state\=unchecked\]\:translate-x-0[data-state=unchecked]{--tw-translate-x:calc(var(--spacing) * 0);translate:var(--tw-translate-x) var(--tw-translate-y)}.data-\[state\=unchecked\]\:bg-input[data-state=unchecked]{background-color:var(--input)}.data-\[variant\=destructive\]\:text-destructive[data-variant=destructive]{color:var(--destructive)}.data-\[variant\=destructive\]\:focus\:bg-destructive\/10[data-variant=destructive]:focus{background-color:var(--destructive)}@supports (color:color-mix(in lab,red,red)){.data-\[variant\=destructive\]\:focus\:bg-destructive\/10[data-variant=destructive]:focus{background-color:color-mix(in oklab,var(--destructive) 10%,transparent)}}.data-\[variant\=destructive\]\:focus\:text-destructive[data-variant=destructive]:focus{color:var(--destructive)}.data-\[variant\=line\]\:rounded-none[data-variant=line]{border-radius:0}.data-\[spacing\=0\]\:data-\[variant\=outline\]\:border-l-0[data-spacing="0"][data-variant=outline]{border-left-style:var(--tw-border-style);border-left-width:0}.data-\[spacing\=0\]\:data-\[variant\=outline\]\:first\:border-l[data-spacing="0"][data-variant=outline]:first-child{border-left-style:var(--tw-border-style);border-left-width:1px}.data-\[spacing\=default\]\:data-\[variant\=outline\]\:shadow-xs[data-spacing=default][data-variant=outline]{--tw-shadow:0 1px 2px 0 var(--tw-shadow-color,#0000000d);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.data-\[vaul-drawer-direction\=bottom\]\:inset-x-0[data-vaul-drawer-direction=bottom]{inset-inline:calc(var(--spacing) * 0)}.data-\[vaul-drawer-direction\=bottom\]\:bottom-0[data-vaul-drawer-direction=bottom]{bottom:calc(var(--spacing) * 0)}.data-\[vaul-drawer-direction\=bottom\]\:mt-24[data-vaul-drawer-direction=bottom]{margin-top:calc(var(--spacing) * 24)}.data-\[vaul-drawer-direction\=bottom\]\:max-h-\[80vh\][data-vaul-drawer-direction=bottom]{max-height:80vh}.data-\[vaul-drawer-direction\=bottom\]\:rounded-t-lg[data-vaul-drawer-direction=bottom]{border-top-left-radius:var(--radius);border-top-right-radius:var(--radius)}.data-\[vaul-drawer-direction\=bottom\]\:border-t[data-vaul-drawer-direction=bottom]{border-top-style:var(--tw-border-style);border-top-width:1px}.data-\[vaul-drawer-direction\=left\]\:inset-y-0[data-vaul-drawer-direction=left]{inset-block:calc(var(--spacing) * 0)}.data-\[vaul-drawer-direction\=left\]\:left-0[data-vaul-drawer-direction=left]{left:calc(var(--spacing) * 0)}.data-\[vaul-drawer-direction\=left\]\:w-3\/4[data-vaul-drawer-direction=left]{width:75%}.data-\[vaul-drawer-direction\=left\]\:border-r[data-vaul-drawer-direction=left]{border-right-style:var(--tw-border-style);border-right-width:1px}.data-\[vaul-drawer-direction\=right\]\:inset-y-0[data-vaul-drawer-direction=right]{inset-block:calc(var(--spacing) * 0)}.data-\[vaul-drawer-direction\=right\]\:right-0[data-vaul-drawer-direction=right]{right:calc(var(--spacing) * 0)}.data-\[vaul-drawer-direction\=right\]\:w-3\/4[data-vaul-drawer-direction=right]{width:75%}.data-\[vaul-drawer-direction\=right\]\:border-l[data-vaul-drawer-direction=right]{border-left-style:var(--tw-border-style);border-left-width:1px}.data-\[vaul-drawer-direction\=top\]\:inset-x-0[data-vaul-drawer-direction=top]{inset-inline:calc(var(--spacing) * 0)}.data-\[vaul-drawer-direction\=top\]\:top-0[data-vaul-drawer-direction=top]{top:calc(var(--spacing) * 0)}.data-\[vaul-drawer-direction\=top\]\:mb-24[data-vaul-drawer-direction=top]{margin-bottom:calc(var(--spacing) * 24)}.data-\[vaul-drawer-direction\=top\]\:max-h-\[80vh\][data-vaul-drawer-direction=top]{max-height:80vh}.data-\[vaul-drawer-direction\=top\]\:rounded-b-lg[data-vaul-drawer-direction=top]{border-bottom-right-radius:var(--radius);border-bottom-left-radius:var(--radius)}.data-\[vaul-drawer-direction\=top\]\:border-b[data-vaul-drawer-direction=top]{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}@media (min-width:40rem){.sm\:block{display:block}.sm\:max-w-lg{max-width:var(--container-lg)}.sm\:min-w-\[280px\]{min-width:280px}.sm\:min-w-\[320px\]{min-width:320px}.sm\:min-w-\[360px\]{min-width:360px}.sm\:flex-row{flex-direction:row}.sm\:justify-end{justify-content:flex-end}.sm\:pr-2\.5{padding-right:calc(var(--spacing) * 2.5)}.sm\:pl-2\.5{padding-left:calc(var(--spacing) * 2.5)}.sm\:text-left{text-align:left}.data-\[vaul-drawer-direction\=left\]\:sm\:max-w-sm[data-vaul-drawer-direction=left],.data-\[vaul-drawer-direction\=right\]\:sm\:max-w-sm[data-vaul-drawer-direction=right]{max-width:var(--container-sm)}}@media (min-width:48rem){.md\:basis-1\/2{flex-basis:50%}.md\:gap-1\.5{gap:calc(var(--spacing) * 1.5)}.md\:text-left{text-align:left}.md\:text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}}@media (min-width:64rem){.lg\:basis-1\/3{flex-basis:33.3333%}}@media (prefers-color-scheme:dark){.dark\:border-blue-800{border-color:var(--color-blue-800)}.dark\:border-green-800{border-color:var(--color-green-800)}.dark\:border-input{border-color:var(--input)}.dark\:border-yellow-800{border-color:var(--color-yellow-800)}.dark\:bg-blue-950{background-color:var(--color-blue-950)}.dark\:bg-destructive\/60{background-color:var(--destructive)}@supports (color:color-mix(in lab,red,red)){.dark\:bg-destructive\/60{background-color:color-mix(in oklab,var(--destructive) 60%,transparent)}}.dark\:bg-green-950{background-color:var(--color-green-950)}.dark\:bg-input\/30{background-color:var(--input)}@supports (color:color-mix(in lab,red,red)){.dark\:bg-input\/30{background-color:color-mix(in oklab,var(--input) 30%,transparent)}}.dark\:bg-yellow-950{background-color:var(--color-yellow-950)}.dark\:text-blue-100{color:var(--color-blue-100)}.dark\:text-green-100{color:var(--color-green-100)}.dark\:text-muted-foreground{color:var(--muted-foreground)}.dark\:text-yellow-100{color:var(--color-yellow-100)}@media (hover:hover){.dark\:hover\:bg-accent\/50:hover{background-color:var(--accent)}@supports (color:color-mix(in lab,red,red)){.dark\:hover\:bg-accent\/50:hover{background-color:color-mix(in oklab,var(--accent) 50%,transparent)}}.dark\:hover\:bg-input\/50:hover{background-color:var(--input)}@supports (color:color-mix(in lab,red,red)){.dark\:hover\:bg-input\/50:hover{background-color:color-mix(in oklab,var(--input) 50%,transparent)}}.dark\:hover\:text-foreground:hover{color:var(--foreground)}}.dark\:focus-visible\:ring-destructive\/40:focus-visible{--tw-ring-color:var(--destructive)}@supports (color:color-mix(in lab,red,red)){.dark\:focus-visible\:ring-destructive\/40:focus-visible{--tw-ring-color:color-mix(in oklab, var(--destructive) 40%, transparent)}}.dark\:aria-invalid\:ring-destructive\/40[aria-invalid=true]{--tw-ring-color:var(--destructive)}@supports (color:color-mix(in lab,red,red)){.dark\:aria-invalid\:ring-destructive\/40[aria-invalid=true]{--tw-ring-color:color-mix(in oklab, var(--destructive) 40%, transparent)}}.dark\:data-\[state\=active\]\:border-input[data-state=active]{border-color:var(--input)}.dark\:data-\[state\=active\]\:bg-input\/30[data-state=active]{background-color:var(--input)}@supports (color:color-mix(in lab,red,red)){.dark\:data-\[state\=active\]\:bg-input\/30[data-state=active]{background-color:color-mix(in oklab,var(--input) 30%,transparent)}}.dark\:data-\[state\=active\]\:text-foreground[data-state=active]{color:var(--foreground)}.dark\:group-data-\[variant\=line\]\/tabs-list\:data-\[state\=active\]\:border-transparent:is(:where(.group\/tabs-list)[data-variant=line] *)[data-state=active]{border-color:#0000}.dark\:group-data-\[variant\=line\]\/tabs-list\:data-\[state\=active\]\:bg-transparent:is(:where(.group\/tabs-list)[data-variant=line] *)[data-state=active]{background-color:#0000}.dark\:data-\[state\=checked\]\:bg-primary[data-state=checked]{background-color:var(--primary)}.dark\:data-\[state\=checked\]\:bg-primary-foreground[data-state=checked]{background-color:var(--primary-foreground)}.dark\:data-\[state\=unchecked\]\:bg-foreground[data-state=unchecked]{background-color:var(--foreground)}.dark\:data-\[state\=unchecked\]\:bg-input\/80[data-state=unchecked]{background-color:var(--input)}@supports (color:color-mix(in lab,red,red)){.dark\:data-\[state\=unchecked\]\:bg-input\/80[data-state=unchecked]{background-color:color-mix(in oklab,var(--input) 80%,transparent)}}.dark\:data-\[variant\=destructive\]\:focus\:bg-destructive\/20[data-variant=destructive]:focus{background-color:var(--destructive)}@supports (color:color-mix(in lab,red,red)){.dark\:data-\[variant\=destructive\]\:focus\:bg-destructive\/20[data-variant=destructive]:focus{background-color:color-mix(in oklab,var(--destructive) 20%,transparent)}}}.\[\&_p\]\:leading-relaxed p{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.\[\&_svg\]\:pointer-events-none svg{pointer-events:none}.\[\&_svg\]\:shrink-0 svg{flex-shrink:0}.\[\&_svg\:not\(\[class\*\=\'size-\'\]\)\]\:size-3 svg:not([class*=size-]){width:calc(var(--spacing) * 3);height:calc(var(--spacing) * 3)}.\[\&_svg\:not\(\[class\*\=\'size-\'\]\)\]\:size-4 svg:not([class*=size-]){width:calc(var(--spacing) * 4);height:calc(var(--spacing) * 4)}.\[\&_svg\:not\(\[class\*\=\'text-\'\]\)\]\:text-muted-foreground svg:not([class*=text-]){color:var(--muted-foreground)}.\[\&_tr\]\:border-b tr{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.\[\&_tr\:last-child\]\:border-0 tr:last-child{border-style:var(--tw-border-style);border-width:0}.\[\&\:has\(\[role\=checkbox\]\)\]\:pr-0:has([role=checkbox]){padding-right:calc(var(--spacing) * 0)}.\[\.border-b\]\:pb-6.border-b{padding-bottom:calc(var(--spacing) * 6)}.\[\.border-t\]\:pt-6.border-t{padding-top:calc(var(--spacing) * 6)}:is(.\*\:\[span\]\:last\:flex>*):is(span):last-child{display:flex}:is(.\*\:\[span\]\:last\:items-center>*):is(span):last-child{align-items:center}:is(.\*\:\[span\]\:last\:gap-2>*):is(span):last-child{gap:calc(var(--spacing) * 2)}:is(.data-\[variant\=destructive\]\:\*\:\[svg\]\:\!text-destructive[data-variant=destructive]>*):is(svg){color:var(--destructive)!important}.\[\&\>\[role\=checkbox\]\]\:translate-y-\[2px\]>[role=checkbox]{--tw-translate-y:2px;translate:var(--tw-translate-x) var(--tw-translate-y)}.\[\&\>svg\]\:pointer-events-none>svg{pointer-events:none}.\[\&\>svg\]\:size-3>svg{width:calc(var(--spacing) * 3);height:calc(var(--spacing) * 3)}.\[\&\>svg\]\:size-4>svg{width:calc(var(--spacing) * 4);height:calc(var(--spacing) * 4)}.\[\&\>svg\]\:translate-y-0\.5>svg{--tw-translate-y:calc(var(--spacing) * .5);translate:var(--tw-translate-x) var(--tw-translate-y)}.\[\&\>svg\]\:text-current>svg{color:currentColor}.\[\&\>tr\]\:last\:border-b-0>tr:last-child{border-bottom-style:var(--tw-border-style);border-bottom-width:0}.\[\&\[data-state\=open\]\>svg\]\:rotate-180[data-state=open]>svg{rotate:180deg}@media (hover:hover){a.\[a\&\]\:hover\:bg-accent:hover{background-color:var(--accent)}a.\[a\&\]\:hover\:bg-destructive\/90:hover{background-color:var(--destructive)}@supports (color:color-mix(in lab,red,red)){a.\[a\&\]\:hover\:bg-destructive\/90:hover{background-color:color-mix(in oklab,var(--destructive) 90%,transparent)}}a.\[a\&\]\:hover\:bg-primary\/90:hover{background-color:var(--primary)}@supports (color:color-mix(in lab,red,red)){a.\[a\&\]\:hover\:bg-primary\/90:hover{background-color:color-mix(in oklab,var(--primary) 90%,transparent)}}a.\[a\&\]\:hover\:bg-secondary\/90:hover{background-color:var(--secondary)}@supports (color:color-mix(in lab,red,red)){a.\[a\&\]\:hover\:bg-secondary\/90:hover{background-color:color-mix(in oklab,var(--secondary) 90%,transparent)}}a.\[a\&\]\:hover\:text-accent-foreground:hover{color:var(--accent-foreground)}}}:root{--background:#fff;--foreground:#10213f;--card:#fff;--card-foreground:#152b53;--primary:#2454d6;--primary-foreground:#f8fbff;--secondary:#e5efff;--secondary-foreground:#1f3f78;--muted:#f5f9ff;--muted-foreground:#4a6291;--accent:#edf4ff;--accent-foreground:#18305c;--destructive:#b8324f;--border:#c8dafc;--input:#d9e4fb;--ring:#91aff3;--radius:.75rem;font-synthesis:none;text-rendering:optimizelegibility;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;color:#10213f;background:#edf3ff;font-family:Segoe UI,Noto Sans KR,system-ui,-apple-system,sans-serif;font-weight:400;line-height:1.4}*,:before,:after{box-sizing:border-box}html,body{overscroll-behavior:none;background:#edf3ff;width:100%;min-width:320px;height:100%;margin:0;overflow:hidden}#root{width:100%;height:100%;overflow:hidden}a{color:inherit}@property --tw-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-y{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-z{syntax:"*";inherits:false;initial-value:0}@property --tw-rotate-x{syntax:"*";inherits:false}@property --tw-rotate-y{syntax:"*";inherits:false}@property --tw-rotate-z{syntax:"*";inherits:false}@property --tw-skew-x{syntax:"*";inherits:false}@property --tw-skew-y{syntax:"*";inherits:false}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-space-x-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"<length>";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-outline-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-blur{syntax:"*";inherits:false}@property --tw-brightness{syntax:"*";inherits:false}@property --tw-contrast{syntax:"*";inherits:false}@property --tw-grayscale{syntax:"*";inherits:false}@property --tw-hue-rotate{syntax:"*";inherits:false}@property --tw-invert{syntax:"*";inherits:false}@property --tw-opacity{syntax:"*";inherits:false}@property --tw-saturate{syntax:"*";inherits:false}@property --tw-sepia{syntax:"*";inherits:false}@property --tw-drop-shadow{syntax:"*";inherits:false}@property --tw-drop-shadow-color{syntax:"*";inherits:false}@property --tw-drop-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-drop-shadow-size{syntax:"*";inherits:false}@property --tw-duration{syntax:"*";inherits:false}@property --tw-content{syntax:"*";inherits:false;initial-value:""}@keyframes spin{to{transform:rotate(360deg)}}@keyframes pulse{50%{opacity:.5}}
|