sravya commited on
Commit
ac717a9
·
0 Parent(s):

Deploy AI Interview Coach to HF Spaces

Browse files
.gitattributes ADDED
@@ -0,0 +1,35 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ *.7z filter=lfs diff=lfs merge=lfs -text
2
+ *.arrow filter=lfs diff=lfs merge=lfs -text
3
+ *.bin filter=lfs diff=lfs merge=lfs -text
4
+ *.bz2 filter=lfs diff=lfs merge=lfs -text
5
+ *.ckpt filter=lfs diff=lfs merge=lfs -text
6
+ *.ftz filter=lfs diff=lfs merge=lfs -text
7
+ *.gz filter=lfs diff=lfs merge=lfs -text
8
+ *.h5 filter=lfs diff=lfs merge=lfs -text
9
+ *.joblib filter=lfs diff=lfs merge=lfs -text
10
+ *.lfs.* filter=lfs diff=lfs merge=lfs -text
11
+ *.mlmodel filter=lfs diff=lfs merge=lfs -text
12
+ *.model filter=lfs diff=lfs merge=lfs -text
13
+ *.msgpack filter=lfs diff=lfs merge=lfs -text
14
+ *.npy filter=lfs diff=lfs merge=lfs -text
15
+ *.npz filter=lfs diff=lfs merge=lfs -text
16
+ *.onnx filter=lfs diff=lfs merge=lfs -text
17
+ *.ot filter=lfs diff=lfs merge=lfs -text
18
+ *.parquet filter=lfs diff=lfs merge=lfs -text
19
+ *.pb filter=lfs diff=lfs merge=lfs -text
20
+ *.pickle filter=lfs diff=lfs merge=lfs -text
21
+ *.pkl filter=lfs diff=lfs merge=lfs -text
22
+ *.pt filter=lfs diff=lfs merge=lfs -text
23
+ *.pth filter=lfs diff=lfs merge=lfs -text
24
+ *.rar filter=lfs diff=lfs merge=lfs -text
25
+ *.safetensors filter=lfs diff=lfs merge=lfs -text
26
+ saved_model/**/* filter=lfs diff=lfs merge=lfs -text
27
+ *.tar.* filter=lfs diff=lfs merge=lfs -text
28
+ *.tar filter=lfs diff=lfs merge=lfs -text
29
+ *.tflite filter=lfs diff=lfs merge=lfs -text
30
+ *.tgz filter=lfs diff=lfs merge=lfs -text
31
+ *.wasm filter=lfs diff=lfs merge=lfs -text
32
+ *.xz filter=lfs diff=lfs merge=lfs -text
33
+ *.zip filter=lfs diff=lfs merge=lfs -text
34
+ *.zst filter=lfs diff=lfs merge=lfs -text
35
+ *tfevents* filter=lfs diff=lfs merge=lfs -text
.gitignore ADDED
@@ -0,0 +1,4 @@
 
 
 
 
 
1
+ node_modules/
2
+ .env
3
+ dist/
4
+ .DS_Store
Dockerfile ADDED
@@ -0,0 +1,25 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ FROM node:20-slim
2
+
3
+ WORKDIR /app
4
+
5
+ # Build frontend
6
+ COPY frontend/package*.json frontend/
7
+ RUN cd frontend && npm install
8
+
9
+ COPY frontend/ frontend/
10
+ RUN cd frontend && npm run build
11
+
12
+ # Setup backend
13
+ COPY backend/package*.json backend/
14
+ RUN cd backend && npm install --production
15
+
16
+ COPY backend/ backend/
17
+
18
+ # Copy built frontend to backend/public
19
+ RUN cp -r frontend/dist backend/public
20
+
21
+ ENV PORT=7860
22
+ EXPOSE 7860
23
+
24
+ WORKDIR /app/backend
25
+ CMD ["node", "server.js"]
README.md ADDED
@@ -0,0 +1,11 @@
 
 
 
 
 
 
 
 
 
 
 
 
1
+ ---
2
+ title: AI Interview Coach
3
+ emoji: "\U0001F3A4"
4
+ colorFrom: blue
5
+ colorTo: purple
6
+ sdk: docker
7
+ pinned: false
8
+ ---
9
+
10
+ # AI Interview Coach
11
+ Practice job interviews with AI. Get real-time feedback and a final assessment report.
backend/.env.example ADDED
@@ -0,0 +1,13 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # AI Provider - Choose one (or both for fallback)
2
+ # Option 1: Groq (FREE cloud API - recommended for deployment)
3
+ # Get your free key at https://console.groq.com/keys
4
+ GROQ_API_KEY=your_groq_api_key_here
5
+ GROQ_MODEL=llama-3.3-70b-versatile
6
+
7
+ # Option 2: Ollama (FREE local AI - great for development)
8
+ # Install from https://ollama.com then run: ollama pull llama3.2:3b
9
+ OLLAMA_URL=http://localhost:11434
10
+ OLLAMA_MODEL=llama3.2:3b
11
+
12
+ # Server
13
+ PORT=3002
backend/package-lock.json ADDED
@@ -0,0 +1,891 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "interview-coach-backend",
3
+ "version": "1.0.0",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "interview-coach-backend",
9
+ "version": "1.0.0",
10
+ "dependencies": {
11
+ "@google/generative-ai": "^0.21.0",
12
+ "cors": "^2.8.5",
13
+ "dotenv": "^16.4.7",
14
+ "express": "^4.21.2",
15
+ "uuid": "^11.0.5"
16
+ }
17
+ },
18
+ "node_modules/@google/generative-ai": {
19
+ "version": "0.21.0",
20
+ "resolved": "https://registry.npmjs.org/@google/generative-ai/-/generative-ai-0.21.0.tgz",
21
+ "integrity": "sha512-7XhUbtnlkSEZK15kN3t+tzIMxsbKm/dSkKBFalj+20NvPKe1kBY7mR2P7vuijEn+f06z5+A8bVGKO0v39cr6Wg==",
22
+ "license": "Apache-2.0",
23
+ "engines": {
24
+ "node": ">=18.0.0"
25
+ }
26
+ },
27
+ "node_modules/accepts": {
28
+ "version": "1.3.8",
29
+ "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz",
30
+ "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==",
31
+ "license": "MIT",
32
+ "dependencies": {
33
+ "mime-types": "~2.1.34",
34
+ "negotiator": "0.6.3"
35
+ },
36
+ "engines": {
37
+ "node": ">= 0.6"
38
+ }
39
+ },
40
+ "node_modules/array-flatten": {
41
+ "version": "1.1.1",
42
+ "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
43
+ "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==",
44
+ "license": "MIT"
45
+ },
46
+ "node_modules/body-parser": {
47
+ "version": "1.20.4",
48
+ "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.4.tgz",
49
+ "integrity": "sha512-ZTgYYLMOXY9qKU/57FAo8F+HA2dGX7bqGc71txDRC1rS4frdFI5R7NhluHxH6M0YItAP0sHB4uqAOcYKxO6uGA==",
50
+ "license": "MIT",
51
+ "dependencies": {
52
+ "bytes": "~3.1.2",
53
+ "content-type": "~1.0.5",
54
+ "debug": "2.6.9",
55
+ "depd": "2.0.0",
56
+ "destroy": "~1.2.0",
57
+ "http-errors": "~2.0.1",
58
+ "iconv-lite": "~0.4.24",
59
+ "on-finished": "~2.4.1",
60
+ "qs": "~6.14.0",
61
+ "raw-body": "~2.5.3",
62
+ "type-is": "~1.6.18",
63
+ "unpipe": "~1.0.0"
64
+ },
65
+ "engines": {
66
+ "node": ">= 0.8",
67
+ "npm": "1.2.8000 || >= 1.4.16"
68
+ }
69
+ },
70
+ "node_modules/bytes": {
71
+ "version": "3.1.2",
72
+ "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz",
73
+ "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==",
74
+ "license": "MIT",
75
+ "engines": {
76
+ "node": ">= 0.8"
77
+ }
78
+ },
79
+ "node_modules/call-bind-apply-helpers": {
80
+ "version": "1.0.2",
81
+ "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz",
82
+ "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==",
83
+ "license": "MIT",
84
+ "dependencies": {
85
+ "es-errors": "^1.3.0",
86
+ "function-bind": "^1.1.2"
87
+ },
88
+ "engines": {
89
+ "node": ">= 0.4"
90
+ }
91
+ },
92
+ "node_modules/call-bound": {
93
+ "version": "1.0.4",
94
+ "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz",
95
+ "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==",
96
+ "license": "MIT",
97
+ "dependencies": {
98
+ "call-bind-apply-helpers": "^1.0.2",
99
+ "get-intrinsic": "^1.3.0"
100
+ },
101
+ "engines": {
102
+ "node": ">= 0.4"
103
+ },
104
+ "funding": {
105
+ "url": "https://github.com/sponsors/ljharb"
106
+ }
107
+ },
108
+ "node_modules/content-disposition": {
109
+ "version": "0.5.4",
110
+ "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz",
111
+ "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==",
112
+ "license": "MIT",
113
+ "dependencies": {
114
+ "safe-buffer": "5.2.1"
115
+ },
116
+ "engines": {
117
+ "node": ">= 0.6"
118
+ }
119
+ },
120
+ "node_modules/content-type": {
121
+ "version": "1.0.5",
122
+ "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz",
123
+ "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==",
124
+ "license": "MIT",
125
+ "engines": {
126
+ "node": ">= 0.6"
127
+ }
128
+ },
129
+ "node_modules/cookie": {
130
+ "version": "0.7.2",
131
+ "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.2.tgz",
132
+ "integrity": "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w==",
133
+ "license": "MIT",
134
+ "engines": {
135
+ "node": ">= 0.6"
136
+ }
137
+ },
138
+ "node_modules/cookie-signature": {
139
+ "version": "1.0.7",
140
+ "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.7.tgz",
141
+ "integrity": "sha512-NXdYc3dLr47pBkpUCHtKSwIOQXLVn8dZEuywboCOJY/osA0wFSLlSawr3KN8qXJEyX66FcONTH8EIlVuK0yyFA==",
142
+ "license": "MIT"
143
+ },
144
+ "node_modules/cors": {
145
+ "version": "2.8.6",
146
+ "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.6.tgz",
147
+ "integrity": "sha512-tJtZBBHA6vjIAaF6EnIaq6laBBP9aq/Y3ouVJjEfoHbRBcHBAHYcMh/w8LDrk2PvIMMq8gmopa5D4V8RmbrxGw==",
148
+ "license": "MIT",
149
+ "dependencies": {
150
+ "object-assign": "^4",
151
+ "vary": "^1"
152
+ },
153
+ "engines": {
154
+ "node": ">= 0.10"
155
+ },
156
+ "funding": {
157
+ "type": "opencollective",
158
+ "url": "https://opencollective.com/express"
159
+ }
160
+ },
161
+ "node_modules/debug": {
162
+ "version": "2.6.9",
163
+ "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
164
+ "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
165
+ "license": "MIT",
166
+ "dependencies": {
167
+ "ms": "2.0.0"
168
+ }
169
+ },
170
+ "node_modules/depd": {
171
+ "version": "2.0.0",
172
+ "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
173
+ "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==",
174
+ "license": "MIT",
175
+ "engines": {
176
+ "node": ">= 0.8"
177
+ }
178
+ },
179
+ "node_modules/destroy": {
180
+ "version": "1.2.0",
181
+ "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz",
182
+ "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==",
183
+ "license": "MIT",
184
+ "engines": {
185
+ "node": ">= 0.8",
186
+ "npm": "1.2.8000 || >= 1.4.16"
187
+ }
188
+ },
189
+ "node_modules/dotenv": {
190
+ "version": "16.6.1",
191
+ "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.6.1.tgz",
192
+ "integrity": "sha512-uBq4egWHTcTt33a72vpSG0z3HnPuIl6NqYcTrKEg2azoEyl2hpW0zqlxysq2pK9HlDIHyHyakeYaYnSAwd8bow==",
193
+ "license": "BSD-2-Clause",
194
+ "engines": {
195
+ "node": ">=12"
196
+ },
197
+ "funding": {
198
+ "url": "https://dotenvx.com"
199
+ }
200
+ },
201
+ "node_modules/dunder-proto": {
202
+ "version": "1.0.1",
203
+ "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
204
+ "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==",
205
+ "license": "MIT",
206
+ "dependencies": {
207
+ "call-bind-apply-helpers": "^1.0.1",
208
+ "es-errors": "^1.3.0",
209
+ "gopd": "^1.2.0"
210
+ },
211
+ "engines": {
212
+ "node": ">= 0.4"
213
+ }
214
+ },
215
+ "node_modules/ee-first": {
216
+ "version": "1.1.1",
217
+ "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
218
+ "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==",
219
+ "license": "MIT"
220
+ },
221
+ "node_modules/encodeurl": {
222
+ "version": "2.0.0",
223
+ "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz",
224
+ "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==",
225
+ "license": "MIT",
226
+ "engines": {
227
+ "node": ">= 0.8"
228
+ }
229
+ },
230
+ "node_modules/es-define-property": {
231
+ "version": "1.0.1",
232
+ "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
233
+ "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==",
234
+ "license": "MIT",
235
+ "engines": {
236
+ "node": ">= 0.4"
237
+ }
238
+ },
239
+ "node_modules/es-errors": {
240
+ "version": "1.3.0",
241
+ "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
242
+ "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
243
+ "license": "MIT",
244
+ "engines": {
245
+ "node": ">= 0.4"
246
+ }
247
+ },
248
+ "node_modules/es-object-atoms": {
249
+ "version": "1.1.1",
250
+ "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz",
251
+ "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==",
252
+ "license": "MIT",
253
+ "dependencies": {
254
+ "es-errors": "^1.3.0"
255
+ },
256
+ "engines": {
257
+ "node": ">= 0.4"
258
+ }
259
+ },
260
+ "node_modules/escape-html": {
261
+ "version": "1.0.3",
262
+ "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
263
+ "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==",
264
+ "license": "MIT"
265
+ },
266
+ "node_modules/etag": {
267
+ "version": "1.8.1",
268
+ "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
269
+ "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==",
270
+ "license": "MIT",
271
+ "engines": {
272
+ "node": ">= 0.6"
273
+ }
274
+ },
275
+ "node_modules/express": {
276
+ "version": "4.22.1",
277
+ "resolved": "https://registry.npmjs.org/express/-/express-4.22.1.tgz",
278
+ "integrity": "sha512-F2X8g9P1X7uCPZMA3MVf9wcTqlyNp7IhH5qPCI0izhaOIYXaW9L535tGA3qmjRzpH+bZczqq7hVKxTR4NWnu+g==",
279
+ "license": "MIT",
280
+ "dependencies": {
281
+ "accepts": "~1.3.8",
282
+ "array-flatten": "1.1.1",
283
+ "body-parser": "~1.20.3",
284
+ "content-disposition": "~0.5.4",
285
+ "content-type": "~1.0.4",
286
+ "cookie": "~0.7.1",
287
+ "cookie-signature": "~1.0.6",
288
+ "debug": "2.6.9",
289
+ "depd": "2.0.0",
290
+ "encodeurl": "~2.0.0",
291
+ "escape-html": "~1.0.3",
292
+ "etag": "~1.8.1",
293
+ "finalhandler": "~1.3.1",
294
+ "fresh": "~0.5.2",
295
+ "http-errors": "~2.0.0",
296
+ "merge-descriptors": "1.0.3",
297
+ "methods": "~1.1.2",
298
+ "on-finished": "~2.4.1",
299
+ "parseurl": "~1.3.3",
300
+ "path-to-regexp": "~0.1.12",
301
+ "proxy-addr": "~2.0.7",
302
+ "qs": "~6.14.0",
303
+ "range-parser": "~1.2.1",
304
+ "safe-buffer": "5.2.1",
305
+ "send": "~0.19.0",
306
+ "serve-static": "~1.16.2",
307
+ "setprototypeof": "1.2.0",
308
+ "statuses": "~2.0.1",
309
+ "type-is": "~1.6.18",
310
+ "utils-merge": "1.0.1",
311
+ "vary": "~1.1.2"
312
+ },
313
+ "engines": {
314
+ "node": ">= 0.10.0"
315
+ },
316
+ "funding": {
317
+ "type": "opencollective",
318
+ "url": "https://opencollective.com/express"
319
+ }
320
+ },
321
+ "node_modules/finalhandler": {
322
+ "version": "1.3.2",
323
+ "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.2.tgz",
324
+ "integrity": "sha512-aA4RyPcd3badbdABGDuTXCMTtOneUCAYH/gxoYRTZlIJdF0YPWuGqiAsIrhNnnqdXGswYk6dGujem4w80UJFhg==",
325
+ "license": "MIT",
326
+ "dependencies": {
327
+ "debug": "2.6.9",
328
+ "encodeurl": "~2.0.0",
329
+ "escape-html": "~1.0.3",
330
+ "on-finished": "~2.4.1",
331
+ "parseurl": "~1.3.3",
332
+ "statuses": "~2.0.2",
333
+ "unpipe": "~1.0.0"
334
+ },
335
+ "engines": {
336
+ "node": ">= 0.8"
337
+ }
338
+ },
339
+ "node_modules/forwarded": {
340
+ "version": "0.2.0",
341
+ "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
342
+ "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==",
343
+ "license": "MIT",
344
+ "engines": {
345
+ "node": ">= 0.6"
346
+ }
347
+ },
348
+ "node_modules/fresh": {
349
+ "version": "0.5.2",
350
+ "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
351
+ "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==",
352
+ "license": "MIT",
353
+ "engines": {
354
+ "node": ">= 0.6"
355
+ }
356
+ },
357
+ "node_modules/function-bind": {
358
+ "version": "1.1.2",
359
+ "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
360
+ "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
361
+ "license": "MIT",
362
+ "funding": {
363
+ "url": "https://github.com/sponsors/ljharb"
364
+ }
365
+ },
366
+ "node_modules/get-intrinsic": {
367
+ "version": "1.3.0",
368
+ "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
369
+ "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==",
370
+ "license": "MIT",
371
+ "dependencies": {
372
+ "call-bind-apply-helpers": "^1.0.2",
373
+ "es-define-property": "^1.0.1",
374
+ "es-errors": "^1.3.0",
375
+ "es-object-atoms": "^1.1.1",
376
+ "function-bind": "^1.1.2",
377
+ "get-proto": "^1.0.1",
378
+ "gopd": "^1.2.0",
379
+ "has-symbols": "^1.1.0",
380
+ "hasown": "^2.0.2",
381
+ "math-intrinsics": "^1.1.0"
382
+ },
383
+ "engines": {
384
+ "node": ">= 0.4"
385
+ },
386
+ "funding": {
387
+ "url": "https://github.com/sponsors/ljharb"
388
+ }
389
+ },
390
+ "node_modules/get-proto": {
391
+ "version": "1.0.1",
392
+ "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
393
+ "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==",
394
+ "license": "MIT",
395
+ "dependencies": {
396
+ "dunder-proto": "^1.0.1",
397
+ "es-object-atoms": "^1.0.0"
398
+ },
399
+ "engines": {
400
+ "node": ">= 0.4"
401
+ }
402
+ },
403
+ "node_modules/gopd": {
404
+ "version": "1.2.0",
405
+ "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz",
406
+ "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==",
407
+ "license": "MIT",
408
+ "engines": {
409
+ "node": ">= 0.4"
410
+ },
411
+ "funding": {
412
+ "url": "https://github.com/sponsors/ljharb"
413
+ }
414
+ },
415
+ "node_modules/has-symbols": {
416
+ "version": "1.1.0",
417
+ "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz",
418
+ "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==",
419
+ "license": "MIT",
420
+ "engines": {
421
+ "node": ">= 0.4"
422
+ },
423
+ "funding": {
424
+ "url": "https://github.com/sponsors/ljharb"
425
+ }
426
+ },
427
+ "node_modules/hasown": {
428
+ "version": "2.0.2",
429
+ "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
430
+ "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==",
431
+ "license": "MIT",
432
+ "dependencies": {
433
+ "function-bind": "^1.1.2"
434
+ },
435
+ "engines": {
436
+ "node": ">= 0.4"
437
+ }
438
+ },
439
+ "node_modules/http-errors": {
440
+ "version": "2.0.1",
441
+ "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.1.tgz",
442
+ "integrity": "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ==",
443
+ "license": "MIT",
444
+ "dependencies": {
445
+ "depd": "~2.0.0",
446
+ "inherits": "~2.0.4",
447
+ "setprototypeof": "~1.2.0",
448
+ "statuses": "~2.0.2",
449
+ "toidentifier": "~1.0.1"
450
+ },
451
+ "engines": {
452
+ "node": ">= 0.8"
453
+ },
454
+ "funding": {
455
+ "type": "opencollective",
456
+ "url": "https://opencollective.com/express"
457
+ }
458
+ },
459
+ "node_modules/iconv-lite": {
460
+ "version": "0.4.24",
461
+ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
462
+ "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
463
+ "license": "MIT",
464
+ "dependencies": {
465
+ "safer-buffer": ">= 2.1.2 < 3"
466
+ },
467
+ "engines": {
468
+ "node": ">=0.10.0"
469
+ }
470
+ },
471
+ "node_modules/inherits": {
472
+ "version": "2.0.4",
473
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
474
+ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
475
+ "license": "ISC"
476
+ },
477
+ "node_modules/ipaddr.js": {
478
+ "version": "1.9.1",
479
+ "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
480
+ "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==",
481
+ "license": "MIT",
482
+ "engines": {
483
+ "node": ">= 0.10"
484
+ }
485
+ },
486
+ "node_modules/math-intrinsics": {
487
+ "version": "1.1.0",
488
+ "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz",
489
+ "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==",
490
+ "license": "MIT",
491
+ "engines": {
492
+ "node": ">= 0.4"
493
+ }
494
+ },
495
+ "node_modules/media-typer": {
496
+ "version": "0.3.0",
497
+ "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
498
+ "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==",
499
+ "license": "MIT",
500
+ "engines": {
501
+ "node": ">= 0.6"
502
+ }
503
+ },
504
+ "node_modules/merge-descriptors": {
505
+ "version": "1.0.3",
506
+ "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz",
507
+ "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==",
508
+ "license": "MIT",
509
+ "funding": {
510
+ "url": "https://github.com/sponsors/sindresorhus"
511
+ }
512
+ },
513
+ "node_modules/methods": {
514
+ "version": "1.1.2",
515
+ "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
516
+ "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==",
517
+ "license": "MIT",
518
+ "engines": {
519
+ "node": ">= 0.6"
520
+ }
521
+ },
522
+ "node_modules/mime": {
523
+ "version": "1.6.0",
524
+ "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
525
+ "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==",
526
+ "license": "MIT",
527
+ "bin": {
528
+ "mime": "cli.js"
529
+ },
530
+ "engines": {
531
+ "node": ">=4"
532
+ }
533
+ },
534
+ "node_modules/mime-db": {
535
+ "version": "1.52.0",
536
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
537
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
538
+ "license": "MIT",
539
+ "engines": {
540
+ "node": ">= 0.6"
541
+ }
542
+ },
543
+ "node_modules/mime-types": {
544
+ "version": "2.1.35",
545
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
546
+ "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
547
+ "license": "MIT",
548
+ "dependencies": {
549
+ "mime-db": "1.52.0"
550
+ },
551
+ "engines": {
552
+ "node": ">= 0.6"
553
+ }
554
+ },
555
+ "node_modules/ms": {
556
+ "version": "2.0.0",
557
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
558
+ "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
559
+ "license": "MIT"
560
+ },
561
+ "node_modules/negotiator": {
562
+ "version": "0.6.3",
563
+ "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz",
564
+ "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==",
565
+ "license": "MIT",
566
+ "engines": {
567
+ "node": ">= 0.6"
568
+ }
569
+ },
570
+ "node_modules/object-assign": {
571
+ "version": "4.1.1",
572
+ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
573
+ "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
574
+ "license": "MIT",
575
+ "engines": {
576
+ "node": ">=0.10.0"
577
+ }
578
+ },
579
+ "node_modules/object-inspect": {
580
+ "version": "1.13.4",
581
+ "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz",
582
+ "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==",
583
+ "license": "MIT",
584
+ "engines": {
585
+ "node": ">= 0.4"
586
+ },
587
+ "funding": {
588
+ "url": "https://github.com/sponsors/ljharb"
589
+ }
590
+ },
591
+ "node_modules/on-finished": {
592
+ "version": "2.4.1",
593
+ "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz",
594
+ "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==",
595
+ "license": "MIT",
596
+ "dependencies": {
597
+ "ee-first": "1.1.1"
598
+ },
599
+ "engines": {
600
+ "node": ">= 0.8"
601
+ }
602
+ },
603
+ "node_modules/parseurl": {
604
+ "version": "1.3.3",
605
+ "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
606
+ "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==",
607
+ "license": "MIT",
608
+ "engines": {
609
+ "node": ">= 0.8"
610
+ }
611
+ },
612
+ "node_modules/path-to-regexp": {
613
+ "version": "0.1.12",
614
+ "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.12.tgz",
615
+ "integrity": "sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==",
616
+ "license": "MIT"
617
+ },
618
+ "node_modules/proxy-addr": {
619
+ "version": "2.0.7",
620
+ "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
621
+ "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
622
+ "license": "MIT",
623
+ "dependencies": {
624
+ "forwarded": "0.2.0",
625
+ "ipaddr.js": "1.9.1"
626
+ },
627
+ "engines": {
628
+ "node": ">= 0.10"
629
+ }
630
+ },
631
+ "node_modules/qs": {
632
+ "version": "6.14.2",
633
+ "resolved": "https://registry.npmjs.org/qs/-/qs-6.14.2.tgz",
634
+ "integrity": "sha512-V/yCWTTF7VJ9hIh18Ugr2zhJMP01MY7c5kh4J870L7imm6/DIzBsNLTXzMwUA3yZ5b/KBqLx8Kp3uRvd7xSe3Q==",
635
+ "license": "BSD-3-Clause",
636
+ "dependencies": {
637
+ "side-channel": "^1.1.0"
638
+ },
639
+ "engines": {
640
+ "node": ">=0.6"
641
+ },
642
+ "funding": {
643
+ "url": "https://github.com/sponsors/ljharb"
644
+ }
645
+ },
646
+ "node_modules/range-parser": {
647
+ "version": "1.2.1",
648
+ "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
649
+ "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
650
+ "license": "MIT",
651
+ "engines": {
652
+ "node": ">= 0.6"
653
+ }
654
+ },
655
+ "node_modules/raw-body": {
656
+ "version": "2.5.3",
657
+ "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.3.tgz",
658
+ "integrity": "sha512-s4VSOf6yN0rvbRZGxs8Om5CWj6seneMwK3oDb4lWDH0UPhWcxwOWw5+qk24bxq87szX1ydrwylIOp2uG1ojUpA==",
659
+ "license": "MIT",
660
+ "dependencies": {
661
+ "bytes": "~3.1.2",
662
+ "http-errors": "~2.0.1",
663
+ "iconv-lite": "~0.4.24",
664
+ "unpipe": "~1.0.0"
665
+ },
666
+ "engines": {
667
+ "node": ">= 0.8"
668
+ }
669
+ },
670
+ "node_modules/safe-buffer": {
671
+ "version": "5.2.1",
672
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
673
+ "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
674
+ "funding": [
675
+ {
676
+ "type": "github",
677
+ "url": "https://github.com/sponsors/feross"
678
+ },
679
+ {
680
+ "type": "patreon",
681
+ "url": "https://www.patreon.com/feross"
682
+ },
683
+ {
684
+ "type": "consulting",
685
+ "url": "https://feross.org/support"
686
+ }
687
+ ],
688
+ "license": "MIT"
689
+ },
690
+ "node_modules/safer-buffer": {
691
+ "version": "2.1.2",
692
+ "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
693
+ "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
694
+ "license": "MIT"
695
+ },
696
+ "node_modules/send": {
697
+ "version": "0.19.2",
698
+ "resolved": "https://registry.npmjs.org/send/-/send-0.19.2.tgz",
699
+ "integrity": "sha512-VMbMxbDeehAxpOtWJXlcUS5E8iXh6QmN+BkRX1GARS3wRaXEEgzCcB10gTQazO42tpNIya8xIyNx8fll1OFPrg==",
700
+ "license": "MIT",
701
+ "dependencies": {
702
+ "debug": "2.6.9",
703
+ "depd": "2.0.0",
704
+ "destroy": "1.2.0",
705
+ "encodeurl": "~2.0.0",
706
+ "escape-html": "~1.0.3",
707
+ "etag": "~1.8.1",
708
+ "fresh": "~0.5.2",
709
+ "http-errors": "~2.0.1",
710
+ "mime": "1.6.0",
711
+ "ms": "2.1.3",
712
+ "on-finished": "~2.4.1",
713
+ "range-parser": "~1.2.1",
714
+ "statuses": "~2.0.2"
715
+ },
716
+ "engines": {
717
+ "node": ">= 0.8.0"
718
+ }
719
+ },
720
+ "node_modules/send/node_modules/ms": {
721
+ "version": "2.1.3",
722
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
723
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
724
+ "license": "MIT"
725
+ },
726
+ "node_modules/serve-static": {
727
+ "version": "1.16.3",
728
+ "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.3.tgz",
729
+ "integrity": "sha512-x0RTqQel6g5SY7Lg6ZreMmsOzncHFU7nhnRWkKgWuMTu5NN0DR5oruckMqRvacAN9d5w6ARnRBXl9xhDCgfMeA==",
730
+ "license": "MIT",
731
+ "dependencies": {
732
+ "encodeurl": "~2.0.0",
733
+ "escape-html": "~1.0.3",
734
+ "parseurl": "~1.3.3",
735
+ "send": "~0.19.1"
736
+ },
737
+ "engines": {
738
+ "node": ">= 0.8.0"
739
+ }
740
+ },
741
+ "node_modules/setprototypeof": {
742
+ "version": "1.2.0",
743
+ "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
744
+ "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==",
745
+ "license": "ISC"
746
+ },
747
+ "node_modules/side-channel": {
748
+ "version": "1.1.0",
749
+ "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz",
750
+ "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==",
751
+ "license": "MIT",
752
+ "dependencies": {
753
+ "es-errors": "^1.3.0",
754
+ "object-inspect": "^1.13.3",
755
+ "side-channel-list": "^1.0.0",
756
+ "side-channel-map": "^1.0.1",
757
+ "side-channel-weakmap": "^1.0.2"
758
+ },
759
+ "engines": {
760
+ "node": ">= 0.4"
761
+ },
762
+ "funding": {
763
+ "url": "https://github.com/sponsors/ljharb"
764
+ }
765
+ },
766
+ "node_modules/side-channel-list": {
767
+ "version": "1.0.0",
768
+ "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz",
769
+ "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==",
770
+ "license": "MIT",
771
+ "dependencies": {
772
+ "es-errors": "^1.3.0",
773
+ "object-inspect": "^1.13.3"
774
+ },
775
+ "engines": {
776
+ "node": ">= 0.4"
777
+ },
778
+ "funding": {
779
+ "url": "https://github.com/sponsors/ljharb"
780
+ }
781
+ },
782
+ "node_modules/side-channel-map": {
783
+ "version": "1.0.1",
784
+ "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz",
785
+ "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==",
786
+ "license": "MIT",
787
+ "dependencies": {
788
+ "call-bound": "^1.0.2",
789
+ "es-errors": "^1.3.0",
790
+ "get-intrinsic": "^1.2.5",
791
+ "object-inspect": "^1.13.3"
792
+ },
793
+ "engines": {
794
+ "node": ">= 0.4"
795
+ },
796
+ "funding": {
797
+ "url": "https://github.com/sponsors/ljharb"
798
+ }
799
+ },
800
+ "node_modules/side-channel-weakmap": {
801
+ "version": "1.0.2",
802
+ "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz",
803
+ "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==",
804
+ "license": "MIT",
805
+ "dependencies": {
806
+ "call-bound": "^1.0.2",
807
+ "es-errors": "^1.3.0",
808
+ "get-intrinsic": "^1.2.5",
809
+ "object-inspect": "^1.13.3",
810
+ "side-channel-map": "^1.0.1"
811
+ },
812
+ "engines": {
813
+ "node": ">= 0.4"
814
+ },
815
+ "funding": {
816
+ "url": "https://github.com/sponsors/ljharb"
817
+ }
818
+ },
819
+ "node_modules/statuses": {
820
+ "version": "2.0.2",
821
+ "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.2.tgz",
822
+ "integrity": "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw==",
823
+ "license": "MIT",
824
+ "engines": {
825
+ "node": ">= 0.8"
826
+ }
827
+ },
828
+ "node_modules/toidentifier": {
829
+ "version": "1.0.1",
830
+ "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
831
+ "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==",
832
+ "license": "MIT",
833
+ "engines": {
834
+ "node": ">=0.6"
835
+ }
836
+ },
837
+ "node_modules/type-is": {
838
+ "version": "1.6.18",
839
+ "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
840
+ "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
841
+ "license": "MIT",
842
+ "dependencies": {
843
+ "media-typer": "0.3.0",
844
+ "mime-types": "~2.1.24"
845
+ },
846
+ "engines": {
847
+ "node": ">= 0.6"
848
+ }
849
+ },
850
+ "node_modules/unpipe": {
851
+ "version": "1.0.0",
852
+ "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
853
+ "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==",
854
+ "license": "MIT",
855
+ "engines": {
856
+ "node": ">= 0.8"
857
+ }
858
+ },
859
+ "node_modules/utils-merge": {
860
+ "version": "1.0.1",
861
+ "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
862
+ "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==",
863
+ "license": "MIT",
864
+ "engines": {
865
+ "node": ">= 0.4.0"
866
+ }
867
+ },
868
+ "node_modules/uuid": {
869
+ "version": "11.1.0",
870
+ "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.1.0.tgz",
871
+ "integrity": "sha512-0/A9rDy9P7cJ+8w1c9WD9V//9Wj15Ce2MPz8Ri6032usz+NfePxx5AcN3bN+r6ZL6jEo066/yNYB3tn4pQEx+A==",
872
+ "funding": [
873
+ "https://github.com/sponsors/broofa",
874
+ "https://github.com/sponsors/ctavan"
875
+ ],
876
+ "license": "MIT",
877
+ "bin": {
878
+ "uuid": "dist/esm/bin/uuid"
879
+ }
880
+ },
881
+ "node_modules/vary": {
882
+ "version": "1.1.2",
883
+ "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
884
+ "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==",
885
+ "license": "MIT",
886
+ "engines": {
887
+ "node": ">= 0.8"
888
+ }
889
+ }
890
+ }
891
+ }
backend/package.json ADDED
@@ -0,0 +1,16 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "interview-coach-backend",
3
+ "version": "1.0.0",
4
+ "description": "AI Interview Coach API",
5
+ "main": "server.js",
6
+ "scripts": {
7
+ "start": "node server.js",
8
+ "dev": "node server.js"
9
+ },
10
+ "dependencies": {
11
+ "cors": "^2.8.5",
12
+ "dotenv": "^16.4.7",
13
+ "express": "^4.21.2",
14
+ "uuid": "^11.0.5"
15
+ }
16
+ }
backend/routes/interview.js ADDED
@@ -0,0 +1,143 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ const express = require('express');
2
+ const { v4: uuidv4 } = require('uuid');
3
+ const { generateQuestions, evaluateAnswer, generateFinalReport } = require('../services/ai');
4
+
5
+ const router = express.Router();
6
+
7
+ // In-memory session store
8
+ const sessions = new Map();
9
+
10
+ // POST /api/interview/start - Start a new interview session
11
+ router.post('/start', async (req, res) => {
12
+ try {
13
+ const { role, difficulty, interviewType, questionCount } = req.body;
14
+
15
+ if (!role) {
16
+ return res.status(400).json({ error: 'Role is required' });
17
+ }
18
+
19
+ const count = Math.min(Math.max(questionCount || 5, 3), 10);
20
+ const data = await generateQuestions(
21
+ role,
22
+ difficulty || 'medium',
23
+ interviewType || 'mixed',
24
+ count
25
+ );
26
+
27
+ const sessionId = uuidv4();
28
+ sessions.set(sessionId, {
29
+ role,
30
+ difficulty: difficulty || 'medium',
31
+ interviewType: interviewType || 'mixed',
32
+ questions: data.questions,
33
+ answers: [],
34
+ currentIndex: 0,
35
+ startedAt: new Date().toISOString()
36
+ });
37
+
38
+ // Clean old sessions (older than 2 hours)
39
+ const twoHoursAgo = Date.now() - 2 * 60 * 60 * 1000;
40
+ for (const [id, session] of sessions) {
41
+ if (new Date(session.startedAt).getTime() < twoHoursAgo) {
42
+ sessions.delete(id);
43
+ }
44
+ }
45
+
46
+ res.json({
47
+ sessionId,
48
+ totalQuestions: data.questions.length,
49
+ currentQuestion: data.questions[0],
50
+ role,
51
+ difficulty: difficulty || 'medium'
52
+ });
53
+ } catch (err) {
54
+ console.error('Start error:', err);
55
+ res.status(500).json({ error: err.message || 'Failed to start interview' });
56
+ }
57
+ });
58
+
59
+ // POST /api/interview/answer - Submit an answer
60
+ router.post('/answer', async (req, res) => {
61
+ try {
62
+ const { sessionId, answer } = req.body;
63
+
64
+ if (!sessionId || !answer) {
65
+ return res.status(400).json({ error: 'sessionId and answer are required' });
66
+ }
67
+
68
+ const session = sessions.get(sessionId);
69
+ if (!session) {
70
+ return res.status(404).json({ error: 'Session not found or expired' });
71
+ }
72
+
73
+ const currentQ = session.questions[session.currentIndex];
74
+ const evaluation = await evaluateAnswer(
75
+ session.role,
76
+ currentQ.question,
77
+ answer,
78
+ session.difficulty
79
+ );
80
+
81
+ session.answers.push({
82
+ question: currentQ.question,
83
+ answer,
84
+ score: evaluation.score,
85
+ evaluation
86
+ });
87
+
88
+ session.currentIndex++;
89
+
90
+ const isComplete = session.currentIndex >= session.questions.length;
91
+ const nextQuestion = isComplete ? null : session.questions[session.currentIndex];
92
+
93
+ res.json({
94
+ evaluation,
95
+ isComplete,
96
+ nextQuestion,
97
+ progress: {
98
+ answered: session.currentIndex,
99
+ total: session.questions.length
100
+ }
101
+ });
102
+ } catch (err) {
103
+ console.error('Answer error:', err);
104
+ res.status(500).json({ error: err.message || 'Failed to evaluate answer' });
105
+ }
106
+ });
107
+
108
+ // GET /api/interview/report/:sessionId - Get final report
109
+ router.get('/report/:sessionId', async (req, res) => {
110
+ try {
111
+ const session = sessions.get(req.params.sessionId);
112
+ if (!session) {
113
+ return res.status(404).json({ error: 'Session not found or expired' });
114
+ }
115
+
116
+ if (session.answers.length < session.questions.length) {
117
+ return res.status(400).json({ error: 'Interview not complete yet' });
118
+ }
119
+
120
+ const report = await generateFinalReport(session.role, session.answers);
121
+
122
+ res.json({
123
+ report,
124
+ details: session.answers.map(a => ({
125
+ question: a.question,
126
+ answer: a.answer,
127
+ score: a.score,
128
+ feedback: a.evaluation.feedback
129
+ })),
130
+ meta: {
131
+ role: session.role,
132
+ difficulty: session.difficulty,
133
+ type: session.interviewType,
134
+ duration: session.startedAt
135
+ }
136
+ });
137
+ } catch (err) {
138
+ console.error('Report error:', err);
139
+ res.status(500).json({ error: err.message || 'Failed to generate report' });
140
+ }
141
+ });
142
+
143
+ module.exports = router;
backend/server.js ADDED
@@ -0,0 +1,29 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ const express = require('express');
2
+ const cors = require('cors');
3
+ const dotenv = require('dotenv');
4
+ const path = require('path');
5
+ const interviewRoutes = require('./routes/interview');
6
+
7
+ dotenv.config();
8
+
9
+ const app = express();
10
+ const PORT = process.env.PORT || 3002;
11
+
12
+ app.use(cors());
13
+ app.use(express.json());
14
+
15
+ app.use('/api/interview', interviewRoutes);
16
+
17
+ app.get('/api/health', (req, res) => {
18
+ res.json({ status: 'ok', service: 'interview-coach' });
19
+ });
20
+
21
+ app.use(express.static(path.join(__dirname, 'public')));
22
+
23
+ app.get('*', (req, res) => {
24
+ res.sendFile(path.join(__dirname, 'public', 'index.html'));
25
+ });
26
+
27
+ app.listen(PORT, () => {
28
+ console.log(`Interview Coach API running on port ${PORT}`);
29
+ });
backend/services/ai.js ADDED
@@ -0,0 +1,80 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ const GROQ_API_KEY = process.env.GROQ_API_KEY;
2
+ const OLLAMA_URL = process.env.OLLAMA_URL || 'http://localhost:11434';
3
+ const GROQ_MODEL = process.env.GROQ_MODEL || 'llama-3.3-70b-versatile';
4
+ const OLLAMA_MODEL = process.env.OLLAMA_MODEL || 'llama3.2:3b';
5
+
6
+ async function callAI(prompt) {
7
+ if (GROQ_API_KEY) {
8
+ const res = await fetch('https://api.groq.com/openai/v1/chat/completions', {
9
+ method: 'POST',
10
+ headers: { 'Content-Type': 'application/json', 'Authorization': `Bearer ${GROQ_API_KEY}` },
11
+ body: JSON.stringify({ model: GROQ_MODEL, messages: [{ role: 'user', content: prompt }], temperature: 0.7 }),
12
+ });
13
+ if (res.ok) { const data = await res.json(); return data.choices[0].message.content; }
14
+ console.warn('Groq failed, falling back to Ollama...');
15
+ }
16
+ const res = await fetch(`${OLLAMA_URL}/api/generate`, {
17
+ method: 'POST',
18
+ headers: { 'Content-Type': 'application/json' },
19
+ body: JSON.stringify({ model: OLLAMA_MODEL, prompt, stream: false }),
20
+ });
21
+ if (!res.ok) throw new Error('Both Groq and Ollama failed. Set GROQ_API_KEY or start Ollama.');
22
+ return (await res.json()).response;
23
+ }
24
+
25
+ function parseJSON(text) {
26
+ try { return JSON.parse(text.trim()); }
27
+ catch { const m = text.match(/\{[\s\S]*\}/); if (m) return JSON.parse(m[0]); throw new Error('Failed to parse AI response'); }
28
+ }
29
+
30
+ const DIFFICULTY_MAP = {
31
+ easy: 'Ask straightforward, common interview questions. Be encouraging.',
32
+ medium: 'Ask standard industry interview questions with some follow-ups. Be professional.',
33
+ hard: 'Ask challenging, senior-level questions with tricky follow-ups. Probe deeply into answers.'
34
+ };
35
+ const INTERVIEW_TYPES = {
36
+ behavioral: 'Focus on behavioral questions (STAR method).',
37
+ technical: 'Focus on technical knowledge, system design, problem-solving.',
38
+ mixed: 'Mix behavioral and technical questions naturally.'
39
+ };
40
+
41
+ async function generateQuestions(role, difficulty, interviewType, count = 5) {
42
+ const text = await callAI(`You are an expert interviewer for ${role} positions.
43
+ DIFFICULTY: ${difficulty} - ${DIFFICULTY_MAP[difficulty] || DIFFICULTY_MAP.medium}
44
+ TYPE: ${interviewType} - ${INTERVIEW_TYPES[interviewType] || INTERVIEW_TYPES.mixed}
45
+
46
+ Generate exactly ${count} interview questions. Return ONLY valid JSON:
47
+ {"questions":[{"id":1,"question":"<question>","category":"<behavioral|technical|situational>","hint":"<hint>"}]}
48
+
49
+ Return ONLY JSON, no markdown.`);
50
+ return parseJSON(text);
51
+ }
52
+
53
+ async function evaluateAnswer(role, question, answer, difficulty) {
54
+ const text = await callAI(`You are an expert interviewer evaluating a candidate for a ${role} position.
55
+ QUESTION: ${question}
56
+ CANDIDATE'S ANSWER: ${answer}
57
+ DIFFICULTY: ${difficulty}
58
+
59
+ Return ONLY valid JSON:
60
+ {"score":<1-10>,"feedback":"<2-3 sentences>","strengths":["<strength>"],"improvements":["<suggestion 1>","<suggestion 2>"],"sampleAnswer":"<strong example answer>"}
61
+
62
+ Return ONLY JSON, no markdown.`);
63
+ return parseJSON(text);
64
+ }
65
+
66
+ async function generateFinalReport(role, questionsAndAnswers) {
67
+ const qaPairs = questionsAndAnswers.map((qa, i) => `Q${i+1}: ${qa.question}\nAnswer: ${qa.answer}\nScore: ${qa.score}/10`).join('\n\n');
68
+ const text = await callAI(`You are an expert interviewer writing a final assessment for a ${role} candidate.
69
+
70
+ INTERVIEW TRANSCRIPT:
71
+ ${qaPairs}
72
+
73
+ Return ONLY valid JSON:
74
+ {"overallScore":<1-100>,"verdict":"<STRONG_HIRE|HIRE|MAYBE|NO_HIRE>","summary":"<3-4 sentences>","topStrengths":["<s1>","<s2>","<s3>"],"areasToImprove":["<a1>","<a2>","<a3>"],"recommendation":"<2-3 sentences>","readinessLevel":"<Not Ready|Getting There|Almost Ready|Interview Ready>"}
75
+
76
+ Return ONLY JSON, no markdown.`);
77
+ return parseJSON(text);
78
+ }
79
+
80
+ module.exports = { generateQuestions, evaluateAnswer, generateFinalReport };
frontend/index.html ADDED
@@ -0,0 +1,13 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
+ <title>AI Interview Coach</title>
7
+ <link rel="icon" type="image/svg+xml" href="data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 100 100'><text y='.9em' font-size='90'>🎤</text></svg>" />
8
+ </head>
9
+ <body>
10
+ <div id="root"></div>
11
+ <script type="module" src="/src/main.jsx"></script>
12
+ </body>
13
+ </html>
frontend/package-lock.json ADDED
@@ -0,0 +1,1773 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "interview-coach-frontend",
3
+ "version": "1.0.0",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "interview-coach-frontend",
9
+ "version": "1.0.0",
10
+ "dependencies": {
11
+ "react": "^19.0.0",
12
+ "react-dom": "^19.0.0"
13
+ },
14
+ "devDependencies": {
15
+ "@vitejs/plugin-react": "^4.3.4",
16
+ "vite": "^6.0.0"
17
+ }
18
+ },
19
+ "node_modules/@babel/code-frame": {
20
+ "version": "7.29.0",
21
+ "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.29.0.tgz",
22
+ "integrity": "sha512-9NhCeYjq9+3uxgdtp20LSiJXJvN0FeCtNGpJxuMFZ1Kv3cWUNb6DOhJwUvcVCzKGR66cw4njwM6hrJLqgOwbcw==",
23
+ "dev": true,
24
+ "license": "MIT",
25
+ "dependencies": {
26
+ "@babel/helper-validator-identifier": "^7.28.5",
27
+ "js-tokens": "^4.0.0",
28
+ "picocolors": "^1.1.1"
29
+ },
30
+ "engines": {
31
+ "node": ">=6.9.0"
32
+ }
33
+ },
34
+ "node_modules/@babel/compat-data": {
35
+ "version": "7.29.0",
36
+ "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.29.0.tgz",
37
+ "integrity": "sha512-T1NCJqT/j9+cn8fvkt7jtwbLBfLC/1y1c7NtCeXFRgzGTsafi68MRv8yzkYSapBnFA6L3U2VSc02ciDzoAJhJg==",
38
+ "dev": true,
39
+ "license": "MIT",
40
+ "engines": {
41
+ "node": ">=6.9.0"
42
+ }
43
+ },
44
+ "node_modules/@babel/core": {
45
+ "version": "7.29.0",
46
+ "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.29.0.tgz",
47
+ "integrity": "sha512-CGOfOJqWjg2qW/Mb6zNsDm+u5vFQ8DxXfbM09z69p5Z6+mE1ikP2jUXw+j42Pf1XTYED2Rni5f95npYeuwMDQA==",
48
+ "dev": true,
49
+ "license": "MIT",
50
+ "dependencies": {
51
+ "@babel/code-frame": "^7.29.0",
52
+ "@babel/generator": "^7.29.0",
53
+ "@babel/helper-compilation-targets": "^7.28.6",
54
+ "@babel/helper-module-transforms": "^7.28.6",
55
+ "@babel/helpers": "^7.28.6",
56
+ "@babel/parser": "^7.29.0",
57
+ "@babel/template": "^7.28.6",
58
+ "@babel/traverse": "^7.29.0",
59
+ "@babel/types": "^7.29.0",
60
+ "@jridgewell/remapping": "^2.3.5",
61
+ "convert-source-map": "^2.0.0",
62
+ "debug": "^4.1.0",
63
+ "gensync": "^1.0.0-beta.2",
64
+ "json5": "^2.2.3",
65
+ "semver": "^6.3.1"
66
+ },
67
+ "engines": {
68
+ "node": ">=6.9.0"
69
+ },
70
+ "funding": {
71
+ "type": "opencollective",
72
+ "url": "https://opencollective.com/babel"
73
+ }
74
+ },
75
+ "node_modules/@babel/generator": {
76
+ "version": "7.29.1",
77
+ "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.29.1.tgz",
78
+ "integrity": "sha512-qsaF+9Qcm2Qv8SRIMMscAvG4O3lJ0F1GuMo5HR/Bp02LopNgnZBC/EkbevHFeGs4ls/oPz9v+Bsmzbkbe+0dUw==",
79
+ "dev": true,
80
+ "license": "MIT",
81
+ "dependencies": {
82
+ "@babel/parser": "^7.29.0",
83
+ "@babel/types": "^7.29.0",
84
+ "@jridgewell/gen-mapping": "^0.3.12",
85
+ "@jridgewell/trace-mapping": "^0.3.28",
86
+ "jsesc": "^3.0.2"
87
+ },
88
+ "engines": {
89
+ "node": ">=6.9.0"
90
+ }
91
+ },
92
+ "node_modules/@babel/helper-compilation-targets": {
93
+ "version": "7.28.6",
94
+ "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.28.6.tgz",
95
+ "integrity": "sha512-JYtls3hqi15fcx5GaSNL7SCTJ2MNmjrkHXg4FSpOA/grxK8KwyZ5bubHsCq8FXCkua6xhuaaBit+3b7+VZRfcA==",
96
+ "dev": true,
97
+ "license": "MIT",
98
+ "dependencies": {
99
+ "@babel/compat-data": "^7.28.6",
100
+ "@babel/helper-validator-option": "^7.27.1",
101
+ "browserslist": "^4.24.0",
102
+ "lru-cache": "^5.1.1",
103
+ "semver": "^6.3.1"
104
+ },
105
+ "engines": {
106
+ "node": ">=6.9.0"
107
+ }
108
+ },
109
+ "node_modules/@babel/helper-globals": {
110
+ "version": "7.28.0",
111
+ "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz",
112
+ "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==",
113
+ "dev": true,
114
+ "license": "MIT",
115
+ "engines": {
116
+ "node": ">=6.9.0"
117
+ }
118
+ },
119
+ "node_modules/@babel/helper-module-imports": {
120
+ "version": "7.28.6",
121
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.28.6.tgz",
122
+ "integrity": "sha512-l5XkZK7r7wa9LucGw9LwZyyCUscb4x37JWTPz7swwFE/0FMQAGpiWUZn8u9DzkSBWEcK25jmvubfpw2dnAMdbw==",
123
+ "dev": true,
124
+ "license": "MIT",
125
+ "dependencies": {
126
+ "@babel/traverse": "^7.28.6",
127
+ "@babel/types": "^7.28.6"
128
+ },
129
+ "engines": {
130
+ "node": ">=6.9.0"
131
+ }
132
+ },
133
+ "node_modules/@babel/helper-module-transforms": {
134
+ "version": "7.28.6",
135
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.6.tgz",
136
+ "integrity": "sha512-67oXFAYr2cDLDVGLXTEABjdBJZ6drElUSI7WKp70NrpyISso3plG9SAGEF6y7zbha/wOzUByWWTJvEDVNIUGcA==",
137
+ "dev": true,
138
+ "license": "MIT",
139
+ "dependencies": {
140
+ "@babel/helper-module-imports": "^7.28.6",
141
+ "@babel/helper-validator-identifier": "^7.28.5",
142
+ "@babel/traverse": "^7.28.6"
143
+ },
144
+ "engines": {
145
+ "node": ">=6.9.0"
146
+ },
147
+ "peerDependencies": {
148
+ "@babel/core": "^7.0.0"
149
+ }
150
+ },
151
+ "node_modules/@babel/helper-plugin-utils": {
152
+ "version": "7.28.6",
153
+ "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.28.6.tgz",
154
+ "integrity": "sha512-S9gzZ/bz83GRysI7gAD4wPT/AI3uCnY+9xn+Mx/KPs2JwHJIz1W8PZkg2cqyt3RNOBM8ejcXhV6y8Og7ly/Dug==",
155
+ "dev": true,
156
+ "license": "MIT",
157
+ "engines": {
158
+ "node": ">=6.9.0"
159
+ }
160
+ },
161
+ "node_modules/@babel/helper-string-parser": {
162
+ "version": "7.27.1",
163
+ "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz",
164
+ "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==",
165
+ "dev": true,
166
+ "license": "MIT",
167
+ "engines": {
168
+ "node": ">=6.9.0"
169
+ }
170
+ },
171
+ "node_modules/@babel/helper-validator-identifier": {
172
+ "version": "7.28.5",
173
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz",
174
+ "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==",
175
+ "dev": true,
176
+ "license": "MIT",
177
+ "engines": {
178
+ "node": ">=6.9.0"
179
+ }
180
+ },
181
+ "node_modules/@babel/helper-validator-option": {
182
+ "version": "7.27.1",
183
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz",
184
+ "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==",
185
+ "dev": true,
186
+ "license": "MIT",
187
+ "engines": {
188
+ "node": ">=6.9.0"
189
+ }
190
+ },
191
+ "node_modules/@babel/helpers": {
192
+ "version": "7.28.6",
193
+ "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.6.tgz",
194
+ "integrity": "sha512-xOBvwq86HHdB7WUDTfKfT/Vuxh7gElQ+Sfti2Cy6yIWNW05P8iUslOVcZ4/sKbE+/jQaukQAdz/gf3724kYdqw==",
195
+ "dev": true,
196
+ "license": "MIT",
197
+ "dependencies": {
198
+ "@babel/template": "^7.28.6",
199
+ "@babel/types": "^7.28.6"
200
+ },
201
+ "engines": {
202
+ "node": ">=6.9.0"
203
+ }
204
+ },
205
+ "node_modules/@babel/parser": {
206
+ "version": "7.29.0",
207
+ "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.0.tgz",
208
+ "integrity": "sha512-IyDgFV5GeDUVX4YdF/3CPULtVGSXXMLh1xVIgdCgxApktqnQV0r7/8Nqthg+8YLGaAtdyIlo2qIdZrbCv4+7ww==",
209
+ "dev": true,
210
+ "license": "MIT",
211
+ "dependencies": {
212
+ "@babel/types": "^7.29.0"
213
+ },
214
+ "bin": {
215
+ "parser": "bin/babel-parser.js"
216
+ },
217
+ "engines": {
218
+ "node": ">=6.0.0"
219
+ }
220
+ },
221
+ "node_modules/@babel/plugin-transform-react-jsx-self": {
222
+ "version": "7.27.1",
223
+ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.27.1.tgz",
224
+ "integrity": "sha512-6UzkCs+ejGdZ5mFFC/OCUrv028ab2fp1znZmCZjAOBKiBK2jXD1O+BPSfX8X2qjJ75fZBMSnQn3Rq2mrBJK2mw==",
225
+ "dev": true,
226
+ "license": "MIT",
227
+ "dependencies": {
228
+ "@babel/helper-plugin-utils": "^7.27.1"
229
+ },
230
+ "engines": {
231
+ "node": ">=6.9.0"
232
+ },
233
+ "peerDependencies": {
234
+ "@babel/core": "^7.0.0-0"
235
+ }
236
+ },
237
+ "node_modules/@babel/plugin-transform-react-jsx-source": {
238
+ "version": "7.27.1",
239
+ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.27.1.tgz",
240
+ "integrity": "sha512-zbwoTsBruTeKB9hSq73ha66iFeJHuaFkUbwvqElnygoNbj/jHRsSeokowZFN3CZ64IvEqcmmkVe89OPXc7ldAw==",
241
+ "dev": true,
242
+ "license": "MIT",
243
+ "dependencies": {
244
+ "@babel/helper-plugin-utils": "^7.27.1"
245
+ },
246
+ "engines": {
247
+ "node": ">=6.9.0"
248
+ },
249
+ "peerDependencies": {
250
+ "@babel/core": "^7.0.0-0"
251
+ }
252
+ },
253
+ "node_modules/@babel/template": {
254
+ "version": "7.28.6",
255
+ "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz",
256
+ "integrity": "sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==",
257
+ "dev": true,
258
+ "license": "MIT",
259
+ "dependencies": {
260
+ "@babel/code-frame": "^7.28.6",
261
+ "@babel/parser": "^7.28.6",
262
+ "@babel/types": "^7.28.6"
263
+ },
264
+ "engines": {
265
+ "node": ">=6.9.0"
266
+ }
267
+ },
268
+ "node_modules/@babel/traverse": {
269
+ "version": "7.29.0",
270
+ "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.29.0.tgz",
271
+ "integrity": "sha512-4HPiQr0X7+waHfyXPZpWPfWL/J7dcN1mx9gL6WdQVMbPnF3+ZhSMs8tCxN7oHddJE9fhNE7+lxdnlyemKfJRuA==",
272
+ "dev": true,
273
+ "license": "MIT",
274
+ "dependencies": {
275
+ "@babel/code-frame": "^7.29.0",
276
+ "@babel/generator": "^7.29.0",
277
+ "@babel/helper-globals": "^7.28.0",
278
+ "@babel/parser": "^7.29.0",
279
+ "@babel/template": "^7.28.6",
280
+ "@babel/types": "^7.29.0",
281
+ "debug": "^4.3.1"
282
+ },
283
+ "engines": {
284
+ "node": ">=6.9.0"
285
+ }
286
+ },
287
+ "node_modules/@babel/types": {
288
+ "version": "7.29.0",
289
+ "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.0.tgz",
290
+ "integrity": "sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A==",
291
+ "dev": true,
292
+ "license": "MIT",
293
+ "dependencies": {
294
+ "@babel/helper-string-parser": "^7.27.1",
295
+ "@babel/helper-validator-identifier": "^7.28.5"
296
+ },
297
+ "engines": {
298
+ "node": ">=6.9.0"
299
+ }
300
+ },
301
+ "node_modules/@esbuild/aix-ppc64": {
302
+ "version": "0.25.12",
303
+ "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.12.tgz",
304
+ "integrity": "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA==",
305
+ "cpu": [
306
+ "ppc64"
307
+ ],
308
+ "dev": true,
309
+ "license": "MIT",
310
+ "optional": true,
311
+ "os": [
312
+ "aix"
313
+ ],
314
+ "engines": {
315
+ "node": ">=18"
316
+ }
317
+ },
318
+ "node_modules/@esbuild/android-arm": {
319
+ "version": "0.25.12",
320
+ "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.12.tgz",
321
+ "integrity": "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg==",
322
+ "cpu": [
323
+ "arm"
324
+ ],
325
+ "dev": true,
326
+ "license": "MIT",
327
+ "optional": true,
328
+ "os": [
329
+ "android"
330
+ ],
331
+ "engines": {
332
+ "node": ">=18"
333
+ }
334
+ },
335
+ "node_modules/@esbuild/android-arm64": {
336
+ "version": "0.25.12",
337
+ "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.12.tgz",
338
+ "integrity": "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg==",
339
+ "cpu": [
340
+ "arm64"
341
+ ],
342
+ "dev": true,
343
+ "license": "MIT",
344
+ "optional": true,
345
+ "os": [
346
+ "android"
347
+ ],
348
+ "engines": {
349
+ "node": ">=18"
350
+ }
351
+ },
352
+ "node_modules/@esbuild/android-x64": {
353
+ "version": "0.25.12",
354
+ "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.12.tgz",
355
+ "integrity": "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg==",
356
+ "cpu": [
357
+ "x64"
358
+ ],
359
+ "dev": true,
360
+ "license": "MIT",
361
+ "optional": true,
362
+ "os": [
363
+ "android"
364
+ ],
365
+ "engines": {
366
+ "node": ">=18"
367
+ }
368
+ },
369
+ "node_modules/@esbuild/darwin-arm64": {
370
+ "version": "0.25.12",
371
+ "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.12.tgz",
372
+ "integrity": "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==",
373
+ "cpu": [
374
+ "arm64"
375
+ ],
376
+ "dev": true,
377
+ "license": "MIT",
378
+ "optional": true,
379
+ "os": [
380
+ "darwin"
381
+ ],
382
+ "engines": {
383
+ "node": ">=18"
384
+ }
385
+ },
386
+ "node_modules/@esbuild/darwin-x64": {
387
+ "version": "0.25.12",
388
+ "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.12.tgz",
389
+ "integrity": "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA==",
390
+ "cpu": [
391
+ "x64"
392
+ ],
393
+ "dev": true,
394
+ "license": "MIT",
395
+ "optional": true,
396
+ "os": [
397
+ "darwin"
398
+ ],
399
+ "engines": {
400
+ "node": ">=18"
401
+ }
402
+ },
403
+ "node_modules/@esbuild/freebsd-arm64": {
404
+ "version": "0.25.12",
405
+ "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.12.tgz",
406
+ "integrity": "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg==",
407
+ "cpu": [
408
+ "arm64"
409
+ ],
410
+ "dev": true,
411
+ "license": "MIT",
412
+ "optional": true,
413
+ "os": [
414
+ "freebsd"
415
+ ],
416
+ "engines": {
417
+ "node": ">=18"
418
+ }
419
+ },
420
+ "node_modules/@esbuild/freebsd-x64": {
421
+ "version": "0.25.12",
422
+ "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.12.tgz",
423
+ "integrity": "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ==",
424
+ "cpu": [
425
+ "x64"
426
+ ],
427
+ "dev": true,
428
+ "license": "MIT",
429
+ "optional": true,
430
+ "os": [
431
+ "freebsd"
432
+ ],
433
+ "engines": {
434
+ "node": ">=18"
435
+ }
436
+ },
437
+ "node_modules/@esbuild/linux-arm": {
438
+ "version": "0.25.12",
439
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.12.tgz",
440
+ "integrity": "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw==",
441
+ "cpu": [
442
+ "arm"
443
+ ],
444
+ "dev": true,
445
+ "license": "MIT",
446
+ "optional": true,
447
+ "os": [
448
+ "linux"
449
+ ],
450
+ "engines": {
451
+ "node": ">=18"
452
+ }
453
+ },
454
+ "node_modules/@esbuild/linux-arm64": {
455
+ "version": "0.25.12",
456
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.12.tgz",
457
+ "integrity": "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ==",
458
+ "cpu": [
459
+ "arm64"
460
+ ],
461
+ "dev": true,
462
+ "license": "MIT",
463
+ "optional": true,
464
+ "os": [
465
+ "linux"
466
+ ],
467
+ "engines": {
468
+ "node": ">=18"
469
+ }
470
+ },
471
+ "node_modules/@esbuild/linux-ia32": {
472
+ "version": "0.25.12",
473
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.12.tgz",
474
+ "integrity": "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA==",
475
+ "cpu": [
476
+ "ia32"
477
+ ],
478
+ "dev": true,
479
+ "license": "MIT",
480
+ "optional": true,
481
+ "os": [
482
+ "linux"
483
+ ],
484
+ "engines": {
485
+ "node": ">=18"
486
+ }
487
+ },
488
+ "node_modules/@esbuild/linux-loong64": {
489
+ "version": "0.25.12",
490
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.12.tgz",
491
+ "integrity": "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng==",
492
+ "cpu": [
493
+ "loong64"
494
+ ],
495
+ "dev": true,
496
+ "license": "MIT",
497
+ "optional": true,
498
+ "os": [
499
+ "linux"
500
+ ],
501
+ "engines": {
502
+ "node": ">=18"
503
+ }
504
+ },
505
+ "node_modules/@esbuild/linux-mips64el": {
506
+ "version": "0.25.12",
507
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.12.tgz",
508
+ "integrity": "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw==",
509
+ "cpu": [
510
+ "mips64el"
511
+ ],
512
+ "dev": true,
513
+ "license": "MIT",
514
+ "optional": true,
515
+ "os": [
516
+ "linux"
517
+ ],
518
+ "engines": {
519
+ "node": ">=18"
520
+ }
521
+ },
522
+ "node_modules/@esbuild/linux-ppc64": {
523
+ "version": "0.25.12",
524
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.12.tgz",
525
+ "integrity": "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA==",
526
+ "cpu": [
527
+ "ppc64"
528
+ ],
529
+ "dev": true,
530
+ "license": "MIT",
531
+ "optional": true,
532
+ "os": [
533
+ "linux"
534
+ ],
535
+ "engines": {
536
+ "node": ">=18"
537
+ }
538
+ },
539
+ "node_modules/@esbuild/linux-riscv64": {
540
+ "version": "0.25.12",
541
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.12.tgz",
542
+ "integrity": "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w==",
543
+ "cpu": [
544
+ "riscv64"
545
+ ],
546
+ "dev": true,
547
+ "license": "MIT",
548
+ "optional": true,
549
+ "os": [
550
+ "linux"
551
+ ],
552
+ "engines": {
553
+ "node": ">=18"
554
+ }
555
+ },
556
+ "node_modules/@esbuild/linux-s390x": {
557
+ "version": "0.25.12",
558
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.12.tgz",
559
+ "integrity": "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg==",
560
+ "cpu": [
561
+ "s390x"
562
+ ],
563
+ "dev": true,
564
+ "license": "MIT",
565
+ "optional": true,
566
+ "os": [
567
+ "linux"
568
+ ],
569
+ "engines": {
570
+ "node": ">=18"
571
+ }
572
+ },
573
+ "node_modules/@esbuild/linux-x64": {
574
+ "version": "0.25.12",
575
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.12.tgz",
576
+ "integrity": "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw==",
577
+ "cpu": [
578
+ "x64"
579
+ ],
580
+ "dev": true,
581
+ "license": "MIT",
582
+ "optional": true,
583
+ "os": [
584
+ "linux"
585
+ ],
586
+ "engines": {
587
+ "node": ">=18"
588
+ }
589
+ },
590
+ "node_modules/@esbuild/netbsd-arm64": {
591
+ "version": "0.25.12",
592
+ "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.12.tgz",
593
+ "integrity": "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg==",
594
+ "cpu": [
595
+ "arm64"
596
+ ],
597
+ "dev": true,
598
+ "license": "MIT",
599
+ "optional": true,
600
+ "os": [
601
+ "netbsd"
602
+ ],
603
+ "engines": {
604
+ "node": ">=18"
605
+ }
606
+ },
607
+ "node_modules/@esbuild/netbsd-x64": {
608
+ "version": "0.25.12",
609
+ "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.12.tgz",
610
+ "integrity": "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ==",
611
+ "cpu": [
612
+ "x64"
613
+ ],
614
+ "dev": true,
615
+ "license": "MIT",
616
+ "optional": true,
617
+ "os": [
618
+ "netbsd"
619
+ ],
620
+ "engines": {
621
+ "node": ">=18"
622
+ }
623
+ },
624
+ "node_modules/@esbuild/openbsd-arm64": {
625
+ "version": "0.25.12",
626
+ "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.12.tgz",
627
+ "integrity": "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A==",
628
+ "cpu": [
629
+ "arm64"
630
+ ],
631
+ "dev": true,
632
+ "license": "MIT",
633
+ "optional": true,
634
+ "os": [
635
+ "openbsd"
636
+ ],
637
+ "engines": {
638
+ "node": ">=18"
639
+ }
640
+ },
641
+ "node_modules/@esbuild/openbsd-x64": {
642
+ "version": "0.25.12",
643
+ "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.12.tgz",
644
+ "integrity": "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw==",
645
+ "cpu": [
646
+ "x64"
647
+ ],
648
+ "dev": true,
649
+ "license": "MIT",
650
+ "optional": true,
651
+ "os": [
652
+ "openbsd"
653
+ ],
654
+ "engines": {
655
+ "node": ">=18"
656
+ }
657
+ },
658
+ "node_modules/@esbuild/openharmony-arm64": {
659
+ "version": "0.25.12",
660
+ "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.12.tgz",
661
+ "integrity": "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg==",
662
+ "cpu": [
663
+ "arm64"
664
+ ],
665
+ "dev": true,
666
+ "license": "MIT",
667
+ "optional": true,
668
+ "os": [
669
+ "openharmony"
670
+ ],
671
+ "engines": {
672
+ "node": ">=18"
673
+ }
674
+ },
675
+ "node_modules/@esbuild/sunos-x64": {
676
+ "version": "0.25.12",
677
+ "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.12.tgz",
678
+ "integrity": "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w==",
679
+ "cpu": [
680
+ "x64"
681
+ ],
682
+ "dev": true,
683
+ "license": "MIT",
684
+ "optional": true,
685
+ "os": [
686
+ "sunos"
687
+ ],
688
+ "engines": {
689
+ "node": ">=18"
690
+ }
691
+ },
692
+ "node_modules/@esbuild/win32-arm64": {
693
+ "version": "0.25.12",
694
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.12.tgz",
695
+ "integrity": "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg==",
696
+ "cpu": [
697
+ "arm64"
698
+ ],
699
+ "dev": true,
700
+ "license": "MIT",
701
+ "optional": true,
702
+ "os": [
703
+ "win32"
704
+ ],
705
+ "engines": {
706
+ "node": ">=18"
707
+ }
708
+ },
709
+ "node_modules/@esbuild/win32-ia32": {
710
+ "version": "0.25.12",
711
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.12.tgz",
712
+ "integrity": "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ==",
713
+ "cpu": [
714
+ "ia32"
715
+ ],
716
+ "dev": true,
717
+ "license": "MIT",
718
+ "optional": true,
719
+ "os": [
720
+ "win32"
721
+ ],
722
+ "engines": {
723
+ "node": ">=18"
724
+ }
725
+ },
726
+ "node_modules/@esbuild/win32-x64": {
727
+ "version": "0.25.12",
728
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.12.tgz",
729
+ "integrity": "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA==",
730
+ "cpu": [
731
+ "x64"
732
+ ],
733
+ "dev": true,
734
+ "license": "MIT",
735
+ "optional": true,
736
+ "os": [
737
+ "win32"
738
+ ],
739
+ "engines": {
740
+ "node": ">=18"
741
+ }
742
+ },
743
+ "node_modules/@jridgewell/gen-mapping": {
744
+ "version": "0.3.13",
745
+ "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz",
746
+ "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==",
747
+ "dev": true,
748
+ "license": "MIT",
749
+ "dependencies": {
750
+ "@jridgewell/sourcemap-codec": "^1.5.0",
751
+ "@jridgewell/trace-mapping": "^0.3.24"
752
+ }
753
+ },
754
+ "node_modules/@jridgewell/remapping": {
755
+ "version": "2.3.5",
756
+ "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz",
757
+ "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==",
758
+ "dev": true,
759
+ "license": "MIT",
760
+ "dependencies": {
761
+ "@jridgewell/gen-mapping": "^0.3.5",
762
+ "@jridgewell/trace-mapping": "^0.3.24"
763
+ }
764
+ },
765
+ "node_modules/@jridgewell/resolve-uri": {
766
+ "version": "3.1.2",
767
+ "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz",
768
+ "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==",
769
+ "dev": true,
770
+ "license": "MIT",
771
+ "engines": {
772
+ "node": ">=6.0.0"
773
+ }
774
+ },
775
+ "node_modules/@jridgewell/sourcemap-codec": {
776
+ "version": "1.5.5",
777
+ "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz",
778
+ "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==",
779
+ "dev": true,
780
+ "license": "MIT"
781
+ },
782
+ "node_modules/@jridgewell/trace-mapping": {
783
+ "version": "0.3.31",
784
+ "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz",
785
+ "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==",
786
+ "dev": true,
787
+ "license": "MIT",
788
+ "dependencies": {
789
+ "@jridgewell/resolve-uri": "^3.1.0",
790
+ "@jridgewell/sourcemap-codec": "^1.4.14"
791
+ }
792
+ },
793
+ "node_modules/@rolldown/pluginutils": {
794
+ "version": "1.0.0-beta.27",
795
+ "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.27.tgz",
796
+ "integrity": "sha512-+d0F4MKMCbeVUJwG96uQ4SgAznZNSq93I3V+9NHA4OpvqG8mRCpGdKmK8l/dl02h2CCDHwW2FqilnTyDcAnqjA==",
797
+ "dev": true,
798
+ "license": "MIT"
799
+ },
800
+ "node_modules/@rollup/rollup-android-arm-eabi": {
801
+ "version": "4.57.1",
802
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.57.1.tgz",
803
+ "integrity": "sha512-A6ehUVSiSaaliTxai040ZpZ2zTevHYbvu/lDoeAteHI8QnaosIzm4qwtezfRg1jOYaUmnzLX1AOD6Z+UJjtifg==",
804
+ "cpu": [
805
+ "arm"
806
+ ],
807
+ "dev": true,
808
+ "license": "MIT",
809
+ "optional": true,
810
+ "os": [
811
+ "android"
812
+ ]
813
+ },
814
+ "node_modules/@rollup/rollup-android-arm64": {
815
+ "version": "4.57.1",
816
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.57.1.tgz",
817
+ "integrity": "sha512-dQaAddCY9YgkFHZcFNS/606Exo8vcLHwArFZ7vxXq4rigo2bb494/xKMMwRRQW6ug7Js6yXmBZhSBRuBvCCQ3w==",
818
+ "cpu": [
819
+ "arm64"
820
+ ],
821
+ "dev": true,
822
+ "license": "MIT",
823
+ "optional": true,
824
+ "os": [
825
+ "android"
826
+ ]
827
+ },
828
+ "node_modules/@rollup/rollup-darwin-arm64": {
829
+ "version": "4.57.1",
830
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.57.1.tgz",
831
+ "integrity": "sha512-crNPrwJOrRxagUYeMn/DZwqN88SDmwaJ8Cvi/TN1HnWBU7GwknckyosC2gd0IqYRsHDEnXf328o9/HC6OkPgOg==",
832
+ "cpu": [
833
+ "arm64"
834
+ ],
835
+ "dev": true,
836
+ "license": "MIT",
837
+ "optional": true,
838
+ "os": [
839
+ "darwin"
840
+ ]
841
+ },
842
+ "node_modules/@rollup/rollup-darwin-x64": {
843
+ "version": "4.57.1",
844
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.57.1.tgz",
845
+ "integrity": "sha512-Ji8g8ChVbKrhFtig5QBV7iMaJrGtpHelkB3lsaKzadFBe58gmjfGXAOfI5FV0lYMH8wiqsxKQ1C9B0YTRXVy4w==",
846
+ "cpu": [
847
+ "x64"
848
+ ],
849
+ "dev": true,
850
+ "license": "MIT",
851
+ "optional": true,
852
+ "os": [
853
+ "darwin"
854
+ ]
855
+ },
856
+ "node_modules/@rollup/rollup-freebsd-arm64": {
857
+ "version": "4.57.1",
858
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.57.1.tgz",
859
+ "integrity": "sha512-R+/WwhsjmwodAcz65guCGFRkMb4gKWTcIeLy60JJQbXrJ97BOXHxnkPFrP+YwFlaS0m+uWJTstrUA9o+UchFug==",
860
+ "cpu": [
861
+ "arm64"
862
+ ],
863
+ "dev": true,
864
+ "license": "MIT",
865
+ "optional": true,
866
+ "os": [
867
+ "freebsd"
868
+ ]
869
+ },
870
+ "node_modules/@rollup/rollup-freebsd-x64": {
871
+ "version": "4.57.1",
872
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.57.1.tgz",
873
+ "integrity": "sha512-IEQTCHeiTOnAUC3IDQdzRAGj3jOAYNr9kBguI7MQAAZK3caezRrg0GxAb6Hchg4lxdZEI5Oq3iov/w/hnFWY9Q==",
874
+ "cpu": [
875
+ "x64"
876
+ ],
877
+ "dev": true,
878
+ "license": "MIT",
879
+ "optional": true,
880
+ "os": [
881
+ "freebsd"
882
+ ]
883
+ },
884
+ "node_modules/@rollup/rollup-linux-arm-gnueabihf": {
885
+ "version": "4.57.1",
886
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.57.1.tgz",
887
+ "integrity": "sha512-F8sWbhZ7tyuEfsmOxwc2giKDQzN3+kuBLPwwZGyVkLlKGdV1nvnNwYD0fKQ8+XS6hp9nY7B+ZeK01EBUE7aHaw==",
888
+ "cpu": [
889
+ "arm"
890
+ ],
891
+ "dev": true,
892
+ "license": "MIT",
893
+ "optional": true,
894
+ "os": [
895
+ "linux"
896
+ ]
897
+ },
898
+ "node_modules/@rollup/rollup-linux-arm-musleabihf": {
899
+ "version": "4.57.1",
900
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.57.1.tgz",
901
+ "integrity": "sha512-rGfNUfn0GIeXtBP1wL5MnzSj98+PZe/AXaGBCRmT0ts80lU5CATYGxXukeTX39XBKsxzFpEeK+Mrp9faXOlmrw==",
902
+ "cpu": [
903
+ "arm"
904
+ ],
905
+ "dev": true,
906
+ "license": "MIT",
907
+ "optional": true,
908
+ "os": [
909
+ "linux"
910
+ ]
911
+ },
912
+ "node_modules/@rollup/rollup-linux-arm64-gnu": {
913
+ "version": "4.57.1",
914
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.57.1.tgz",
915
+ "integrity": "sha512-MMtej3YHWeg/0klK2Qodf3yrNzz6CGjo2UntLvk2RSPlhzgLvYEB3frRvbEF2wRKh1Z2fDIg9KRPe1fawv7C+g==",
916
+ "cpu": [
917
+ "arm64"
918
+ ],
919
+ "dev": true,
920
+ "license": "MIT",
921
+ "optional": true,
922
+ "os": [
923
+ "linux"
924
+ ]
925
+ },
926
+ "node_modules/@rollup/rollup-linux-arm64-musl": {
927
+ "version": "4.57.1",
928
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.57.1.tgz",
929
+ "integrity": "sha512-1a/qhaaOXhqXGpMFMET9VqwZakkljWHLmZOX48R0I/YLbhdxr1m4gtG1Hq7++VhVUmf+L3sTAf9op4JlhQ5u1Q==",
930
+ "cpu": [
931
+ "arm64"
932
+ ],
933
+ "dev": true,
934
+ "license": "MIT",
935
+ "optional": true,
936
+ "os": [
937
+ "linux"
938
+ ]
939
+ },
940
+ "node_modules/@rollup/rollup-linux-loong64-gnu": {
941
+ "version": "4.57.1",
942
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.57.1.tgz",
943
+ "integrity": "sha512-QWO6RQTZ/cqYtJMtxhkRkidoNGXc7ERPbZN7dVW5SdURuLeVU7lwKMpo18XdcmpWYd0qsP1bwKPf7DNSUinhvA==",
944
+ "cpu": [
945
+ "loong64"
946
+ ],
947
+ "dev": true,
948
+ "license": "MIT",
949
+ "optional": true,
950
+ "os": [
951
+ "linux"
952
+ ]
953
+ },
954
+ "node_modules/@rollup/rollup-linux-loong64-musl": {
955
+ "version": "4.57.1",
956
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.57.1.tgz",
957
+ "integrity": "sha512-xpObYIf+8gprgWaPP32xiN5RVTi/s5FCR+XMXSKmhfoJjrpRAjCuuqQXyxUa/eJTdAE6eJ+KDKaoEqjZQxh3Gw==",
958
+ "cpu": [
959
+ "loong64"
960
+ ],
961
+ "dev": true,
962
+ "license": "MIT",
963
+ "optional": true,
964
+ "os": [
965
+ "linux"
966
+ ]
967
+ },
968
+ "node_modules/@rollup/rollup-linux-ppc64-gnu": {
969
+ "version": "4.57.1",
970
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.57.1.tgz",
971
+ "integrity": "sha512-4BrCgrpZo4hvzMDKRqEaW1zeecScDCR+2nZ86ATLhAoJ5FQ+lbHVD3ttKe74/c7tNT9c6F2viwB3ufwp01Oh2w==",
972
+ "cpu": [
973
+ "ppc64"
974
+ ],
975
+ "dev": true,
976
+ "license": "MIT",
977
+ "optional": true,
978
+ "os": [
979
+ "linux"
980
+ ]
981
+ },
982
+ "node_modules/@rollup/rollup-linux-ppc64-musl": {
983
+ "version": "4.57.1",
984
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.57.1.tgz",
985
+ "integrity": "sha512-NOlUuzesGauESAyEYFSe3QTUguL+lvrN1HtwEEsU2rOwdUDeTMJdO5dUYl/2hKf9jWydJrO9OL/XSSf65R5+Xw==",
986
+ "cpu": [
987
+ "ppc64"
988
+ ],
989
+ "dev": true,
990
+ "license": "MIT",
991
+ "optional": true,
992
+ "os": [
993
+ "linux"
994
+ ]
995
+ },
996
+ "node_modules/@rollup/rollup-linux-riscv64-gnu": {
997
+ "version": "4.57.1",
998
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.57.1.tgz",
999
+ "integrity": "sha512-ptA88htVp0AwUUqhVghwDIKlvJMD/fmL/wrQj99PRHFRAG6Z5nbWoWG4o81Nt9FT+IuqUQi+L31ZKAFeJ5Is+A==",
1000
+ "cpu": [
1001
+ "riscv64"
1002
+ ],
1003
+ "dev": true,
1004
+ "license": "MIT",
1005
+ "optional": true,
1006
+ "os": [
1007
+ "linux"
1008
+ ]
1009
+ },
1010
+ "node_modules/@rollup/rollup-linux-riscv64-musl": {
1011
+ "version": "4.57.1",
1012
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.57.1.tgz",
1013
+ "integrity": "sha512-S51t7aMMTNdmAMPpBg7OOsTdn4tySRQvklmL3RpDRyknk87+Sp3xaumlatU+ppQ+5raY7sSTcC2beGgvhENfuw==",
1014
+ "cpu": [
1015
+ "riscv64"
1016
+ ],
1017
+ "dev": true,
1018
+ "license": "MIT",
1019
+ "optional": true,
1020
+ "os": [
1021
+ "linux"
1022
+ ]
1023
+ },
1024
+ "node_modules/@rollup/rollup-linux-s390x-gnu": {
1025
+ "version": "4.57.1",
1026
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.57.1.tgz",
1027
+ "integrity": "sha512-Bl00OFnVFkL82FHbEqy3k5CUCKH6OEJL54KCyx2oqsmZnFTR8IoNqBF+mjQVcRCT5sB6yOvK8A37LNm/kPJiZg==",
1028
+ "cpu": [
1029
+ "s390x"
1030
+ ],
1031
+ "dev": true,
1032
+ "license": "MIT",
1033
+ "optional": true,
1034
+ "os": [
1035
+ "linux"
1036
+ ]
1037
+ },
1038
+ "node_modules/@rollup/rollup-linux-x64-gnu": {
1039
+ "version": "4.57.1",
1040
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.57.1.tgz",
1041
+ "integrity": "sha512-ABca4ceT4N+Tv/GtotnWAeXZUZuM/9AQyCyKYyKnpk4yoA7QIAuBt6Hkgpw8kActYlew2mvckXkvx0FfoInnLg==",
1042
+ "cpu": [
1043
+ "x64"
1044
+ ],
1045
+ "dev": true,
1046
+ "license": "MIT",
1047
+ "optional": true,
1048
+ "os": [
1049
+ "linux"
1050
+ ]
1051
+ },
1052
+ "node_modules/@rollup/rollup-linux-x64-musl": {
1053
+ "version": "4.57.1",
1054
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.57.1.tgz",
1055
+ "integrity": "sha512-HFps0JeGtuOR2convgRRkHCekD7j+gdAuXM+/i6kGzQtFhlCtQkpwtNzkNj6QhCDp7DRJ7+qC/1Vg2jt5iSOFw==",
1056
+ "cpu": [
1057
+ "x64"
1058
+ ],
1059
+ "dev": true,
1060
+ "license": "MIT",
1061
+ "optional": true,
1062
+ "os": [
1063
+ "linux"
1064
+ ]
1065
+ },
1066
+ "node_modules/@rollup/rollup-openbsd-x64": {
1067
+ "version": "4.57.1",
1068
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.57.1.tgz",
1069
+ "integrity": "sha512-H+hXEv9gdVQuDTgnqD+SQffoWoc0Of59AStSzTEj/feWTBAnSfSD3+Dql1ZruJQxmykT/JVY0dE8Ka7z0DH1hw==",
1070
+ "cpu": [
1071
+ "x64"
1072
+ ],
1073
+ "dev": true,
1074
+ "license": "MIT",
1075
+ "optional": true,
1076
+ "os": [
1077
+ "openbsd"
1078
+ ]
1079
+ },
1080
+ "node_modules/@rollup/rollup-openharmony-arm64": {
1081
+ "version": "4.57.1",
1082
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.57.1.tgz",
1083
+ "integrity": "sha512-4wYoDpNg6o/oPximyc/NG+mYUejZrCU2q+2w6YZqrAs2UcNUChIZXjtafAiiZSUc7On8v5NyNj34Kzj/Ltk6dQ==",
1084
+ "cpu": [
1085
+ "arm64"
1086
+ ],
1087
+ "dev": true,
1088
+ "license": "MIT",
1089
+ "optional": true,
1090
+ "os": [
1091
+ "openharmony"
1092
+ ]
1093
+ },
1094
+ "node_modules/@rollup/rollup-win32-arm64-msvc": {
1095
+ "version": "4.57.1",
1096
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.57.1.tgz",
1097
+ "integrity": "sha512-O54mtsV/6LW3P8qdTcamQmuC990HDfR71lo44oZMZlXU4tzLrbvTii87Ni9opq60ds0YzuAlEr/GNwuNluZyMQ==",
1098
+ "cpu": [
1099
+ "arm64"
1100
+ ],
1101
+ "dev": true,
1102
+ "license": "MIT",
1103
+ "optional": true,
1104
+ "os": [
1105
+ "win32"
1106
+ ]
1107
+ },
1108
+ "node_modules/@rollup/rollup-win32-ia32-msvc": {
1109
+ "version": "4.57.1",
1110
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.57.1.tgz",
1111
+ "integrity": "sha512-P3dLS+IerxCT/7D2q2FYcRdWRl22dNbrbBEtxdWhXrfIMPP9lQhb5h4Du04mdl5Woq05jVCDPCMF7Ub0NAjIew==",
1112
+ "cpu": [
1113
+ "ia32"
1114
+ ],
1115
+ "dev": true,
1116
+ "license": "MIT",
1117
+ "optional": true,
1118
+ "os": [
1119
+ "win32"
1120
+ ]
1121
+ },
1122
+ "node_modules/@rollup/rollup-win32-x64-gnu": {
1123
+ "version": "4.57.1",
1124
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.57.1.tgz",
1125
+ "integrity": "sha512-VMBH2eOOaKGtIJYleXsi2B8CPVADrh+TyNxJ4mWPnKfLB/DBUmzW+5m1xUrcwWoMfSLagIRpjUFeW5CO5hyciQ==",
1126
+ "cpu": [
1127
+ "x64"
1128
+ ],
1129
+ "dev": true,
1130
+ "license": "MIT",
1131
+ "optional": true,
1132
+ "os": [
1133
+ "win32"
1134
+ ]
1135
+ },
1136
+ "node_modules/@rollup/rollup-win32-x64-msvc": {
1137
+ "version": "4.57.1",
1138
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.57.1.tgz",
1139
+ "integrity": "sha512-mxRFDdHIWRxg3UfIIAwCm6NzvxG0jDX/wBN6KsQFTvKFqqg9vTrWUE68qEjHt19A5wwx5X5aUi2zuZT7YR0jrA==",
1140
+ "cpu": [
1141
+ "x64"
1142
+ ],
1143
+ "dev": true,
1144
+ "license": "MIT",
1145
+ "optional": true,
1146
+ "os": [
1147
+ "win32"
1148
+ ]
1149
+ },
1150
+ "node_modules/@types/babel__core": {
1151
+ "version": "7.20.5",
1152
+ "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz",
1153
+ "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==",
1154
+ "dev": true,
1155
+ "license": "MIT",
1156
+ "dependencies": {
1157
+ "@babel/parser": "^7.20.7",
1158
+ "@babel/types": "^7.20.7",
1159
+ "@types/babel__generator": "*",
1160
+ "@types/babel__template": "*",
1161
+ "@types/babel__traverse": "*"
1162
+ }
1163
+ },
1164
+ "node_modules/@types/babel__generator": {
1165
+ "version": "7.27.0",
1166
+ "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.27.0.tgz",
1167
+ "integrity": "sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==",
1168
+ "dev": true,
1169
+ "license": "MIT",
1170
+ "dependencies": {
1171
+ "@babel/types": "^7.0.0"
1172
+ }
1173
+ },
1174
+ "node_modules/@types/babel__template": {
1175
+ "version": "7.4.4",
1176
+ "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz",
1177
+ "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==",
1178
+ "dev": true,
1179
+ "license": "MIT",
1180
+ "dependencies": {
1181
+ "@babel/parser": "^7.1.0",
1182
+ "@babel/types": "^7.0.0"
1183
+ }
1184
+ },
1185
+ "node_modules/@types/babel__traverse": {
1186
+ "version": "7.28.0",
1187
+ "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz",
1188
+ "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==",
1189
+ "dev": true,
1190
+ "license": "MIT",
1191
+ "dependencies": {
1192
+ "@babel/types": "^7.28.2"
1193
+ }
1194
+ },
1195
+ "node_modules/@types/estree": {
1196
+ "version": "1.0.8",
1197
+ "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz",
1198
+ "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==",
1199
+ "dev": true,
1200
+ "license": "MIT"
1201
+ },
1202
+ "node_modules/@vitejs/plugin-react": {
1203
+ "version": "4.7.0",
1204
+ "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.7.0.tgz",
1205
+ "integrity": "sha512-gUu9hwfWvvEDBBmgtAowQCojwZmJ5mcLn3aufeCsitijs3+f2NsrPtlAWIR6OPiqljl96GVCUbLe0HyqIpVaoA==",
1206
+ "dev": true,
1207
+ "license": "MIT",
1208
+ "dependencies": {
1209
+ "@babel/core": "^7.28.0",
1210
+ "@babel/plugin-transform-react-jsx-self": "^7.27.1",
1211
+ "@babel/plugin-transform-react-jsx-source": "^7.27.1",
1212
+ "@rolldown/pluginutils": "1.0.0-beta.27",
1213
+ "@types/babel__core": "^7.20.5",
1214
+ "react-refresh": "^0.17.0"
1215
+ },
1216
+ "engines": {
1217
+ "node": "^14.18.0 || >=16.0.0"
1218
+ },
1219
+ "peerDependencies": {
1220
+ "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0"
1221
+ }
1222
+ },
1223
+ "node_modules/baseline-browser-mapping": {
1224
+ "version": "2.9.19",
1225
+ "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.9.19.tgz",
1226
+ "integrity": "sha512-ipDqC8FrAl/76p2SSWKSI+H9tFwm7vYqXQrItCuiVPt26Km0jS+NzSsBWAaBusvSbQcfJG+JitdMm+wZAgTYqg==",
1227
+ "dev": true,
1228
+ "license": "Apache-2.0",
1229
+ "bin": {
1230
+ "baseline-browser-mapping": "dist/cli.js"
1231
+ }
1232
+ },
1233
+ "node_modules/browserslist": {
1234
+ "version": "4.28.1",
1235
+ "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.1.tgz",
1236
+ "integrity": "sha512-ZC5Bd0LgJXgwGqUknZY/vkUQ04r8NXnJZ3yYi4vDmSiZmC/pdSN0NbNRPxZpbtO4uAfDUAFffO8IZoM3Gj8IkA==",
1237
+ "dev": true,
1238
+ "funding": [
1239
+ {
1240
+ "type": "opencollective",
1241
+ "url": "https://opencollective.com/browserslist"
1242
+ },
1243
+ {
1244
+ "type": "tidelift",
1245
+ "url": "https://tidelift.com/funding/github/npm/browserslist"
1246
+ },
1247
+ {
1248
+ "type": "github",
1249
+ "url": "https://github.com/sponsors/ai"
1250
+ }
1251
+ ],
1252
+ "license": "MIT",
1253
+ "dependencies": {
1254
+ "baseline-browser-mapping": "^2.9.0",
1255
+ "caniuse-lite": "^1.0.30001759",
1256
+ "electron-to-chromium": "^1.5.263",
1257
+ "node-releases": "^2.0.27",
1258
+ "update-browserslist-db": "^1.2.0"
1259
+ },
1260
+ "bin": {
1261
+ "browserslist": "cli.js"
1262
+ },
1263
+ "engines": {
1264
+ "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7"
1265
+ }
1266
+ },
1267
+ "node_modules/caniuse-lite": {
1268
+ "version": "1.0.30001769",
1269
+ "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001769.tgz",
1270
+ "integrity": "sha512-BCfFL1sHijQlBGWBMuJyhZUhzo7wer5sVj9hqekB/7xn0Ypy+pER/edCYQm4exbXj4WiySGp40P8UuTh6w1srg==",
1271
+ "dev": true,
1272
+ "funding": [
1273
+ {
1274
+ "type": "opencollective",
1275
+ "url": "https://opencollective.com/browserslist"
1276
+ },
1277
+ {
1278
+ "type": "tidelift",
1279
+ "url": "https://tidelift.com/funding/github/npm/caniuse-lite"
1280
+ },
1281
+ {
1282
+ "type": "github",
1283
+ "url": "https://github.com/sponsors/ai"
1284
+ }
1285
+ ],
1286
+ "license": "CC-BY-4.0"
1287
+ },
1288
+ "node_modules/convert-source-map": {
1289
+ "version": "2.0.0",
1290
+ "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz",
1291
+ "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==",
1292
+ "dev": true,
1293
+ "license": "MIT"
1294
+ },
1295
+ "node_modules/debug": {
1296
+ "version": "4.4.3",
1297
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
1298
+ "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==",
1299
+ "dev": true,
1300
+ "license": "MIT",
1301
+ "dependencies": {
1302
+ "ms": "^2.1.3"
1303
+ },
1304
+ "engines": {
1305
+ "node": ">=6.0"
1306
+ },
1307
+ "peerDependenciesMeta": {
1308
+ "supports-color": {
1309
+ "optional": true
1310
+ }
1311
+ }
1312
+ },
1313
+ "node_modules/electron-to-chromium": {
1314
+ "version": "1.5.286",
1315
+ "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.286.tgz",
1316
+ "integrity": "sha512-9tfDXhJ4RKFNerfjdCcZfufu49vg620741MNs26a9+bhLThdB+plgMeou98CAaHu/WATj2iHOOHTp1hWtABj2A==",
1317
+ "dev": true,
1318
+ "license": "ISC"
1319
+ },
1320
+ "node_modules/esbuild": {
1321
+ "version": "0.25.12",
1322
+ "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.12.tgz",
1323
+ "integrity": "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==",
1324
+ "dev": true,
1325
+ "hasInstallScript": true,
1326
+ "license": "MIT",
1327
+ "bin": {
1328
+ "esbuild": "bin/esbuild"
1329
+ },
1330
+ "engines": {
1331
+ "node": ">=18"
1332
+ },
1333
+ "optionalDependencies": {
1334
+ "@esbuild/aix-ppc64": "0.25.12",
1335
+ "@esbuild/android-arm": "0.25.12",
1336
+ "@esbuild/android-arm64": "0.25.12",
1337
+ "@esbuild/android-x64": "0.25.12",
1338
+ "@esbuild/darwin-arm64": "0.25.12",
1339
+ "@esbuild/darwin-x64": "0.25.12",
1340
+ "@esbuild/freebsd-arm64": "0.25.12",
1341
+ "@esbuild/freebsd-x64": "0.25.12",
1342
+ "@esbuild/linux-arm": "0.25.12",
1343
+ "@esbuild/linux-arm64": "0.25.12",
1344
+ "@esbuild/linux-ia32": "0.25.12",
1345
+ "@esbuild/linux-loong64": "0.25.12",
1346
+ "@esbuild/linux-mips64el": "0.25.12",
1347
+ "@esbuild/linux-ppc64": "0.25.12",
1348
+ "@esbuild/linux-riscv64": "0.25.12",
1349
+ "@esbuild/linux-s390x": "0.25.12",
1350
+ "@esbuild/linux-x64": "0.25.12",
1351
+ "@esbuild/netbsd-arm64": "0.25.12",
1352
+ "@esbuild/netbsd-x64": "0.25.12",
1353
+ "@esbuild/openbsd-arm64": "0.25.12",
1354
+ "@esbuild/openbsd-x64": "0.25.12",
1355
+ "@esbuild/openharmony-arm64": "0.25.12",
1356
+ "@esbuild/sunos-x64": "0.25.12",
1357
+ "@esbuild/win32-arm64": "0.25.12",
1358
+ "@esbuild/win32-ia32": "0.25.12",
1359
+ "@esbuild/win32-x64": "0.25.12"
1360
+ }
1361
+ },
1362
+ "node_modules/escalade": {
1363
+ "version": "3.2.0",
1364
+ "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
1365
+ "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==",
1366
+ "dev": true,
1367
+ "license": "MIT",
1368
+ "engines": {
1369
+ "node": ">=6"
1370
+ }
1371
+ },
1372
+ "node_modules/fdir": {
1373
+ "version": "6.5.0",
1374
+ "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz",
1375
+ "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==",
1376
+ "dev": true,
1377
+ "license": "MIT",
1378
+ "engines": {
1379
+ "node": ">=12.0.0"
1380
+ },
1381
+ "peerDependencies": {
1382
+ "picomatch": "^3 || ^4"
1383
+ },
1384
+ "peerDependenciesMeta": {
1385
+ "picomatch": {
1386
+ "optional": true
1387
+ }
1388
+ }
1389
+ },
1390
+ "node_modules/fsevents": {
1391
+ "version": "2.3.3",
1392
+ "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
1393
+ "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
1394
+ "dev": true,
1395
+ "hasInstallScript": true,
1396
+ "license": "MIT",
1397
+ "optional": true,
1398
+ "os": [
1399
+ "darwin"
1400
+ ],
1401
+ "engines": {
1402
+ "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
1403
+ }
1404
+ },
1405
+ "node_modules/gensync": {
1406
+ "version": "1.0.0-beta.2",
1407
+ "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz",
1408
+ "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==",
1409
+ "dev": true,
1410
+ "license": "MIT",
1411
+ "engines": {
1412
+ "node": ">=6.9.0"
1413
+ }
1414
+ },
1415
+ "node_modules/js-tokens": {
1416
+ "version": "4.0.0",
1417
+ "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
1418
+ "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
1419
+ "dev": true,
1420
+ "license": "MIT"
1421
+ },
1422
+ "node_modules/jsesc": {
1423
+ "version": "3.1.0",
1424
+ "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz",
1425
+ "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==",
1426
+ "dev": true,
1427
+ "license": "MIT",
1428
+ "bin": {
1429
+ "jsesc": "bin/jsesc"
1430
+ },
1431
+ "engines": {
1432
+ "node": ">=6"
1433
+ }
1434
+ },
1435
+ "node_modules/json5": {
1436
+ "version": "2.2.3",
1437
+ "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz",
1438
+ "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==",
1439
+ "dev": true,
1440
+ "license": "MIT",
1441
+ "bin": {
1442
+ "json5": "lib/cli.js"
1443
+ },
1444
+ "engines": {
1445
+ "node": ">=6"
1446
+ }
1447
+ },
1448
+ "node_modules/lru-cache": {
1449
+ "version": "5.1.1",
1450
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz",
1451
+ "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==",
1452
+ "dev": true,
1453
+ "license": "ISC",
1454
+ "dependencies": {
1455
+ "yallist": "^3.0.2"
1456
+ }
1457
+ },
1458
+ "node_modules/ms": {
1459
+ "version": "2.1.3",
1460
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
1461
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
1462
+ "dev": true,
1463
+ "license": "MIT"
1464
+ },
1465
+ "node_modules/nanoid": {
1466
+ "version": "3.3.11",
1467
+ "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz",
1468
+ "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==",
1469
+ "dev": true,
1470
+ "funding": [
1471
+ {
1472
+ "type": "github",
1473
+ "url": "https://github.com/sponsors/ai"
1474
+ }
1475
+ ],
1476
+ "license": "MIT",
1477
+ "bin": {
1478
+ "nanoid": "bin/nanoid.cjs"
1479
+ },
1480
+ "engines": {
1481
+ "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1"
1482
+ }
1483
+ },
1484
+ "node_modules/node-releases": {
1485
+ "version": "2.0.27",
1486
+ "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz",
1487
+ "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==",
1488
+ "dev": true,
1489
+ "license": "MIT"
1490
+ },
1491
+ "node_modules/picocolors": {
1492
+ "version": "1.1.1",
1493
+ "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
1494
+ "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==",
1495
+ "dev": true,
1496
+ "license": "ISC"
1497
+ },
1498
+ "node_modules/picomatch": {
1499
+ "version": "4.0.3",
1500
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz",
1501
+ "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==",
1502
+ "dev": true,
1503
+ "license": "MIT",
1504
+ "engines": {
1505
+ "node": ">=12"
1506
+ },
1507
+ "funding": {
1508
+ "url": "https://github.com/sponsors/jonschlinkert"
1509
+ }
1510
+ },
1511
+ "node_modules/postcss": {
1512
+ "version": "8.5.6",
1513
+ "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.6.tgz",
1514
+ "integrity": "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg==",
1515
+ "dev": true,
1516
+ "funding": [
1517
+ {
1518
+ "type": "opencollective",
1519
+ "url": "https://opencollective.com/postcss/"
1520
+ },
1521
+ {
1522
+ "type": "tidelift",
1523
+ "url": "https://tidelift.com/funding/github/npm/postcss"
1524
+ },
1525
+ {
1526
+ "type": "github",
1527
+ "url": "https://github.com/sponsors/ai"
1528
+ }
1529
+ ],
1530
+ "license": "MIT",
1531
+ "dependencies": {
1532
+ "nanoid": "^3.3.11",
1533
+ "picocolors": "^1.1.1",
1534
+ "source-map-js": "^1.2.1"
1535
+ },
1536
+ "engines": {
1537
+ "node": "^10 || ^12 || >=14"
1538
+ }
1539
+ },
1540
+ "node_modules/react": {
1541
+ "version": "19.2.4",
1542
+ "resolved": "https://registry.npmjs.org/react/-/react-19.2.4.tgz",
1543
+ "integrity": "sha512-9nfp2hYpCwOjAN+8TZFGhtWEwgvWHXqESH8qT89AT/lWklpLON22Lc8pEtnpsZz7VmawabSU0gCjnj8aC0euHQ==",
1544
+ "license": "MIT",
1545
+ "engines": {
1546
+ "node": ">=0.10.0"
1547
+ }
1548
+ },
1549
+ "node_modules/react-dom": {
1550
+ "version": "19.2.4",
1551
+ "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.4.tgz",
1552
+ "integrity": "sha512-AXJdLo8kgMbimY95O2aKQqsz2iWi9jMgKJhRBAxECE4IFxfcazB2LmzloIoibJI3C12IlY20+KFaLv+71bUJeQ==",
1553
+ "license": "MIT",
1554
+ "dependencies": {
1555
+ "scheduler": "^0.27.0"
1556
+ },
1557
+ "peerDependencies": {
1558
+ "react": "^19.2.4"
1559
+ }
1560
+ },
1561
+ "node_modules/react-refresh": {
1562
+ "version": "0.17.0",
1563
+ "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.17.0.tgz",
1564
+ "integrity": "sha512-z6F7K9bV85EfseRCp2bzrpyQ0Gkw1uLoCel9XBVWPg/TjRj94SkJzUTGfOa4bs7iJvBWtQG0Wq7wnI0syw3EBQ==",
1565
+ "dev": true,
1566
+ "license": "MIT",
1567
+ "engines": {
1568
+ "node": ">=0.10.0"
1569
+ }
1570
+ },
1571
+ "node_modules/rollup": {
1572
+ "version": "4.57.1",
1573
+ "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.57.1.tgz",
1574
+ "integrity": "sha512-oQL6lgK3e2QZeQ7gcgIkS2YZPg5slw37hYufJ3edKlfQSGGm8ICoxswK15ntSzF/a8+h7ekRy7k7oWc3BQ7y8A==",
1575
+ "dev": true,
1576
+ "license": "MIT",
1577
+ "dependencies": {
1578
+ "@types/estree": "1.0.8"
1579
+ },
1580
+ "bin": {
1581
+ "rollup": "dist/bin/rollup"
1582
+ },
1583
+ "engines": {
1584
+ "node": ">=18.0.0",
1585
+ "npm": ">=8.0.0"
1586
+ },
1587
+ "optionalDependencies": {
1588
+ "@rollup/rollup-android-arm-eabi": "4.57.1",
1589
+ "@rollup/rollup-android-arm64": "4.57.1",
1590
+ "@rollup/rollup-darwin-arm64": "4.57.1",
1591
+ "@rollup/rollup-darwin-x64": "4.57.1",
1592
+ "@rollup/rollup-freebsd-arm64": "4.57.1",
1593
+ "@rollup/rollup-freebsd-x64": "4.57.1",
1594
+ "@rollup/rollup-linux-arm-gnueabihf": "4.57.1",
1595
+ "@rollup/rollup-linux-arm-musleabihf": "4.57.1",
1596
+ "@rollup/rollup-linux-arm64-gnu": "4.57.1",
1597
+ "@rollup/rollup-linux-arm64-musl": "4.57.1",
1598
+ "@rollup/rollup-linux-loong64-gnu": "4.57.1",
1599
+ "@rollup/rollup-linux-loong64-musl": "4.57.1",
1600
+ "@rollup/rollup-linux-ppc64-gnu": "4.57.1",
1601
+ "@rollup/rollup-linux-ppc64-musl": "4.57.1",
1602
+ "@rollup/rollup-linux-riscv64-gnu": "4.57.1",
1603
+ "@rollup/rollup-linux-riscv64-musl": "4.57.1",
1604
+ "@rollup/rollup-linux-s390x-gnu": "4.57.1",
1605
+ "@rollup/rollup-linux-x64-gnu": "4.57.1",
1606
+ "@rollup/rollup-linux-x64-musl": "4.57.1",
1607
+ "@rollup/rollup-openbsd-x64": "4.57.1",
1608
+ "@rollup/rollup-openharmony-arm64": "4.57.1",
1609
+ "@rollup/rollup-win32-arm64-msvc": "4.57.1",
1610
+ "@rollup/rollup-win32-ia32-msvc": "4.57.1",
1611
+ "@rollup/rollup-win32-x64-gnu": "4.57.1",
1612
+ "@rollup/rollup-win32-x64-msvc": "4.57.1",
1613
+ "fsevents": "~2.3.2"
1614
+ }
1615
+ },
1616
+ "node_modules/scheduler": {
1617
+ "version": "0.27.0",
1618
+ "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz",
1619
+ "integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==",
1620
+ "license": "MIT"
1621
+ },
1622
+ "node_modules/semver": {
1623
+ "version": "6.3.1",
1624
+ "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
1625
+ "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==",
1626
+ "dev": true,
1627
+ "license": "ISC",
1628
+ "bin": {
1629
+ "semver": "bin/semver.js"
1630
+ }
1631
+ },
1632
+ "node_modules/source-map-js": {
1633
+ "version": "1.2.1",
1634
+ "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
1635
+ "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==",
1636
+ "dev": true,
1637
+ "license": "BSD-3-Clause",
1638
+ "engines": {
1639
+ "node": ">=0.10.0"
1640
+ }
1641
+ },
1642
+ "node_modules/tinyglobby": {
1643
+ "version": "0.2.15",
1644
+ "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.15.tgz",
1645
+ "integrity": "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ==",
1646
+ "dev": true,
1647
+ "license": "MIT",
1648
+ "dependencies": {
1649
+ "fdir": "^6.5.0",
1650
+ "picomatch": "^4.0.3"
1651
+ },
1652
+ "engines": {
1653
+ "node": ">=12.0.0"
1654
+ },
1655
+ "funding": {
1656
+ "url": "https://github.com/sponsors/SuperchupuDev"
1657
+ }
1658
+ },
1659
+ "node_modules/update-browserslist-db": {
1660
+ "version": "1.2.3",
1661
+ "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz",
1662
+ "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==",
1663
+ "dev": true,
1664
+ "funding": [
1665
+ {
1666
+ "type": "opencollective",
1667
+ "url": "https://opencollective.com/browserslist"
1668
+ },
1669
+ {
1670
+ "type": "tidelift",
1671
+ "url": "https://tidelift.com/funding/github/npm/browserslist"
1672
+ },
1673
+ {
1674
+ "type": "github",
1675
+ "url": "https://github.com/sponsors/ai"
1676
+ }
1677
+ ],
1678
+ "license": "MIT",
1679
+ "dependencies": {
1680
+ "escalade": "^3.2.0",
1681
+ "picocolors": "^1.1.1"
1682
+ },
1683
+ "bin": {
1684
+ "update-browserslist-db": "cli.js"
1685
+ },
1686
+ "peerDependencies": {
1687
+ "browserslist": ">= 4.21.0"
1688
+ }
1689
+ },
1690
+ "node_modules/vite": {
1691
+ "version": "6.4.1",
1692
+ "resolved": "https://registry.npmjs.org/vite/-/vite-6.4.1.tgz",
1693
+ "integrity": "sha512-+Oxm7q9hDoLMyJOYfUYBuHQo+dkAloi33apOPP56pzj+vsdJDzr+j1NISE5pyaAuKL4A3UD34qd0lx5+kfKp2g==",
1694
+ "dev": true,
1695
+ "license": "MIT",
1696
+ "dependencies": {
1697
+ "esbuild": "^0.25.0",
1698
+ "fdir": "^6.4.4",
1699
+ "picomatch": "^4.0.2",
1700
+ "postcss": "^8.5.3",
1701
+ "rollup": "^4.34.9",
1702
+ "tinyglobby": "^0.2.13"
1703
+ },
1704
+ "bin": {
1705
+ "vite": "bin/vite.js"
1706
+ },
1707
+ "engines": {
1708
+ "node": "^18.0.0 || ^20.0.0 || >=22.0.0"
1709
+ },
1710
+ "funding": {
1711
+ "url": "https://github.com/vitejs/vite?sponsor=1"
1712
+ },
1713
+ "optionalDependencies": {
1714
+ "fsevents": "~2.3.3"
1715
+ },
1716
+ "peerDependencies": {
1717
+ "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0",
1718
+ "jiti": ">=1.21.0",
1719
+ "less": "*",
1720
+ "lightningcss": "^1.21.0",
1721
+ "sass": "*",
1722
+ "sass-embedded": "*",
1723
+ "stylus": "*",
1724
+ "sugarss": "*",
1725
+ "terser": "^5.16.0",
1726
+ "tsx": "^4.8.1",
1727
+ "yaml": "^2.4.2"
1728
+ },
1729
+ "peerDependenciesMeta": {
1730
+ "@types/node": {
1731
+ "optional": true
1732
+ },
1733
+ "jiti": {
1734
+ "optional": true
1735
+ },
1736
+ "less": {
1737
+ "optional": true
1738
+ },
1739
+ "lightningcss": {
1740
+ "optional": true
1741
+ },
1742
+ "sass": {
1743
+ "optional": true
1744
+ },
1745
+ "sass-embedded": {
1746
+ "optional": true
1747
+ },
1748
+ "stylus": {
1749
+ "optional": true
1750
+ },
1751
+ "sugarss": {
1752
+ "optional": true
1753
+ },
1754
+ "terser": {
1755
+ "optional": true
1756
+ },
1757
+ "tsx": {
1758
+ "optional": true
1759
+ },
1760
+ "yaml": {
1761
+ "optional": true
1762
+ }
1763
+ }
1764
+ },
1765
+ "node_modules/yallist": {
1766
+ "version": "3.1.1",
1767
+ "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
1768
+ "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
1769
+ "dev": true,
1770
+ "license": "ISC"
1771
+ }
1772
+ }
1773
+ }
frontend/package.json ADDED
@@ -0,0 +1,19 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "interview-coach-frontend",
3
+ "version": "1.0.0",
4
+ "private": true,
5
+ "type": "module",
6
+ "scripts": {
7
+ "dev": "vite",
8
+ "build": "vite build",
9
+ "preview": "vite preview"
10
+ },
11
+ "dependencies": {
12
+ "react": "^19.0.0",
13
+ "react-dom": "^19.0.0"
14
+ },
15
+ "devDependencies": {
16
+ "@vitejs/plugin-react": "^4.3.4",
17
+ "vite": "^6.0.0"
18
+ }
19
+ }
frontend/src/App.jsx ADDED
@@ -0,0 +1,109 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import React, { useState } from 'react';
2
+ import Setup from './components/Setup';
3
+ import Interview from './components/Interview';
4
+ import Report from './components/Report';
5
+ import { startInterview, submitAnswer, getReport } from './utils/api';
6
+
7
+ export default function App() {
8
+ const [phase, setPhase] = useState('setup'); // setup | interview | report
9
+ const [loading, setLoading] = useState(false);
10
+ const [error, setError] = useState('');
11
+ const [sessionId, setSessionId] = useState(null);
12
+ const [currentQuestion, setCurrentQuestion] = useState(null);
13
+ const [progress, setProgress] = useState({ answered: 0, total: 0 });
14
+ const [lastEval, setLastEval] = useState(null);
15
+ const [reportData, setReportData] = useState(null);
16
+
17
+ const handleStart = async ({ role, difficulty, interviewType, questionCount }) => {
18
+ setLoading(true);
19
+ setError('');
20
+ try {
21
+ const data = await startInterview(role, difficulty, interviewType, questionCount);
22
+ setSessionId(data.sessionId);
23
+ setCurrentQuestion(data.currentQuestion);
24
+ setProgress({ answered: 0, total: data.totalQuestions });
25
+ setLastEval(null);
26
+ setPhase('interview');
27
+ } catch (err) {
28
+ setError(err.message);
29
+ }
30
+ setLoading(false);
31
+ };
32
+
33
+ const handleAnswer = async (answer) => {
34
+ setLoading(true);
35
+ setError('');
36
+ try {
37
+ const data = await submitAnswer(sessionId, answer);
38
+ setLastEval(data.evaluation);
39
+ setProgress(data.progress);
40
+
41
+ if (data.isComplete) {
42
+ // Small delay to show last evaluation before report
43
+ setTimeout(async () => {
44
+ try {
45
+ const report = await getReport(sessionId);
46
+ setReportData(report);
47
+ setPhase('report');
48
+ } catch (err) {
49
+ setError(err.message);
50
+ }
51
+ setLoading(false);
52
+ }, 3000);
53
+ } else {
54
+ setCurrentQuestion(data.nextQuestion);
55
+ setLoading(false);
56
+ }
57
+ } catch (err) {
58
+ setError(err.message);
59
+ setLoading(false);
60
+ }
61
+ };
62
+
63
+ const handleReset = () => {
64
+ setPhase('setup');
65
+ setSessionId(null);
66
+ setCurrentQuestion(null);
67
+ setLastEval(null);
68
+ setReportData(null);
69
+ setError('');
70
+ };
71
+
72
+ return (
73
+ <div className="app">
74
+ <header className="header">
75
+ <h1>AI Interview Coach</h1>
76
+ <p>Practice interviews. Get real feedback. Land the job.</p>
77
+ </header>
78
+
79
+ {error && (
80
+ <div style={{ background: 'rgba(225,112,85,0.1)', padding: '1rem', borderRadius: '10px', marginBottom: '1rem', textAlign: 'center', color: 'var(--danger)' }}>
81
+ {error}
82
+ </div>
83
+ )}
84
+
85
+ {phase === 'setup' && <Setup onStart={handleStart} loading={loading} />}
86
+
87
+ {phase === 'interview' && currentQuestion && (
88
+ <Interview
89
+ question={currentQuestion}
90
+ progress={progress}
91
+ onSubmit={handleAnswer}
92
+ evaluation={lastEval}
93
+ loading={loading}
94
+ />
95
+ )}
96
+
97
+ {phase === 'interview' && loading && !currentQuestion && (
98
+ <div className="loading">
99
+ <div className="spinner" />
100
+ <p>Generating your final report...</p>
101
+ </div>
102
+ )}
103
+
104
+ {phase === 'report' && reportData && (
105
+ <Report data={reportData} onReset={handleReset} />
106
+ )}
107
+ </div>
108
+ );
109
+ }
frontend/src/components/Interview.jsx ADDED
@@ -0,0 +1,85 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import React, { useState } from 'react';
2
+
3
+ function scoreClass(score) {
4
+ if (score >= 7) return 'good';
5
+ if (score >= 4) return 'ok';
6
+ return 'bad';
7
+ }
8
+
9
+ export default function Interview({ question, progress, onSubmit, evaluation, loading }) {
10
+ const [answer, setAnswer] = useState('');
11
+
12
+ const handleSubmit = () => {
13
+ if (answer.trim().length >= 10) {
14
+ onSubmit(answer.trim());
15
+ setAnswer('');
16
+ }
17
+ };
18
+
19
+ const pct = (progress.answered / progress.total) * 100;
20
+
21
+ return (
22
+ <div className="interview-panel">
23
+ <div className="progress-bar">
24
+ <span className="progress-text">Q{progress.answered + 1}/{progress.total}</span>
25
+ <div className="progress-track">
26
+ <div className="progress-fill" style={{ width: `${pct}%` }} />
27
+ </div>
28
+ </div>
29
+
30
+ <div className="question-card">
31
+ <div className="q-label">{question.category} Question</div>
32
+ <div className="q-text">{question.question}</div>
33
+ {question.hint && <div className="q-hint">Hint: {question.hint}</div>}
34
+ </div>
35
+
36
+ {evaluation && (
37
+ <div className="eval-card">
38
+ <div className={`eval-score ${scoreClass(evaluation.score)}`}>
39
+ {evaluation.score}/10
40
+ </div>
41
+ <p className="eval-feedback">{evaluation.feedback}</p>
42
+
43
+ {evaluation.strengths?.length > 0 && (
44
+ <div className="eval-section strengths">
45
+ <h4>Strengths</h4>
46
+ <ul>{evaluation.strengths.map((s, i) => <li key={i}>{s}</li>)}</ul>
47
+ </div>
48
+ )}
49
+
50
+ {evaluation.improvements?.length > 0 && (
51
+ <div className="eval-section improvements">
52
+ <h4>How to improve</h4>
53
+ <ul>{evaluation.improvements.map((s, i) => <li key={i}>{s}</li>)}</ul>
54
+ </div>
55
+ )}
56
+
57
+ {evaluation.sampleAnswer && (
58
+ <div className="sample-answer">
59
+ <h4>Strong answer example</h4>
60
+ <p>{evaluation.sampleAnswer}</p>
61
+ </div>
62
+ )}
63
+ </div>
64
+ )}
65
+
66
+ <div className="answer-area">
67
+ <textarea
68
+ placeholder="Type your answer here... (minimum 10 characters)"
69
+ value={answer}
70
+ onChange={(e) => setAnswer(e.target.value)}
71
+ disabled={loading}
72
+ />
73
+ <div className="char-count">{answer.length} characters</div>
74
+ </div>
75
+
76
+ <button
77
+ className="btn-primary"
78
+ onClick={handleSubmit}
79
+ disabled={answer.trim().length < 10 || loading}
80
+ >
81
+ {loading ? 'Evaluating...' : 'Submit Answer'}
82
+ </button>
83
+ </div>
84
+ );
85
+ }
frontend/src/components/Report.jsx ADDED
@@ -0,0 +1,66 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import React from 'react';
2
+
3
+ function scoreClass(score) {
4
+ if (score >= 70) return 'good';
5
+ if (score >= 40) return 'ok';
6
+ return 'bad';
7
+ }
8
+
9
+ export default function Report({ data, onReset }) {
10
+ const { report, details, meta } = data;
11
+
12
+ return (
13
+ <div className="report">
14
+ <div className="report-header">
15
+ <div className={`report-score ${scoreClass(report.overallScore)}`}>
16
+ {report.overallScore}
17
+ </div>
18
+ <div className="score-label" style={{ color: 'var(--text-dim)' }}>Overall Score</div>
19
+ <div className={`verdict ${report.verdict}`}>{report.verdict.replace('_', ' ')}</div>
20
+ <div className="readiness">{report.readinessLevel}</div>
21
+ </div>
22
+
23
+ <div className="report-card">
24
+ <h3>Summary</h3>
25
+ <p className="report-summary">{report.summary}</p>
26
+ </div>
27
+
28
+ <div className="report-card strengths">
29
+ <h3>Top Strengths</h3>
30
+ <ul>{report.topStrengths.map((s, i) => <li key={i}>{s}</li>)}</ul>
31
+ </div>
32
+
33
+ <div className="report-card improve">
34
+ <h3>Areas to Improve</h3>
35
+ <ul>{report.areasToImprove.map((a, i) => <li key={i}>{a}</li>)}</ul>
36
+ </div>
37
+
38
+ <div className="report-card">
39
+ <h3>Recommendation</h3>
40
+ <p className="report-summary">{report.recommendation}</p>
41
+ </div>
42
+
43
+ <div className="report-card">
44
+ <h3>Question-by-Question Breakdown</h3>
45
+ {details.map((d, i) => (
46
+ <div key={i} style={{ marginTop: '1rem', paddingTop: '1rem', borderTop: i > 0 ? '1px solid var(--border)' : 'none' }}>
47
+ <p style={{ fontWeight: 600, marginBottom: '0.3rem' }}>Q{i + 1}: {d.question}</p>
48
+ <p style={{ color: 'var(--text-dim)', fontSize: '0.9rem', marginBottom: '0.3rem' }}>
49
+ Your answer: {d.answer.substring(0, 150)}{d.answer.length > 150 ? '...' : ''}
50
+ </p>
51
+ <p style={{ fontSize: '0.9rem' }}>
52
+ Score: <span style={{ fontWeight: 700, color: d.score >= 7 ? 'var(--success)' : d.score >= 4 ? 'var(--warning)' : 'var(--danger)' }}>
53
+ {d.score}/10
54
+ </span>
55
+ {' - '}{d.feedback}
56
+ </p>
57
+ </div>
58
+ ))}
59
+ </div>
60
+
61
+ <button className="btn-secondary" onClick={onReset}>
62
+ Start New Interview
63
+ </button>
64
+ </div>
65
+ );
66
+ }
frontend/src/components/Setup.jsx ADDED
@@ -0,0 +1,63 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import React, { useState } from 'react';
2
+
3
+ export default function Setup({ onStart, loading }) {
4
+ const [role, setRole] = useState('');
5
+ const [difficulty, setDifficulty] = useState('medium');
6
+ const [interviewType, setInterviewType] = useState('mixed');
7
+ const [questionCount, setQuestionCount] = useState(5);
8
+
9
+ const handleStart = () => {
10
+ if (role.trim()) {
11
+ onStart({ role: role.trim(), difficulty, interviewType, questionCount });
12
+ }
13
+ };
14
+
15
+ return (
16
+ <div className="setup-card">
17
+ <div className="form-group">
18
+ <label>Target Role</label>
19
+ <input
20
+ type="text"
21
+ placeholder="e.g., Senior Software Engineer, Product Manager, Data Scientist..."
22
+ value={role}
23
+ onChange={(e) => setRole(e.target.value)}
24
+ />
25
+ </div>
26
+
27
+ <div className="form-row">
28
+ <div className="form-group">
29
+ <label>Difficulty</label>
30
+ <select value={difficulty} onChange={(e) => setDifficulty(e.target.value)}>
31
+ <option value="easy">Easy - Entry level</option>
32
+ <option value="medium">Medium - Mid level</option>
33
+ <option value="hard">Hard - Senior level</option>
34
+ </select>
35
+ </div>
36
+ <div className="form-group">
37
+ <label>Interview Type</label>
38
+ <select value={interviewType} onChange={(e) => setInterviewType(e.target.value)}>
39
+ <option value="mixed">Mixed</option>
40
+ <option value="behavioral">Behavioral</option>
41
+ <option value="technical">Technical</option>
42
+ </select>
43
+ </div>
44
+ </div>
45
+
46
+ <div className="form-group">
47
+ <label>Number of Questions ({questionCount})</label>
48
+ <input
49
+ type="range"
50
+ min="3"
51
+ max="10"
52
+ value={questionCount}
53
+ onChange={(e) => setQuestionCount(Number(e.target.value))}
54
+ style={{ width: '100%', accentColor: 'var(--accent)' }}
55
+ />
56
+ </div>
57
+
58
+ <button className="btn-primary" onClick={handleStart} disabled={!role.trim() || loading}>
59
+ {loading ? 'Preparing interview...' : 'Start Interview'}
60
+ </button>
61
+ </div>
62
+ );
63
+ }
frontend/src/main.jsx ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ import React from 'react';
2
+ import ReactDOM from 'react-dom/client';
3
+ import App from './App';
4
+ import './styles/global.css';
5
+
6
+ ReactDOM.createRoot(document.getElementById('root')).render(
7
+ <React.StrictMode>
8
+ <App />
9
+ </React.StrictMode>
10
+ );
frontend/src/styles/global.css ADDED
@@ -0,0 +1,262 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ * { margin: 0; padding: 0; box-sizing: border-box; }
2
+
3
+ :root {
4
+ --bg: #0a0a1a;
5
+ --bg-card: #131332;
6
+ --accent: #6c5ce7;
7
+ --accent-light: #a29bfe;
8
+ --success: #00b894;
9
+ --warning: #fdcb6e;
10
+ --danger: #e17055;
11
+ --text: #eee;
12
+ --text-dim: #999;
13
+ --border: #2d2d5e;
14
+ }
15
+
16
+ body {
17
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
18
+ background: var(--bg);
19
+ color: var(--text);
20
+ min-height: 100vh;
21
+ }
22
+
23
+ .app { max-width: 800px; margin: 0 auto; padding: 2rem 1.5rem; }
24
+
25
+ .header { text-align: center; margin-bottom: 2rem; }
26
+ .header h1 {
27
+ font-size: 2.2rem;
28
+ font-weight: 800;
29
+ background: linear-gradient(135deg, #6c5ce7, #a29bfe, #00b894);
30
+ -webkit-background-clip: text;
31
+ -webkit-text-fill-color: transparent;
32
+ background-clip: text;
33
+ }
34
+ .header p { color: var(--text-dim); margin-top: 0.4rem; }
35
+
36
+ /* Setup Form */
37
+ .setup-card {
38
+ background: var(--bg-card);
39
+ border-radius: 16px;
40
+ padding: 2rem;
41
+ }
42
+
43
+ .form-group { margin-bottom: 1.2rem; }
44
+ .form-group label {
45
+ display: block;
46
+ font-size: 0.85rem;
47
+ color: var(--text-dim);
48
+ margin-bottom: 0.4rem;
49
+ }
50
+ .form-group input, .form-group select {
51
+ width: 100%;
52
+ padding: 0.7rem 1rem;
53
+ background: var(--bg);
54
+ border: 1px solid var(--border);
55
+ border-radius: 10px;
56
+ color: var(--text);
57
+ font-size: 1rem;
58
+ }
59
+
60
+ .form-row { display: grid; grid-template-columns: 1fr 1fr; gap: 1rem; }
61
+
62
+ .btn-primary {
63
+ width: 100%;
64
+ padding: 1rem;
65
+ margin-top: 1rem;
66
+ background: linear-gradient(135deg, #6c5ce7, #a29bfe);
67
+ border: none;
68
+ border-radius: 12px;
69
+ color: white;
70
+ font-size: 1.1rem;
71
+ font-weight: 700;
72
+ cursor: pointer;
73
+ transition: transform 0.2s;
74
+ }
75
+ .btn-primary:hover { transform: translateY(-1px); }
76
+ .btn-primary:disabled { opacity: 0.5; cursor: not-allowed; transform: none; }
77
+
78
+ /* Interview Chat */
79
+ .interview-panel { }
80
+
81
+ .progress-bar {
82
+ display: flex;
83
+ align-items: center;
84
+ gap: 0.8rem;
85
+ margin-bottom: 1.5rem;
86
+ padding: 0.8rem 1rem;
87
+ background: var(--bg-card);
88
+ border-radius: 12px;
89
+ }
90
+ .progress-track {
91
+ flex: 1;
92
+ height: 8px;
93
+ background: var(--border);
94
+ border-radius: 4px;
95
+ overflow: hidden;
96
+ }
97
+ .progress-fill {
98
+ height: 100%;
99
+ background: linear-gradient(90deg, #6c5ce7, #00b894);
100
+ transition: width 0.5s ease;
101
+ border-radius: 4px;
102
+ }
103
+ .progress-text { font-size: 0.85rem; color: var(--text-dim); white-space: nowrap; }
104
+
105
+ .question-card {
106
+ background: var(--bg-card);
107
+ border-radius: 16px;
108
+ padding: 1.5rem;
109
+ margin-bottom: 1.5rem;
110
+ border-left: 4px solid var(--accent);
111
+ }
112
+ .question-card .q-label {
113
+ font-size: 0.8rem;
114
+ color: var(--accent-light);
115
+ text-transform: uppercase;
116
+ letter-spacing: 1px;
117
+ margin-bottom: 0.5rem;
118
+ }
119
+ .question-card .q-text { font-size: 1.15rem; line-height: 1.5; }
120
+ .question-card .q-hint {
121
+ margin-top: 0.8rem;
122
+ font-size: 0.85rem;
123
+ color: var(--text-dim);
124
+ font-style: italic;
125
+ }
126
+
127
+ .answer-area textarea {
128
+ width: 100%;
129
+ min-height: 150px;
130
+ padding: 1rem;
131
+ background: var(--bg-card);
132
+ border: 1px solid var(--border);
133
+ border-radius: 12px;
134
+ color: var(--text);
135
+ font-size: 1rem;
136
+ resize: vertical;
137
+ line-height: 1.5;
138
+ }
139
+ .answer-area textarea:focus { outline: none; border-color: var(--accent); }
140
+
141
+ .char-count {
142
+ text-align: right;
143
+ font-size: 0.8rem;
144
+ color: var(--text-dim);
145
+ margin-top: 0.3rem;
146
+ }
147
+
148
+ /* Evaluation */
149
+ .eval-card {
150
+ background: var(--bg-card);
151
+ border-radius: 16px;
152
+ padding: 1.5rem;
153
+ margin: 1rem 0;
154
+ animation: slideIn 0.4s ease;
155
+ }
156
+ @keyframes slideIn {
157
+ from { opacity: 0; transform: translateY(10px); }
158
+ to { opacity: 1; transform: translateY(0); }
159
+ }
160
+
161
+ .eval-score {
162
+ font-size: 2rem;
163
+ font-weight: 800;
164
+ text-align: center;
165
+ margin-bottom: 0.5rem;
166
+ }
167
+ .eval-score.good { color: var(--success); }
168
+ .eval-score.ok { color: var(--warning); }
169
+ .eval-score.bad { color: var(--danger); }
170
+
171
+ .eval-feedback { color: var(--text-dim); line-height: 1.5; margin-bottom: 1rem; text-align: center; }
172
+
173
+ .eval-section { margin-top: 0.8rem; }
174
+ .eval-section h4 { font-size: 0.9rem; margin-bottom: 0.4rem; }
175
+ .eval-section ul { list-style: none; padding: 0; }
176
+ .eval-section li { padding: 0.3rem 0; padding-left: 1.2rem; position: relative; color: var(--text-dim); font-size: 0.9rem; }
177
+ .eval-section li::before { position: absolute; left: 0; }
178
+ .eval-section.strengths li::before { content: "✅"; }
179
+ .eval-section.improvements li::before { content: "💡"; }
180
+
181
+ .sample-answer {
182
+ margin-top: 1rem;
183
+ padding: 1rem;
184
+ background: rgba(0, 184, 148, 0.08);
185
+ border-radius: 8px;
186
+ border-left: 3px solid var(--success);
187
+ }
188
+ .sample-answer h4 { color: var(--success); margin-bottom: 0.4rem; font-size: 0.9rem; }
189
+ .sample-answer p { color: var(--text-dim); font-size: 0.9rem; line-height: 1.5; }
190
+
191
+ /* Report */
192
+ .report { animation: slideIn 0.5s ease; }
193
+ .report-header {
194
+ text-align: center;
195
+ padding: 2rem;
196
+ background: var(--bg-card);
197
+ border-radius: 16px;
198
+ margin-bottom: 1.5rem;
199
+ }
200
+ .report-score { font-size: 4rem; font-weight: 800; }
201
+ .report-score.good { color: var(--success); }
202
+ .report-score.ok { color: var(--warning); }
203
+ .report-score.bad { color: var(--danger); }
204
+
205
+ .verdict {
206
+ display: inline-block;
207
+ padding: 0.3rem 1rem;
208
+ border-radius: 20px;
209
+ font-weight: 700;
210
+ font-size: 0.9rem;
211
+ margin-top: 0.5rem;
212
+ }
213
+ .verdict.STRONG_HIRE { background: rgba(0,184,148,0.2); color: var(--success); }
214
+ .verdict.HIRE { background: rgba(0,184,148,0.15); color: var(--success); }
215
+ .verdict.MAYBE { background: rgba(253,203,110,0.2); color: var(--warning); }
216
+ .verdict.NO_HIRE { background: rgba(225,112,85,0.2); color: var(--danger); }
217
+
218
+ .report-card {
219
+ background: var(--bg-card);
220
+ border-radius: 16px;
221
+ padding: 1.5rem;
222
+ margin-bottom: 1rem;
223
+ }
224
+ .report-card h3 { margin-bottom: 0.8rem; }
225
+ .report-card ul { list-style: none; padding: 0; }
226
+ .report-card li { padding: 0.4rem 0; padding-left: 1.5rem; position: relative; color: var(--text-dim); }
227
+ .report-card li::before { position: absolute; left: 0; }
228
+ .report-card.strengths li::before { content: "💪"; }
229
+ .report-card.improve li::before { content: "📈"; }
230
+
231
+ .report-summary { color: var(--text-dim); line-height: 1.6; margin-top: 0.5rem; }
232
+ .readiness { text-align: center; margin-top: 0.8rem; font-size: 1.1rem; }
233
+
234
+ .btn-secondary {
235
+ display: block;
236
+ margin: 2rem auto 0;
237
+ padding: 0.8rem 2rem;
238
+ background: transparent;
239
+ border: 2px solid var(--accent);
240
+ border-radius: 12px;
241
+ color: var(--accent);
242
+ font-size: 1rem;
243
+ cursor: pointer;
244
+ }
245
+ .btn-secondary:hover { background: var(--accent); color: white; }
246
+
247
+ /* Loading */
248
+ .loading { text-align: center; padding: 2rem; }
249
+ .spinner {
250
+ width: 40px; height: 40px;
251
+ border: 4px solid var(--border);
252
+ border-top-color: var(--accent);
253
+ border-radius: 50%;
254
+ animation: spin 1s linear infinite;
255
+ margin: 0 auto 1rem;
256
+ }
257
+ @keyframes spin { to { transform: rotate(360deg); } }
258
+
259
+ @media (max-width: 600px) {
260
+ .form-row { grid-template-columns: 1fr; }
261
+ .header h1 { font-size: 1.6rem; }
262
+ }
frontend/src/utils/api.js ADDED
@@ -0,0 +1,27 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ const API = '/api/interview';
2
+
3
+ export async function startInterview(role, difficulty, interviewType, questionCount) {
4
+ const res = await fetch(`${API}/start`, {
5
+ method: 'POST',
6
+ headers: { 'Content-Type': 'application/json' },
7
+ body: JSON.stringify({ role, difficulty, interviewType, questionCount }),
8
+ });
9
+ if (!res.ok) throw new Error((await res.json()).error);
10
+ return res.json();
11
+ }
12
+
13
+ export async function submitAnswer(sessionId, answer) {
14
+ const res = await fetch(`${API}/answer`, {
15
+ method: 'POST',
16
+ headers: { 'Content-Type': 'application/json' },
17
+ body: JSON.stringify({ sessionId, answer }),
18
+ });
19
+ if (!res.ok) throw new Error((await res.json()).error);
20
+ return res.json();
21
+ }
22
+
23
+ export async function getReport(sessionId) {
24
+ const res = await fetch(`${API}/report/${sessionId}`);
25
+ if (!res.ok) throw new Error((await res.json()).error);
26
+ return res.json();
27
+ }
frontend/vite.config.js ADDED
@@ -0,0 +1,12 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import { defineConfig } from 'vite';
2
+ import react from '@vitejs/plugin-react';
3
+
4
+ export default defineConfig({
5
+ plugins: [react()],
6
+ server: {
7
+ port: 3000,
8
+ proxy: {
9
+ '/api': 'http://localhost:3002'
10
+ }
11
+ }
12
+ });