lakshmisravya123 commited on
Commit
e1d7ef4
·
1 Parent(s): 8c172bd

Deploy Smart Document Search to HF Spaces

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