lakshmisravya123 commited on
Commit
e6b5231
·
1 Parent(s): 0825248

Deploy Negotiation Simulator app

Browse files
.gitignore ADDED
@@ -0,0 +1,4 @@
 
 
 
 
 
1
+ node_modules/
2
+ dist/
3
+ .env
4
+ .DS_Store
Dockerfile ADDED
@@ -0,0 +1,14 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ FROM node:20-slim
2
+ WORKDIR /app
3
+ COPY frontend/package*.json frontend/
4
+ RUN cd frontend && npm install
5
+ COPY frontend/ frontend/
6
+ RUN cd frontend && npm run build
7
+ COPY backend/package*.json backend/
8
+ RUN cd backend && npm install --production
9
+ COPY backend/ backend/
10
+ RUN cp -r frontend/dist backend/public
11
+ ENV PORT=7860
12
+ EXPOSE 7860
13
+ WORKDIR /app/backend
14
+ CMD ["node", "server.js"]
README.md CHANGED
@@ -1,10 +1,30 @@
1
  ---
2
- title: Negotiation Sim
3
- emoji: 🐨
4
  colorFrom: blue
5
- colorTo: gray
6
  sdk: docker
7
  pinned: false
8
  ---
9
 
10
- Check out the configuration reference at https://huggingface.co/docs/hub/spaces-config-reference
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
  ---
2
+ title: Negotiation Simulator
3
+ emoji: 💼
4
  colorFrom: blue
5
+ colorTo: indigo
6
  sdk: docker
7
  pinned: false
8
  ---
9
 
10
+ # Negotiation Simulator
11
+
12
+ Practice salary negotiations with AI playing the hiring manager. Get real-time feedback and a performance report.
13
+
14
+ ## Features
15
+ - **Real Tactics** - AI uses anchoring, urgency, questioning, and more
16
+ - **3 Difficulty Levels** - Easy, Medium, Hard
17
+ - **Chat Interface** - Real-time back-and-forth negotiation
18
+ - **Tactic Detection** - See what tactics the hiring manager uses
19
+ - **Performance Report** - Score, strengths, missed opportunities, pro tips
20
+
21
+ ## Quick Start
22
+ ```bash
23
+ cd backend && cp .env.example .env && npm install && npm start
24
+ cd frontend && npm install && npm run dev
25
+ ```
26
+
27
+ ## Tech Stack
28
+ - **Frontend**: React 19 + Vite
29
+ - **Backend**: Express.js
30
+ - **AI**: Groq (cloud) / Ollama (local) - both free
backend/.env.example ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
 
1
+ GROQ_API_KEY=your_groq_api_key_here
2
+ GROQ_MODEL=llama-3.3-70b-versatile
3
+ OLLAMA_URL=http://localhost:11434
4
+ OLLAMA_MODEL=llama3.2:3b
5
+ PORT=3008
backend/package-lock.json ADDED
@@ -0,0 +1,881 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "negotiation-sim-backend",
3
+ "version": "1.0.0",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "negotiation-sim-backend",
9
+ "version": "1.0.0",
10
+ "dependencies": {
11
+ "cors": "^2.8.5",
12
+ "dotenv": "^16.4.7",
13
+ "express": "^4.21.2",
14
+ "uuid": "^11.0.3"
15
+ }
16
+ },
17
+ "node_modules/accepts": {
18
+ "version": "1.3.8",
19
+ "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz",
20
+ "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==",
21
+ "license": "MIT",
22
+ "dependencies": {
23
+ "mime-types": "~2.1.34",
24
+ "negotiator": "0.6.3"
25
+ },
26
+ "engines": {
27
+ "node": ">= 0.6"
28
+ }
29
+ },
30
+ "node_modules/array-flatten": {
31
+ "version": "1.1.1",
32
+ "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
33
+ "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==",
34
+ "license": "MIT"
35
+ },
36
+ "node_modules/body-parser": {
37
+ "version": "1.20.4",
38
+ "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.4.tgz",
39
+ "integrity": "sha512-ZTgYYLMOXY9qKU/57FAo8F+HA2dGX7bqGc71txDRC1rS4frdFI5R7NhluHxH6M0YItAP0sHB4uqAOcYKxO6uGA==",
40
+ "license": "MIT",
41
+ "dependencies": {
42
+ "bytes": "~3.1.2",
43
+ "content-type": "~1.0.5",
44
+ "debug": "2.6.9",
45
+ "depd": "2.0.0",
46
+ "destroy": "~1.2.0",
47
+ "http-errors": "~2.0.1",
48
+ "iconv-lite": "~0.4.24",
49
+ "on-finished": "~2.4.1",
50
+ "qs": "~6.14.0",
51
+ "raw-body": "~2.5.3",
52
+ "type-is": "~1.6.18",
53
+ "unpipe": "~1.0.0"
54
+ },
55
+ "engines": {
56
+ "node": ">= 0.8",
57
+ "npm": "1.2.8000 || >= 1.4.16"
58
+ }
59
+ },
60
+ "node_modules/bytes": {
61
+ "version": "3.1.2",
62
+ "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz",
63
+ "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==",
64
+ "license": "MIT",
65
+ "engines": {
66
+ "node": ">= 0.8"
67
+ }
68
+ },
69
+ "node_modules/call-bind-apply-helpers": {
70
+ "version": "1.0.2",
71
+ "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz",
72
+ "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==",
73
+ "license": "MIT",
74
+ "dependencies": {
75
+ "es-errors": "^1.3.0",
76
+ "function-bind": "^1.1.2"
77
+ },
78
+ "engines": {
79
+ "node": ">= 0.4"
80
+ }
81
+ },
82
+ "node_modules/call-bound": {
83
+ "version": "1.0.4",
84
+ "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz",
85
+ "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==",
86
+ "license": "MIT",
87
+ "dependencies": {
88
+ "call-bind-apply-helpers": "^1.0.2",
89
+ "get-intrinsic": "^1.3.0"
90
+ },
91
+ "engines": {
92
+ "node": ">= 0.4"
93
+ },
94
+ "funding": {
95
+ "url": "https://github.com/sponsors/ljharb"
96
+ }
97
+ },
98
+ "node_modules/content-disposition": {
99
+ "version": "0.5.4",
100
+ "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz",
101
+ "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==",
102
+ "license": "MIT",
103
+ "dependencies": {
104
+ "safe-buffer": "5.2.1"
105
+ },
106
+ "engines": {
107
+ "node": ">= 0.6"
108
+ }
109
+ },
110
+ "node_modules/content-type": {
111
+ "version": "1.0.5",
112
+ "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz",
113
+ "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==",
114
+ "license": "MIT",
115
+ "engines": {
116
+ "node": ">= 0.6"
117
+ }
118
+ },
119
+ "node_modules/cookie": {
120
+ "version": "0.7.2",
121
+ "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.2.tgz",
122
+ "integrity": "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w==",
123
+ "license": "MIT",
124
+ "engines": {
125
+ "node": ">= 0.6"
126
+ }
127
+ },
128
+ "node_modules/cookie-signature": {
129
+ "version": "1.0.7",
130
+ "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.7.tgz",
131
+ "integrity": "sha512-NXdYc3dLr47pBkpUCHtKSwIOQXLVn8dZEuywboCOJY/osA0wFSLlSawr3KN8qXJEyX66FcONTH8EIlVuK0yyFA==",
132
+ "license": "MIT"
133
+ },
134
+ "node_modules/cors": {
135
+ "version": "2.8.6",
136
+ "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.6.tgz",
137
+ "integrity": "sha512-tJtZBBHA6vjIAaF6EnIaq6laBBP9aq/Y3ouVJjEfoHbRBcHBAHYcMh/w8LDrk2PvIMMq8gmopa5D4V8RmbrxGw==",
138
+ "license": "MIT",
139
+ "dependencies": {
140
+ "object-assign": "^4",
141
+ "vary": "^1"
142
+ },
143
+ "engines": {
144
+ "node": ">= 0.10"
145
+ },
146
+ "funding": {
147
+ "type": "opencollective",
148
+ "url": "https://opencollective.com/express"
149
+ }
150
+ },
151
+ "node_modules/debug": {
152
+ "version": "2.6.9",
153
+ "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
154
+ "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
155
+ "license": "MIT",
156
+ "dependencies": {
157
+ "ms": "2.0.0"
158
+ }
159
+ },
160
+ "node_modules/depd": {
161
+ "version": "2.0.0",
162
+ "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
163
+ "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==",
164
+ "license": "MIT",
165
+ "engines": {
166
+ "node": ">= 0.8"
167
+ }
168
+ },
169
+ "node_modules/destroy": {
170
+ "version": "1.2.0",
171
+ "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz",
172
+ "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==",
173
+ "license": "MIT",
174
+ "engines": {
175
+ "node": ">= 0.8",
176
+ "npm": "1.2.8000 || >= 1.4.16"
177
+ }
178
+ },
179
+ "node_modules/dotenv": {
180
+ "version": "16.6.1",
181
+ "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.6.1.tgz",
182
+ "integrity": "sha512-uBq4egWHTcTt33a72vpSG0z3HnPuIl6NqYcTrKEg2azoEyl2hpW0zqlxysq2pK9HlDIHyHyakeYaYnSAwd8bow==",
183
+ "license": "BSD-2-Clause",
184
+ "engines": {
185
+ "node": ">=12"
186
+ },
187
+ "funding": {
188
+ "url": "https://dotenvx.com"
189
+ }
190
+ },
191
+ "node_modules/dunder-proto": {
192
+ "version": "1.0.1",
193
+ "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
194
+ "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==",
195
+ "license": "MIT",
196
+ "dependencies": {
197
+ "call-bind-apply-helpers": "^1.0.1",
198
+ "es-errors": "^1.3.0",
199
+ "gopd": "^1.2.0"
200
+ },
201
+ "engines": {
202
+ "node": ">= 0.4"
203
+ }
204
+ },
205
+ "node_modules/ee-first": {
206
+ "version": "1.1.1",
207
+ "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
208
+ "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==",
209
+ "license": "MIT"
210
+ },
211
+ "node_modules/encodeurl": {
212
+ "version": "2.0.0",
213
+ "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz",
214
+ "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==",
215
+ "license": "MIT",
216
+ "engines": {
217
+ "node": ">= 0.8"
218
+ }
219
+ },
220
+ "node_modules/es-define-property": {
221
+ "version": "1.0.1",
222
+ "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
223
+ "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==",
224
+ "license": "MIT",
225
+ "engines": {
226
+ "node": ">= 0.4"
227
+ }
228
+ },
229
+ "node_modules/es-errors": {
230
+ "version": "1.3.0",
231
+ "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
232
+ "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
233
+ "license": "MIT",
234
+ "engines": {
235
+ "node": ">= 0.4"
236
+ }
237
+ },
238
+ "node_modules/es-object-atoms": {
239
+ "version": "1.1.1",
240
+ "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz",
241
+ "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==",
242
+ "license": "MIT",
243
+ "dependencies": {
244
+ "es-errors": "^1.3.0"
245
+ },
246
+ "engines": {
247
+ "node": ">= 0.4"
248
+ }
249
+ },
250
+ "node_modules/escape-html": {
251
+ "version": "1.0.3",
252
+ "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
253
+ "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==",
254
+ "license": "MIT"
255
+ },
256
+ "node_modules/etag": {
257
+ "version": "1.8.1",
258
+ "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
259
+ "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==",
260
+ "license": "MIT",
261
+ "engines": {
262
+ "node": ">= 0.6"
263
+ }
264
+ },
265
+ "node_modules/express": {
266
+ "version": "4.22.1",
267
+ "resolved": "https://registry.npmjs.org/express/-/express-4.22.1.tgz",
268
+ "integrity": "sha512-F2X8g9P1X7uCPZMA3MVf9wcTqlyNp7IhH5qPCI0izhaOIYXaW9L535tGA3qmjRzpH+bZczqq7hVKxTR4NWnu+g==",
269
+ "license": "MIT",
270
+ "dependencies": {
271
+ "accepts": "~1.3.8",
272
+ "array-flatten": "1.1.1",
273
+ "body-parser": "~1.20.3",
274
+ "content-disposition": "~0.5.4",
275
+ "content-type": "~1.0.4",
276
+ "cookie": "~0.7.1",
277
+ "cookie-signature": "~1.0.6",
278
+ "debug": "2.6.9",
279
+ "depd": "2.0.0",
280
+ "encodeurl": "~2.0.0",
281
+ "escape-html": "~1.0.3",
282
+ "etag": "~1.8.1",
283
+ "finalhandler": "~1.3.1",
284
+ "fresh": "~0.5.2",
285
+ "http-errors": "~2.0.0",
286
+ "merge-descriptors": "1.0.3",
287
+ "methods": "~1.1.2",
288
+ "on-finished": "~2.4.1",
289
+ "parseurl": "~1.3.3",
290
+ "path-to-regexp": "~0.1.12",
291
+ "proxy-addr": "~2.0.7",
292
+ "qs": "~6.14.0",
293
+ "range-parser": "~1.2.1",
294
+ "safe-buffer": "5.2.1",
295
+ "send": "~0.19.0",
296
+ "serve-static": "~1.16.2",
297
+ "setprototypeof": "1.2.0",
298
+ "statuses": "~2.0.1",
299
+ "type-is": "~1.6.18",
300
+ "utils-merge": "1.0.1",
301
+ "vary": "~1.1.2"
302
+ },
303
+ "engines": {
304
+ "node": ">= 0.10.0"
305
+ },
306
+ "funding": {
307
+ "type": "opencollective",
308
+ "url": "https://opencollective.com/express"
309
+ }
310
+ },
311
+ "node_modules/finalhandler": {
312
+ "version": "1.3.2",
313
+ "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.2.tgz",
314
+ "integrity": "sha512-aA4RyPcd3badbdABGDuTXCMTtOneUCAYH/gxoYRTZlIJdF0YPWuGqiAsIrhNnnqdXGswYk6dGujem4w80UJFhg==",
315
+ "license": "MIT",
316
+ "dependencies": {
317
+ "debug": "2.6.9",
318
+ "encodeurl": "~2.0.0",
319
+ "escape-html": "~1.0.3",
320
+ "on-finished": "~2.4.1",
321
+ "parseurl": "~1.3.3",
322
+ "statuses": "~2.0.2",
323
+ "unpipe": "~1.0.0"
324
+ },
325
+ "engines": {
326
+ "node": ">= 0.8"
327
+ }
328
+ },
329
+ "node_modules/forwarded": {
330
+ "version": "0.2.0",
331
+ "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
332
+ "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==",
333
+ "license": "MIT",
334
+ "engines": {
335
+ "node": ">= 0.6"
336
+ }
337
+ },
338
+ "node_modules/fresh": {
339
+ "version": "0.5.2",
340
+ "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
341
+ "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==",
342
+ "license": "MIT",
343
+ "engines": {
344
+ "node": ">= 0.6"
345
+ }
346
+ },
347
+ "node_modules/function-bind": {
348
+ "version": "1.1.2",
349
+ "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
350
+ "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
351
+ "license": "MIT",
352
+ "funding": {
353
+ "url": "https://github.com/sponsors/ljharb"
354
+ }
355
+ },
356
+ "node_modules/get-intrinsic": {
357
+ "version": "1.3.0",
358
+ "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
359
+ "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==",
360
+ "license": "MIT",
361
+ "dependencies": {
362
+ "call-bind-apply-helpers": "^1.0.2",
363
+ "es-define-property": "^1.0.1",
364
+ "es-errors": "^1.3.0",
365
+ "es-object-atoms": "^1.1.1",
366
+ "function-bind": "^1.1.2",
367
+ "get-proto": "^1.0.1",
368
+ "gopd": "^1.2.0",
369
+ "has-symbols": "^1.1.0",
370
+ "hasown": "^2.0.2",
371
+ "math-intrinsics": "^1.1.0"
372
+ },
373
+ "engines": {
374
+ "node": ">= 0.4"
375
+ },
376
+ "funding": {
377
+ "url": "https://github.com/sponsors/ljharb"
378
+ }
379
+ },
380
+ "node_modules/get-proto": {
381
+ "version": "1.0.1",
382
+ "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
383
+ "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==",
384
+ "license": "MIT",
385
+ "dependencies": {
386
+ "dunder-proto": "^1.0.1",
387
+ "es-object-atoms": "^1.0.0"
388
+ },
389
+ "engines": {
390
+ "node": ">= 0.4"
391
+ }
392
+ },
393
+ "node_modules/gopd": {
394
+ "version": "1.2.0",
395
+ "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz",
396
+ "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==",
397
+ "license": "MIT",
398
+ "engines": {
399
+ "node": ">= 0.4"
400
+ },
401
+ "funding": {
402
+ "url": "https://github.com/sponsors/ljharb"
403
+ }
404
+ },
405
+ "node_modules/has-symbols": {
406
+ "version": "1.1.0",
407
+ "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz",
408
+ "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==",
409
+ "license": "MIT",
410
+ "engines": {
411
+ "node": ">= 0.4"
412
+ },
413
+ "funding": {
414
+ "url": "https://github.com/sponsors/ljharb"
415
+ }
416
+ },
417
+ "node_modules/hasown": {
418
+ "version": "2.0.2",
419
+ "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
420
+ "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==",
421
+ "license": "MIT",
422
+ "dependencies": {
423
+ "function-bind": "^1.1.2"
424
+ },
425
+ "engines": {
426
+ "node": ">= 0.4"
427
+ }
428
+ },
429
+ "node_modules/http-errors": {
430
+ "version": "2.0.1",
431
+ "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.1.tgz",
432
+ "integrity": "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ==",
433
+ "license": "MIT",
434
+ "dependencies": {
435
+ "depd": "~2.0.0",
436
+ "inherits": "~2.0.4",
437
+ "setprototypeof": "~1.2.0",
438
+ "statuses": "~2.0.2",
439
+ "toidentifier": "~1.0.1"
440
+ },
441
+ "engines": {
442
+ "node": ">= 0.8"
443
+ },
444
+ "funding": {
445
+ "type": "opencollective",
446
+ "url": "https://opencollective.com/express"
447
+ }
448
+ },
449
+ "node_modules/iconv-lite": {
450
+ "version": "0.4.24",
451
+ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
452
+ "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
453
+ "license": "MIT",
454
+ "dependencies": {
455
+ "safer-buffer": ">= 2.1.2 < 3"
456
+ },
457
+ "engines": {
458
+ "node": ">=0.10.0"
459
+ }
460
+ },
461
+ "node_modules/inherits": {
462
+ "version": "2.0.4",
463
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
464
+ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
465
+ "license": "ISC"
466
+ },
467
+ "node_modules/ipaddr.js": {
468
+ "version": "1.9.1",
469
+ "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
470
+ "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==",
471
+ "license": "MIT",
472
+ "engines": {
473
+ "node": ">= 0.10"
474
+ }
475
+ },
476
+ "node_modules/math-intrinsics": {
477
+ "version": "1.1.0",
478
+ "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz",
479
+ "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==",
480
+ "license": "MIT",
481
+ "engines": {
482
+ "node": ">= 0.4"
483
+ }
484
+ },
485
+ "node_modules/media-typer": {
486
+ "version": "0.3.0",
487
+ "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
488
+ "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==",
489
+ "license": "MIT",
490
+ "engines": {
491
+ "node": ">= 0.6"
492
+ }
493
+ },
494
+ "node_modules/merge-descriptors": {
495
+ "version": "1.0.3",
496
+ "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz",
497
+ "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==",
498
+ "license": "MIT",
499
+ "funding": {
500
+ "url": "https://github.com/sponsors/sindresorhus"
501
+ }
502
+ },
503
+ "node_modules/methods": {
504
+ "version": "1.1.2",
505
+ "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
506
+ "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==",
507
+ "license": "MIT",
508
+ "engines": {
509
+ "node": ">= 0.6"
510
+ }
511
+ },
512
+ "node_modules/mime": {
513
+ "version": "1.6.0",
514
+ "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
515
+ "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==",
516
+ "license": "MIT",
517
+ "bin": {
518
+ "mime": "cli.js"
519
+ },
520
+ "engines": {
521
+ "node": ">=4"
522
+ }
523
+ },
524
+ "node_modules/mime-db": {
525
+ "version": "1.52.0",
526
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
527
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
528
+ "license": "MIT",
529
+ "engines": {
530
+ "node": ">= 0.6"
531
+ }
532
+ },
533
+ "node_modules/mime-types": {
534
+ "version": "2.1.35",
535
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
536
+ "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
537
+ "license": "MIT",
538
+ "dependencies": {
539
+ "mime-db": "1.52.0"
540
+ },
541
+ "engines": {
542
+ "node": ">= 0.6"
543
+ }
544
+ },
545
+ "node_modules/ms": {
546
+ "version": "2.0.0",
547
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
548
+ "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
549
+ "license": "MIT"
550
+ },
551
+ "node_modules/negotiator": {
552
+ "version": "0.6.3",
553
+ "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz",
554
+ "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==",
555
+ "license": "MIT",
556
+ "engines": {
557
+ "node": ">= 0.6"
558
+ }
559
+ },
560
+ "node_modules/object-assign": {
561
+ "version": "4.1.1",
562
+ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
563
+ "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
564
+ "license": "MIT",
565
+ "engines": {
566
+ "node": ">=0.10.0"
567
+ }
568
+ },
569
+ "node_modules/object-inspect": {
570
+ "version": "1.13.4",
571
+ "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz",
572
+ "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==",
573
+ "license": "MIT",
574
+ "engines": {
575
+ "node": ">= 0.4"
576
+ },
577
+ "funding": {
578
+ "url": "https://github.com/sponsors/ljharb"
579
+ }
580
+ },
581
+ "node_modules/on-finished": {
582
+ "version": "2.4.1",
583
+ "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz",
584
+ "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==",
585
+ "license": "MIT",
586
+ "dependencies": {
587
+ "ee-first": "1.1.1"
588
+ },
589
+ "engines": {
590
+ "node": ">= 0.8"
591
+ }
592
+ },
593
+ "node_modules/parseurl": {
594
+ "version": "1.3.3",
595
+ "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
596
+ "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==",
597
+ "license": "MIT",
598
+ "engines": {
599
+ "node": ">= 0.8"
600
+ }
601
+ },
602
+ "node_modules/path-to-regexp": {
603
+ "version": "0.1.12",
604
+ "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.12.tgz",
605
+ "integrity": "sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==",
606
+ "license": "MIT"
607
+ },
608
+ "node_modules/proxy-addr": {
609
+ "version": "2.0.7",
610
+ "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
611
+ "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
612
+ "license": "MIT",
613
+ "dependencies": {
614
+ "forwarded": "0.2.0",
615
+ "ipaddr.js": "1.9.1"
616
+ },
617
+ "engines": {
618
+ "node": ">= 0.10"
619
+ }
620
+ },
621
+ "node_modules/qs": {
622
+ "version": "6.14.2",
623
+ "resolved": "https://registry.npmjs.org/qs/-/qs-6.14.2.tgz",
624
+ "integrity": "sha512-V/yCWTTF7VJ9hIh18Ugr2zhJMP01MY7c5kh4J870L7imm6/DIzBsNLTXzMwUA3yZ5b/KBqLx8Kp3uRvd7xSe3Q==",
625
+ "license": "BSD-3-Clause",
626
+ "dependencies": {
627
+ "side-channel": "^1.1.0"
628
+ },
629
+ "engines": {
630
+ "node": ">=0.6"
631
+ },
632
+ "funding": {
633
+ "url": "https://github.com/sponsors/ljharb"
634
+ }
635
+ },
636
+ "node_modules/range-parser": {
637
+ "version": "1.2.1",
638
+ "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
639
+ "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
640
+ "license": "MIT",
641
+ "engines": {
642
+ "node": ">= 0.6"
643
+ }
644
+ },
645
+ "node_modules/raw-body": {
646
+ "version": "2.5.3",
647
+ "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.3.tgz",
648
+ "integrity": "sha512-s4VSOf6yN0rvbRZGxs8Om5CWj6seneMwK3oDb4lWDH0UPhWcxwOWw5+qk24bxq87szX1ydrwylIOp2uG1ojUpA==",
649
+ "license": "MIT",
650
+ "dependencies": {
651
+ "bytes": "~3.1.2",
652
+ "http-errors": "~2.0.1",
653
+ "iconv-lite": "~0.4.24",
654
+ "unpipe": "~1.0.0"
655
+ },
656
+ "engines": {
657
+ "node": ">= 0.8"
658
+ }
659
+ },
660
+ "node_modules/safe-buffer": {
661
+ "version": "5.2.1",
662
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
663
+ "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
664
+ "funding": [
665
+ {
666
+ "type": "github",
667
+ "url": "https://github.com/sponsors/feross"
668
+ },
669
+ {
670
+ "type": "patreon",
671
+ "url": "https://www.patreon.com/feross"
672
+ },
673
+ {
674
+ "type": "consulting",
675
+ "url": "https://feross.org/support"
676
+ }
677
+ ],
678
+ "license": "MIT"
679
+ },
680
+ "node_modules/safer-buffer": {
681
+ "version": "2.1.2",
682
+ "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
683
+ "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
684
+ "license": "MIT"
685
+ },
686
+ "node_modules/send": {
687
+ "version": "0.19.2",
688
+ "resolved": "https://registry.npmjs.org/send/-/send-0.19.2.tgz",
689
+ "integrity": "sha512-VMbMxbDeehAxpOtWJXlcUS5E8iXh6QmN+BkRX1GARS3wRaXEEgzCcB10gTQazO42tpNIya8xIyNx8fll1OFPrg==",
690
+ "license": "MIT",
691
+ "dependencies": {
692
+ "debug": "2.6.9",
693
+ "depd": "2.0.0",
694
+ "destroy": "1.2.0",
695
+ "encodeurl": "~2.0.0",
696
+ "escape-html": "~1.0.3",
697
+ "etag": "~1.8.1",
698
+ "fresh": "~0.5.2",
699
+ "http-errors": "~2.0.1",
700
+ "mime": "1.6.0",
701
+ "ms": "2.1.3",
702
+ "on-finished": "~2.4.1",
703
+ "range-parser": "~1.2.1",
704
+ "statuses": "~2.0.2"
705
+ },
706
+ "engines": {
707
+ "node": ">= 0.8.0"
708
+ }
709
+ },
710
+ "node_modules/send/node_modules/ms": {
711
+ "version": "2.1.3",
712
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
713
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
714
+ "license": "MIT"
715
+ },
716
+ "node_modules/serve-static": {
717
+ "version": "1.16.3",
718
+ "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.3.tgz",
719
+ "integrity": "sha512-x0RTqQel6g5SY7Lg6ZreMmsOzncHFU7nhnRWkKgWuMTu5NN0DR5oruckMqRvacAN9d5w6ARnRBXl9xhDCgfMeA==",
720
+ "license": "MIT",
721
+ "dependencies": {
722
+ "encodeurl": "~2.0.0",
723
+ "escape-html": "~1.0.3",
724
+ "parseurl": "~1.3.3",
725
+ "send": "~0.19.1"
726
+ },
727
+ "engines": {
728
+ "node": ">= 0.8.0"
729
+ }
730
+ },
731
+ "node_modules/setprototypeof": {
732
+ "version": "1.2.0",
733
+ "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
734
+ "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==",
735
+ "license": "ISC"
736
+ },
737
+ "node_modules/side-channel": {
738
+ "version": "1.1.0",
739
+ "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz",
740
+ "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==",
741
+ "license": "MIT",
742
+ "dependencies": {
743
+ "es-errors": "^1.3.0",
744
+ "object-inspect": "^1.13.3",
745
+ "side-channel-list": "^1.0.0",
746
+ "side-channel-map": "^1.0.1",
747
+ "side-channel-weakmap": "^1.0.2"
748
+ },
749
+ "engines": {
750
+ "node": ">= 0.4"
751
+ },
752
+ "funding": {
753
+ "url": "https://github.com/sponsors/ljharb"
754
+ }
755
+ },
756
+ "node_modules/side-channel-list": {
757
+ "version": "1.0.0",
758
+ "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz",
759
+ "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==",
760
+ "license": "MIT",
761
+ "dependencies": {
762
+ "es-errors": "^1.3.0",
763
+ "object-inspect": "^1.13.3"
764
+ },
765
+ "engines": {
766
+ "node": ">= 0.4"
767
+ },
768
+ "funding": {
769
+ "url": "https://github.com/sponsors/ljharb"
770
+ }
771
+ },
772
+ "node_modules/side-channel-map": {
773
+ "version": "1.0.1",
774
+ "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz",
775
+ "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==",
776
+ "license": "MIT",
777
+ "dependencies": {
778
+ "call-bound": "^1.0.2",
779
+ "es-errors": "^1.3.0",
780
+ "get-intrinsic": "^1.2.5",
781
+ "object-inspect": "^1.13.3"
782
+ },
783
+ "engines": {
784
+ "node": ">= 0.4"
785
+ },
786
+ "funding": {
787
+ "url": "https://github.com/sponsors/ljharb"
788
+ }
789
+ },
790
+ "node_modules/side-channel-weakmap": {
791
+ "version": "1.0.2",
792
+ "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz",
793
+ "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==",
794
+ "license": "MIT",
795
+ "dependencies": {
796
+ "call-bound": "^1.0.2",
797
+ "es-errors": "^1.3.0",
798
+ "get-intrinsic": "^1.2.5",
799
+ "object-inspect": "^1.13.3",
800
+ "side-channel-map": "^1.0.1"
801
+ },
802
+ "engines": {
803
+ "node": ">= 0.4"
804
+ },
805
+ "funding": {
806
+ "url": "https://github.com/sponsors/ljharb"
807
+ }
808
+ },
809
+ "node_modules/statuses": {
810
+ "version": "2.0.2",
811
+ "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.2.tgz",
812
+ "integrity": "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw==",
813
+ "license": "MIT",
814
+ "engines": {
815
+ "node": ">= 0.8"
816
+ }
817
+ },
818
+ "node_modules/toidentifier": {
819
+ "version": "1.0.1",
820
+ "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
821
+ "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==",
822
+ "license": "MIT",
823
+ "engines": {
824
+ "node": ">=0.6"
825
+ }
826
+ },
827
+ "node_modules/type-is": {
828
+ "version": "1.6.18",
829
+ "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
830
+ "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
831
+ "license": "MIT",
832
+ "dependencies": {
833
+ "media-typer": "0.3.0",
834
+ "mime-types": "~2.1.24"
835
+ },
836
+ "engines": {
837
+ "node": ">= 0.6"
838
+ }
839
+ },
840
+ "node_modules/unpipe": {
841
+ "version": "1.0.0",
842
+ "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
843
+ "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==",
844
+ "license": "MIT",
845
+ "engines": {
846
+ "node": ">= 0.8"
847
+ }
848
+ },
849
+ "node_modules/utils-merge": {
850
+ "version": "1.0.1",
851
+ "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
852
+ "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==",
853
+ "license": "MIT",
854
+ "engines": {
855
+ "node": ">= 0.4.0"
856
+ }
857
+ },
858
+ "node_modules/uuid": {
859
+ "version": "11.1.0",
860
+ "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.1.0.tgz",
861
+ "integrity": "sha512-0/A9rDy9P7cJ+8w1c9WD9V//9Wj15Ce2MPz8Ri6032usz+NfePxx5AcN3bN+r6ZL6jEo066/yNYB3tn4pQEx+A==",
862
+ "funding": [
863
+ "https://github.com/sponsors/broofa",
864
+ "https://github.com/sponsors/ctavan"
865
+ ],
866
+ "license": "MIT",
867
+ "bin": {
868
+ "uuid": "dist/esm/bin/uuid"
869
+ }
870
+ },
871
+ "node_modules/vary": {
872
+ "version": "1.1.2",
873
+ "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
874
+ "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==",
875
+ "license": "MIT",
876
+ "engines": {
877
+ "node": ">= 0.8"
878
+ }
879
+ }
880
+ }
881
+ }
backend/package.json ADDED
@@ -0,0 +1,7 @@
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "negotiation-sim-backend",
3
+ "version": "1.0.0",
4
+ "main": "server.js",
5
+ "scripts": { "start": "node server.js" },
6
+ "dependencies": { "cors": "^2.8.5", "dotenv": "^16.4.7", "express": "^4.21.2", "uuid": "^11.0.3" }
7
+ }
backend/routes/negotiate.js ADDED
@@ -0,0 +1,57 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const { v4: uuidv4 } = require('uuid');
4
+ const { startNegotiation, negotiationRound, generateReport } = require('../services/ai');
5
+
6
+ const sessions = new Map();
7
+
8
+ router.post('/start', async (req, res) => {
9
+ try {
10
+ const { role, company, currentSalary, targetSalary, difficulty } = req.body;
11
+ if (!role || !targetSalary) return res.status(400).json({ error: 'Role and target salary required' });
12
+
13
+ const result = await startNegotiation(role, company, currentSalary, targetSalary, difficulty || 'medium');
14
+ const sessionId = uuidv4();
15
+ const maxBudget = Math.round(targetSalary * 0.95);
16
+
17
+ sessions.set(sessionId, {
18
+ role, company: company || 'Tech Company', targetSalary, currentOffer: result.initialOffer,
19
+ maxBudget, difficulty: difficulty || 'medium', hiringManagerName: result.hiringManagerName,
20
+ history: [{ role: 'Hiring Manager', text: result.openingStatement + ` Our initial offer is $${result.initialOffer}.` }],
21
+ rounds: 1
22
+ });
23
+
24
+ res.json({ sessionId, ...result });
25
+ } catch (err) { res.status(500).json({ error: err.message }); }
26
+ });
27
+
28
+ router.post('/respond', async (req, res) => {
29
+ try {
30
+ const { sessionId, response } = req.body;
31
+ const session = sessions.get(sessionId);
32
+ if (!session) return res.status(404).json({ error: 'Session not found' });
33
+
34
+ session.history.push({ role: 'Candidate', text: response });
35
+ const result = await negotiationRound(session, response);
36
+ session.currentOffer = result.newOffer || session.currentOffer;
37
+ session.history.push({ role: 'Hiring Manager', text: result.response });
38
+ session.rounds++;
39
+
40
+ res.json({ ...result, currentOffer: session.currentOffer, round: session.rounds });
41
+ } catch (err) { res.status(500).json({ error: err.message }); }
42
+ });
43
+
44
+ router.post('/accept', async (req, res) => {
45
+ try {
46
+ const { sessionId } = req.body;
47
+ const session = sessions.get(sessionId);
48
+ if (!session) return res.status(404).json({ error: 'Session not found' });
49
+
50
+ session.finalOffer = session.currentOffer;
51
+ const report = await generateReport(session);
52
+ sessions.delete(sessionId);
53
+ res.json(report);
54
+ } catch (err) { res.status(500).json({ error: err.message }); }
55
+ });
56
+
57
+ module.exports = router;
backend/server.js ADDED
@@ -0,0 +1,17 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ const express = require('express');
2
+ const cors = require('cors');
3
+ const dotenv = require('dotenv');
4
+ const path = require('path');
5
+ const negotiateRoutes = require('./routes/negotiate');
6
+
7
+ dotenv.config();
8
+ const app = express();
9
+ const PORT = process.env.PORT || 3008;
10
+
11
+ app.use(cors());
12
+ app.use(express.json());
13
+ app.use('/api/negotiate', negotiateRoutes);
14
+ app.get('/api/health', (req, res) => res.json({ status: 'ok', service: 'negotiation-sim' }));
15
+ app.use(express.static(path.join(__dirname, 'public')));
16
+ app.get('*', (req, res) => res.sendFile(path.join(__dirname, 'public', 'index.html')));
17
+ app.listen(PORT, () => console.log(`Negotiation Simulator running on port ${PORT}`));
backend/services/ai.js ADDED
@@ -0,0 +1,92 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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.');
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
+ async function startNegotiation(role, company, currentSalary, targetSalary, difficulty) {
31
+ const diffStyle = { easy: 'Be somewhat flexible, offer small pushback', medium: 'Be professional but firm, use standard negotiation tactics', hard: 'Be very tough, use aggressive tactics like anchoring low, creating urgency, questioning worth' };
32
+ const text = await callAI(`You are a hiring manager at ${company || 'a tech company'} interviewing a candidate for a ${role} position.
33
+ DIFFICULTY: ${difficulty} - ${diffStyle[difficulty] || diffStyle.medium}
34
+
35
+ The candidate's current salary is $${currentSalary || 'unknown'} and they want $${targetSalary}.
36
+ Your initial budget is $${Math.round(targetSalary * 0.85)} but you can go up to $${Math.round(targetSalary * 0.95)} max.
37
+
38
+ Start the negotiation. Make an opening statement and initial offer. Return ONLY valid JSON:
39
+ {"hiringManagerName":"<a realistic name>","openingStatement":"<2-3 sentences opening the salary discussion>","initialOffer":${Math.round(targetSalary * 0.82)},"roundNumber":1}
40
+
41
+ Return ONLY JSON, no markdown.`);
42
+ return parseJSON(text);
43
+ }
44
+
45
+ async function negotiationRound(context, candidateResponse) {
46
+ const text = await callAI(`You are ${context.hiringManagerName}, a hiring manager negotiating salary.
47
+ CONTEXT: Role: ${context.role}, Company: ${context.company}
48
+ DIFFICULTY: ${context.difficulty}
49
+ BUDGET: Your max is $${context.maxBudget}. Current offer: $${context.currentOffer}.
50
+ HISTORY:
51
+ ${context.history.map(h => `${h.role}: ${h.text}`).join('\n')}
52
+
53
+ CANDIDATE JUST SAID: "${candidateResponse}"
54
+
55
+ Respond as the hiring manager. Use real negotiation tactics. Return ONLY valid JSON:
56
+ {"response":"<your 2-4 sentence response>","newOffer":${context.currentOffer},"tactic":"<name the tactic you used: anchoring|urgency|questioning|concession|silence|package-deal|final-offer>","isAccepting":false,"isFinalOffer":false}
57
+
58
+ Adjust newOffer based on the negotiation flow. Only go up incrementally. If candidate is very persuasive, concede a bit more. If weak argument, hold firm.
59
+ Return ONLY JSON, no markdown.`);
60
+ return parseJSON(text);
61
+ }
62
+
63
+ async function generateReport(context) {
64
+ const text = await callAI(`You are a negotiation coach analyzing a salary negotiation practice session.
65
+
66
+ ROLE: ${context.role} at ${context.company}
67
+ TARGET: $${context.targetSalary}
68
+ FINAL OFFER: $${context.finalOffer}
69
+ ROUNDS: ${context.rounds}
70
+
71
+ FULL TRANSCRIPT:
72
+ ${context.history.map(h => `${h.role}: ${h.text}`).join('\n')}
73
+
74
+ Return ONLY valid JSON:
75
+ {
76
+ "overallScore":<1-100>,
77
+ "finalSalary":${context.finalOffer},
78
+ "targetSalary":${context.targetSalary},
79
+ "percentOfTarget":${Math.round((context.finalOffer / context.targetSalary) * 100)},
80
+ "verdict":"<Master Negotiator|Strong Negotiator|Decent|Needs Practice|Pushover>",
81
+ "summary":"<3-4 sentence assessment>",
82
+ "tacticsUsed":["<tactic the candidate used>"],
83
+ "missedOpportunities":["<what they could have done better>"],
84
+ "strengths":["<s1>","<s2>"],
85
+ "improvements":["<i1>","<i2>","<i3>"],
86
+ "tips":["<tip1>","<tip2>","<tip3>"]
87
+ }
88
+ Return ONLY JSON.`);
89
+ return parseJSON(text);
90
+ }
91
+
92
+ module.exports = { startNegotiation, negotiationRound, generateReport };
frontend/index.html ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
 
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+ <head><meta charset="UTF-8" /><meta name="viewport" content="width=device-width, initial-scale=1.0" /><title>Negotiation Simulator</title></head>
4
+ <body><div id="root"></div><script type="module" src="/src/main.jsx"></script></body>
5
+ </html>
frontend/package-lock.json ADDED
@@ -0,0 +1,1773 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "negotiation-sim-frontend",
3
+ "version": "1.0.0",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "negotiation-sim-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.30001770",
1269
+ "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001770.tgz",
1270
+ "integrity": "sha512-x/2CLQ1jHENRbHg5PSId2sXq1CIO1CISvwWAj027ltMVG2UNgW+w9oH2+HzgEIRFembL8bUlXtfbBHR1fCg2xw==",
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,9 @@
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "negotiation-sim-frontend",
3
+ "version": "1.0.0",
4
+ "private": true,
5
+ "type": "module",
6
+ "scripts": { "dev": "vite", "build": "vite build", "preview": "vite preview" },
7
+ "dependencies": { "react": "^19.0.0", "react-dom": "^19.0.0" },
8
+ "devDependencies": { "@vitejs/plugin-react": "^4.3.4", "vite": "^6.0.0" }
9
+ }
frontend/src/App.jsx ADDED
@@ -0,0 +1,141 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import { useState, useRef, useEffect } from 'react';
2
+ import { startNegotiation, sendResponse, acceptOffer } from './utils/api';
3
+
4
+ export default function App() {
5
+ const [phase, setPhase] = useState('setup');
6
+ const [config, setConfig] = useState({ role: '', company: '', currentSalary: '', targetSalary: '', difficulty: 'medium' });
7
+ const [sessionId, setSessionId] = useState(null);
8
+ const [messages, setMessages] = useState([]);
9
+ const [currentOffer, setCurrentOffer] = useState(0);
10
+ const [response, setResponse] = useState('');
11
+ const [loading, setLoading] = useState(false);
12
+ const [report, setReport] = useState(null);
13
+ const [round, setRound] = useState(1);
14
+ const [tactic, setTactic] = useState('');
15
+ const [error, setError] = useState('');
16
+ const chatRef = useRef(null);
17
+
18
+ useEffect(() => { if (chatRef.current) chatRef.current.scrollTop = chatRef.current.scrollHeight; }, [messages]);
19
+
20
+ const handleStart = async () => {
21
+ if (!config.role || !config.targetSalary) { setError('Role and target salary are required'); return; }
22
+ setLoading(true); setError('');
23
+ try {
24
+ const data = await startNegotiation({ ...config, targetSalary: Number(config.targetSalary), currentSalary: Number(config.currentSalary) || undefined });
25
+ setSessionId(data.sessionId);
26
+ setCurrentOffer(data.initialOffer);
27
+ setMessages([{ role: 'Hiring Manager', text: data.openingStatement + ` Our initial offer is $${data.initialOffer?.toLocaleString()}.`, name: data.hiringManagerName }]);
28
+ setPhase('negotiation');
29
+ } catch (err) { setError(err.message); }
30
+ finally { setLoading(false); }
31
+ };
32
+
33
+ const handleRespond = async () => {
34
+ if (!response.trim()) return;
35
+ setMessages(prev => [...prev, { role: 'You', text: response }]);
36
+ const myResponse = response;
37
+ setResponse('');
38
+ setLoading(true);
39
+ try {
40
+ const data = await sendResponse(sessionId, myResponse);
41
+ setCurrentOffer(data.currentOffer);
42
+ setMessages(prev => [...prev, { role: 'Hiring Manager', text: data.response }]);
43
+ setRound(data.round);
44
+ if (data.tactic) setTactic(data.tactic);
45
+ } catch (err) { setError(err.message); }
46
+ finally { setLoading(false); }
47
+ };
48
+
49
+ const handleAccept = async () => {
50
+ setLoading(true);
51
+ try {
52
+ const data = await acceptOffer(sessionId);
53
+ setReport(data);
54
+ setPhase('report');
55
+ } catch (err) { setError(err.message); }
56
+ finally { setLoading(false); }
57
+ };
58
+
59
+ if (phase === 'report' && report) {
60
+ return (
61
+ <div className="app">
62
+ <h1><span>Negotiation Report</span></h1>
63
+ <div className="report">
64
+ <div className="report-header">
65
+ <div className="report-score">{report.overallScore}/100</div>
66
+ <div className="report-verdict">{report.verdict}</div>
67
+ </div>
68
+ <div className="salary-comparison">
69
+ <div className="salary-box"><div className="label">Target</div><div className="amount">${report.targetSalary?.toLocaleString()}</div></div>
70
+ <div className="salary-box"><div className="label">Got</div><div className="amount">${report.finalSalary?.toLocaleString()}</div></div>
71
+ </div>
72
+ <p style={{ textAlign: 'center', color: '#bbb', marginBottom: '1.5rem' }}>{report.summary}</p>
73
+ <div className="report-section"><h3>Strengths</h3><ul>{report.strengths?.map((s, i) => <li key={i}>{s}</li>)}</ul></div>
74
+ <div className="report-section"><h3>Improvements</h3><ul>{report.improvements?.map((s, i) => <li key={i}>{s}</li>)}</ul></div>
75
+ <div className="report-section"><h3>Missed Opportunities</h3><ul>{report.missedOpportunities?.map((s, i) => <li key={i}>{s}</li>)}</ul></div>
76
+ <div className="report-section"><h3>Pro Tips</h3><ul>{report.tips?.map((s, i) => <li key={i}>{s}</li>)}</ul></div>
77
+ <button className="btn btn-primary" onClick={() => { setPhase('setup'); setMessages([]); setReport(null); setRound(1); }}>Practice Again</button>
78
+ </div>
79
+ </div>
80
+ );
81
+ }
82
+
83
+ if (phase === 'negotiation') {
84
+ return (
85
+ <div className="app">
86
+ <h1><span>Negotiation Simulator</span></h1>
87
+ <div className="offer-display">
88
+ <div style={{ color: '#888', fontSize: '0.9rem' }}>Current Offer (Round {round})</div>
89
+ <div className="offer-amount">${currentOffer?.toLocaleString()}</div>
90
+ <div style={{ color: '#888', fontSize: '0.85rem' }}>Target: ${Number(config.targetSalary).toLocaleString()}</div>
91
+ {tactic && <div className="tactic-badge">Tactic: {tactic}</div>}
92
+ </div>
93
+ <div className="chat-container" ref={chatRef}>
94
+ {messages.map((m, i) => (
95
+ <div key={i} className={`message ${m.role === 'You' ? 'candidate' : 'manager'}`}>
96
+ <div className="role">{m.role}{m.name ? ` (${m.name})` : ''}</div>
97
+ <div className="text">{m.text}</div>
98
+ </div>
99
+ ))}
100
+ {loading && <div className="loading"><div className="spinner"></div>Thinking...</div>}
101
+ </div>
102
+ {error && <p style={{ color: '#ff6b6b', marginBottom: '0.5rem' }}>{error}</p>}
103
+ <div className="response-area">
104
+ <textarea placeholder="Your response..." value={response} onChange={e => setResponse(e.target.value)} onKeyDown={e => { if (e.key === 'Enter' && !e.shiftKey) { e.preventDefault(); handleRespond(); } }} disabled={loading} />
105
+ <button className="btn btn-primary btn-send" onClick={handleRespond} disabled={loading || !response.trim()}>Send</button>
106
+ </div>
107
+ <div className="action-row">
108
+ <button className="btn btn-accept" onClick={handleAccept} disabled={loading} style={{ flex: 1 }}>Accept Offer (${currentOffer?.toLocaleString()})</button>
109
+ <button className="btn-walk" onClick={handleAccept} disabled={loading}>Walk Away</button>
110
+ </div>
111
+ </div>
112
+ );
113
+ }
114
+
115
+ return (
116
+ <div className="app">
117
+ <h1><span>Negotiation Simulator</span></h1>
118
+ <p className="subtitle">Practice salary negotiations with AI. Get scored on your performance.</p>
119
+ {error && <p style={{ color: '#ff6b6b', textAlign: 'center', marginBottom: '1rem' }}>{error}</p>}
120
+ <div className="setup-form">
121
+ <div className="form-row">
122
+ <div className="input-group"><label>Job Role *</label><input placeholder="e.g., Senior Software Engineer" value={config.role} onChange={e => setConfig({ ...config, role: e.target.value })} /></div>
123
+ <div className="input-group"><label>Company (optional)</label><input placeholder="e.g., Google" value={config.company} onChange={e => setConfig({ ...config, company: e.target.value })} /></div>
124
+ </div>
125
+ <div className="form-row">
126
+ <div className="input-group"><label>Current Salary</label><input type="number" placeholder="e.g., 120000" value={config.currentSalary} onChange={e => setConfig({ ...config, currentSalary: e.target.value })} /></div>
127
+ <div className="input-group"><label>Target Salary *</label><input type="number" placeholder="e.g., 150000" value={config.targetSalary} onChange={e => setConfig({ ...config, targetSalary: e.target.value })} /></div>
128
+ </div>
129
+ <div className="input-group">
130
+ <label>Difficulty</label>
131
+ <select value={config.difficulty} onChange={e => setConfig({ ...config, difficulty: e.target.value })}>
132
+ <option value="easy">Easy - Flexible hiring manager</option>
133
+ <option value="medium">Medium - Standard tactics</option>
134
+ <option value="hard">Hard - Tough negotiator</option>
135
+ </select>
136
+ </div>
137
+ <button className="btn btn-primary" onClick={handleStart} disabled={loading}>{loading ? 'Starting...' : 'Start Negotiation'}</button>
138
+ </div>
139
+ </div>
140
+ );
141
+ }
frontend/src/main.jsx ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
 
1
+ import React from 'react';
2
+ import ReactDOM from 'react-dom/client';
3
+ import App from './App';
4
+ import './styles/global.css';
5
+ ReactDOM.createRoot(document.getElementById('root')).render(<App />);
frontend/src/styles/global.css ADDED
@@ -0,0 +1,46 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ * { margin: 0; padding: 0; box-sizing: border-box; }
2
+ body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif; background: #0f0f1a; color: #e0e0e0; min-height: 100vh; }
3
+ .app { max-width: 800px; margin: 0 auto; padding: 2rem 1rem; }
4
+ h1 { text-align: center; font-size: 2.5rem; margin-bottom: 0.5rem; }
5
+ h1 span { background: linear-gradient(135deg, #00b894, #00cec9); -webkit-background-clip: text; -webkit-text-fill-color: transparent; }
6
+ .subtitle { text-align: center; color: #888; margin-bottom: 2rem; }
7
+ .setup-form { background: #1a1a2e; border-radius: 12px; padding: 2rem; }
8
+ .form-row { display: grid; grid-template-columns: 1fr 1fr; gap: 1rem; margin-bottom: 1rem; }
9
+ .input-group { margin-bottom: 1rem; }
10
+ .input-group label { display: block; margin-bottom: 0.3rem; color: #aaa; font-size: 0.9rem; }
11
+ .input-group input, .input-group select { width: 100%; padding: 0.8rem; border: 1px solid #333; border-radius: 8px; background: #12121a; color: #e0e0e0; font-size: 1rem; }
12
+ .btn { width: 100%; padding: 1rem; border: none; border-radius: 8px; font-size: 1.1rem; font-weight: 600; cursor: pointer; transition: transform 0.2s; }
13
+ .btn-primary { background: linear-gradient(135deg, #00b894, #00cec9); color: #000; }
14
+ .btn:hover { transform: scale(1.02); }
15
+ .btn:disabled { opacity: 0.5; cursor: not-allowed; transform: none; }
16
+ .chat-container { background: #1a1a2e; border-radius: 12px; padding: 1.5rem; margin-bottom: 1rem; max-height: 400px; overflow-y: auto; }
17
+ .message { margin-bottom: 1rem; padding: 1rem; border-radius: 10px; }
18
+ .message.manager { background: #16213e; border-left: 3px solid #00cec9; }
19
+ .message.candidate { background: #1e1e30; border-left: 3px solid #00b894; }
20
+ .message .role { font-size: 0.8rem; color: #888; margin-bottom: 0.3rem; }
21
+ .message .text { line-height: 1.5; }
22
+ .offer-display { text-align: center; background: #16213e; border-radius: 12px; padding: 1.5rem; margin-bottom: 1rem; }
23
+ .offer-amount { font-size: 2.5rem; font-weight: 700; color: #00cec9; }
24
+ .tactic-badge { display: inline-block; padding: 0.2rem 0.8rem; border-radius: 15px; background: #2d1b69; color: #a29bfe; font-size: 0.8rem; margin-top: 0.5rem; }
25
+ .response-area { display: flex; gap: 0.5rem; margin-bottom: 1rem; }
26
+ .response-area textarea { flex: 1; padding: 0.8rem; border: 1px solid #333; border-radius: 8px; background: #12121a; color: #e0e0e0; font-size: 1rem; min-height: 60px; resize: none; }
27
+ .btn-send { padding: 0.8rem 1.5rem; width: auto; }
28
+ .btn-accept { background: #238636; color: #fff; margin-top: 0.5rem; }
29
+ .btn-walk { background: #da3633; color: #fff; margin-top: 0.5rem; width: auto; padding: 0.5rem 1rem; font-size: 0.9rem; }
30
+ .action-row { display: flex; gap: 0.5rem; }
31
+ .loading { text-align: center; padding: 1rem; color: #888; }
32
+ .spinner { width: 30px; height: 30px; border: 3px solid #333; border-top-color: #00cec9; border-radius: 50%; animation: spin 1s linear infinite; margin: 0 auto 0.5rem; }
33
+ @keyframes spin { to { transform: rotate(360deg); } }
34
+ .report { background: #1a1a2e; border-radius: 12px; padding: 2rem; }
35
+ .report-header { text-align: center; margin-bottom: 2rem; }
36
+ .report-score { font-size: 4rem; font-weight: 700; color: #00cec9; }
37
+ .report-verdict { font-size: 1.2rem; color: #ffd93d; }
38
+ .report-section { margin-bottom: 1.5rem; }
39
+ .report-section h3 { color: #00cec9; margin-bottom: 0.5rem; }
40
+ .report-section ul { padding-left: 1.2rem; }
41
+ .report-section li { margin-bottom: 0.3rem; color: #bbb; }
42
+ .salary-comparison { display: grid; grid-template-columns: 1fr 1fr; gap: 1rem; margin-bottom: 1.5rem; }
43
+ .salary-box { background: #12121a; border-radius: 8px; padding: 1rem; text-align: center; }
44
+ .salary-box .label { color: #888; font-size: 0.85rem; }
45
+ .salary-box .amount { font-size: 1.8rem; font-weight: 700; color: #00b894; }
46
+ @media (max-width: 768px) { .form-row, .salary-comparison { grid-template-columns: 1fr; } h1 { font-size: 1.8rem; } }
frontend/src/utils/api.js ADDED
@@ -0,0 +1,16 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ const API = '/api/negotiate';
2
+ export async function startNegotiation(data) {
3
+ const res = await fetch(`${API}/start`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(data) });
4
+ if (!res.ok) throw new Error((await res.json()).error);
5
+ return res.json();
6
+ }
7
+ export async function sendResponse(sessionId, response) {
8
+ const res = await fetch(`${API}/respond`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ sessionId, response }) });
9
+ if (!res.ok) throw new Error((await res.json()).error);
10
+ return res.json();
11
+ }
12
+ export async function acceptOffer(sessionId) {
13
+ const res = await fetch(`${API}/accept`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ sessionId }) });
14
+ if (!res.ok) throw new Error((await res.json()).error);
15
+ return res.json();
16
+ }
frontend/vite.config.js ADDED
@@ -0,0 +1,6 @@
 
 
 
 
 
 
 
1
+ import { defineConfig } from 'vite';
2
+ import react from '@vitejs/plugin-react';
3
+ export default defineConfig({
4
+ plugins: [react()],
5
+ server: { port: 5181, proxy: { '/api': 'http://localhost:3008' } }
6
+ });