├── .DS_Store
├── .env.template
├── .gitignore
├── LICENSE
├── README.md
├── docker-compose.yml
├── package-lock.json
├── package.json
├── public
├── .DS_Store
├── css
│ └── style.css
├── desk.html
├── index.html
├── js
│ ├── .DS_Store
│ ├── desk.js
│ ├── new-ticket.js
│ ├── public.js
│ └── socket-client.js
├── new-ticket.html
└── public.html
├── src
├── app.ts
├── config
│ ├── envs.ts
│ └── uuid.adapter.ts
├── domain
│ └── interfaces
│ │ └── ticket.ts
└── presentation
│ ├── routes.ts
│ ├── server.ts
│ ├── services
│ ├── ticket.service.ts
│ └── wss.service.ts
│ └── tickets
│ ├── controller.ts
│ └── routes.ts
└── tsconfig.json
/.DS_Store:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/Klerith/ws-rest-api/d626082f649a9d676c78fc2ad831ef6d342c2a33/.DS_Store
--------------------------------------------------------------------------------
/.env.template:
--------------------------------------------------------------------------------
1 |
2 |
3 | PORT=3000
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 | .env
7 |
8 | mongo/
9 | postgres/
10 | node_modules/
11 | dist/
12 | build/
13 | coverage/
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2023 Fernando Herrera
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Rest Project + TypeScript
2 |
3 | Este proyecto previamente inicializado tiene todo lo necesario para trabajar con TypeScript, Express y Rest.
4 |
5 | Cada paso de su configuración ya se ha realizado previamente en el curso, por lo que solo es necesario clonar el proyecto y comenzar a trabajar.
6 |
7 |
8 | ## Instalación
9 |
10 | 1. Clonar .env.template a .env y configurar las variables de entorno
11 | 2. Ejecutar `npm install` para instalar las dependencias
12 | 3. En caso de necesitar base de datos, configurar el docker-compose.yml y ejecutar `docker-compose up -d` para levantar los servicios deseados.
13 | 4. Ejecutar `npm run dev` para levantar el proyecto en modo desarrollo
14 |
15 |
--------------------------------------------------------------------------------
/docker-compose.yml:
--------------------------------------------------------------------------------
1 | version: '3.8'
2 |
3 |
4 | services:
5 |
6 | # mongo-db:
7 | # image: mongo:6.0.6
8 | # restart: always
9 | # environment:
10 | # MONGO_INITDB_ROOT_USERNAME: mongo-user
11 | # MONGO_INITDB_ROOT_PASSWORD: 123456
12 | # volumes:
13 | # - ./mongo:/data/db
14 | # ports:
15 | # - 27017:27017
16 |
17 | # postgres-db:
18 | # image: postgres:15.3
19 | # restart: always
20 | # environment:
21 | # POSTGRES_USER: postgres
22 | # POSTGRES_DB: my-db
23 | # POSTGRES_PASSWORD: 123456
24 | # volumes:
25 | # - ./postgres:/var/lib/postgresql/data
26 | # ports:
27 | # - 5432:5432
--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "08-user-store",
3 | "version": "1.0.0",
4 | "lockfileVersion": 3,
5 | "requires": true,
6 | "packages": {
7 | "": {
8 | "name": "08-user-store",
9 | "version": "1.0.0",
10 | "license": "ISC",
11 | "dependencies": {
12 | "dotenv": "^16.3.1",
13 | "env-var": "^7.4.1",
14 | "express": "^4.18.2",
15 | "uuid": "^9.0.1",
16 | "ws": "^8.14.2"
17 | },
18 | "devDependencies": {
19 | "@types/express": "^4.17.18",
20 | "@types/node": "^20.8.0",
21 | "@types/uuid": "^9.0.4",
22 | "@types/ws": "^8.5.6",
23 | "rimraf": "^5.0.5",
24 | "ts-node-dev": "^2.0.0",
25 | "typescript": "^5.2.2"
26 | }
27 | },
28 | "node_modules/@cspotcode/source-map-support": {
29 | "version": "0.8.1",
30 | "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz",
31 | "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==",
32 | "dev": true,
33 | "dependencies": {
34 | "@jridgewell/trace-mapping": "0.3.9"
35 | },
36 | "engines": {
37 | "node": ">=12"
38 | }
39 | },
40 | "node_modules/@isaacs/cliui": {
41 | "version": "8.0.2",
42 | "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz",
43 | "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==",
44 | "dev": true,
45 | "dependencies": {
46 | "string-width": "^5.1.2",
47 | "string-width-cjs": "npm:string-width@^4.2.0",
48 | "strip-ansi": "^7.0.1",
49 | "strip-ansi-cjs": "npm:strip-ansi@^6.0.1",
50 | "wrap-ansi": "^8.1.0",
51 | "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0"
52 | },
53 | "engines": {
54 | "node": ">=12"
55 | }
56 | },
57 | "node_modules/@jridgewell/resolve-uri": {
58 | "version": "3.1.1",
59 | "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.1.tgz",
60 | "integrity": "sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA==",
61 | "dev": true,
62 | "engines": {
63 | "node": ">=6.0.0"
64 | }
65 | },
66 | "node_modules/@jridgewell/sourcemap-codec": {
67 | "version": "1.4.15",
68 | "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz",
69 | "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==",
70 | "dev": true
71 | },
72 | "node_modules/@jridgewell/trace-mapping": {
73 | "version": "0.3.9",
74 | "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz",
75 | "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==",
76 | "dev": true,
77 | "dependencies": {
78 | "@jridgewell/resolve-uri": "^3.0.3",
79 | "@jridgewell/sourcemap-codec": "^1.4.10"
80 | }
81 | },
82 | "node_modules/@pkgjs/parseargs": {
83 | "version": "0.11.0",
84 | "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz",
85 | "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==",
86 | "dev": true,
87 | "optional": true,
88 | "engines": {
89 | "node": ">=14"
90 | }
91 | },
92 | "node_modules/@tsconfig/node10": {
93 | "version": "1.0.9",
94 | "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.9.tgz",
95 | "integrity": "sha512-jNsYVVxU8v5g43Erja32laIDHXeoNvFEpX33OK4d6hljo3jDhCBDhx5dhCCTMWUojscpAagGiRkBKxpdl9fxqA==",
96 | "dev": true
97 | },
98 | "node_modules/@tsconfig/node12": {
99 | "version": "1.0.11",
100 | "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.11.tgz",
101 | "integrity": "sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==",
102 | "dev": true
103 | },
104 | "node_modules/@tsconfig/node14": {
105 | "version": "1.0.3",
106 | "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.3.tgz",
107 | "integrity": "sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==",
108 | "dev": true
109 | },
110 | "node_modules/@tsconfig/node16": {
111 | "version": "1.0.4",
112 | "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.4.tgz",
113 | "integrity": "sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==",
114 | "dev": true
115 | },
116 | "node_modules/@types/body-parser": {
117 | "version": "1.19.3",
118 | "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.3.tgz",
119 | "integrity": "sha512-oyl4jvAfTGX9Bt6Or4H9ni1Z447/tQuxnZsytsCaExKlmJiU8sFgnIBRzJUpKwB5eWn9HuBYlUlVA74q/yN0eQ==",
120 | "dev": true,
121 | "dependencies": {
122 | "@types/connect": "*",
123 | "@types/node": "*"
124 | }
125 | },
126 | "node_modules/@types/connect": {
127 | "version": "3.4.36",
128 | "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.36.tgz",
129 | "integrity": "sha512-P63Zd/JUGq+PdrM1lv0Wv5SBYeA2+CORvbrXbngriYY0jzLUWfQMQQxOhjONEz/wlHOAxOdY7CY65rgQdTjq2w==",
130 | "dev": true,
131 | "dependencies": {
132 | "@types/node": "*"
133 | }
134 | },
135 | "node_modules/@types/express": {
136 | "version": "4.17.18",
137 | "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.18.tgz",
138 | "integrity": "sha512-Sxv8BSLLgsBYmcnGdGjjEjqET2U+AKAdCRODmMiq02FgjwuV75Ut85DRpvFjyw/Mk0vgUOliGRU0UUmuuZHByQ==",
139 | "dev": true,
140 | "dependencies": {
141 | "@types/body-parser": "*",
142 | "@types/express-serve-static-core": "^4.17.33",
143 | "@types/qs": "*",
144 | "@types/serve-static": "*"
145 | }
146 | },
147 | "node_modules/@types/express-serve-static-core": {
148 | "version": "4.17.37",
149 | "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.37.tgz",
150 | "integrity": "sha512-ZohaCYTgGFcOP7u6aJOhY9uIZQgZ2vxC2yWoArY+FeDXlqeH66ZVBjgvg+RLVAS/DWNq4Ap9ZXu1+SUQiiWYMg==",
151 | "dev": true,
152 | "dependencies": {
153 | "@types/node": "*",
154 | "@types/qs": "*",
155 | "@types/range-parser": "*",
156 | "@types/send": "*"
157 | }
158 | },
159 | "node_modules/@types/http-errors": {
160 | "version": "2.0.2",
161 | "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.2.tgz",
162 | "integrity": "sha512-lPG6KlZs88gef6aD85z3HNkztpj7w2R7HmR3gygjfXCQmsLloWNARFkMuzKiiY8FGdh1XDpgBdrSf4aKDiA7Kg==",
163 | "dev": true
164 | },
165 | "node_modules/@types/mime": {
166 | "version": "1.3.3",
167 | "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.3.tgz",
168 | "integrity": "sha512-Ys+/St+2VF4+xuY6+kDIXGxbNRO0mesVg0bbxEfB97Od1Vjpjx9KD1qxs64Gcb3CWPirk9Xe+PT4YiiHQ9T+eg==",
169 | "dev": true
170 | },
171 | "node_modules/@types/node": {
172 | "version": "20.8.0",
173 | "resolved": "https://registry.npmjs.org/@types/node/-/node-20.8.0.tgz",
174 | "integrity": "sha512-LzcWltT83s1bthcvjBmiBvGJiiUe84NWRHkw+ZV6Fr41z2FbIzvc815dk2nQ3RAKMuN2fkenM/z3Xv2QzEpYxQ==",
175 | "dev": true
176 | },
177 | "node_modules/@types/qs": {
178 | "version": "6.9.8",
179 | "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.8.tgz",
180 | "integrity": "sha512-u95svzDlTysU5xecFNTgfFG5RUWu1A9P0VzgpcIiGZA9iraHOdSzcxMxQ55DyeRaGCSxQi7LxXDI4rzq/MYfdg==",
181 | "dev": true
182 | },
183 | "node_modules/@types/range-parser": {
184 | "version": "1.2.5",
185 | "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.5.tgz",
186 | "integrity": "sha512-xrO9OoVPqFuYyR/loIHjnbvvyRZREYKLjxV4+dY6v3FQR3stQ9ZxIGkaclF7YhI9hfjpuTbu14hZEy94qKLtOA==",
187 | "dev": true
188 | },
189 | "node_modules/@types/send": {
190 | "version": "0.17.2",
191 | "resolved": "https://registry.npmjs.org/@types/send/-/send-0.17.2.tgz",
192 | "integrity": "sha512-aAG6yRf6r0wQ29bkS+x97BIs64ZLxeE/ARwyS6wrldMm3C1MdKwCcnnEwMC1slI8wuxJOpiUH9MioC0A0i+GJw==",
193 | "dev": true,
194 | "dependencies": {
195 | "@types/mime": "^1",
196 | "@types/node": "*"
197 | }
198 | },
199 | "node_modules/@types/serve-static": {
200 | "version": "1.15.3",
201 | "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.3.tgz",
202 | "integrity": "sha512-yVRvFsEMrv7s0lGhzrggJjNOSmZCdgCjw9xWrPr/kNNLp6FaDfMC1KaYl3TSJ0c58bECwNBMoQrZJ8hA8E1eFg==",
203 | "dev": true,
204 | "dependencies": {
205 | "@types/http-errors": "*",
206 | "@types/mime": "*",
207 | "@types/node": "*"
208 | }
209 | },
210 | "node_modules/@types/strip-bom": {
211 | "version": "3.0.0",
212 | "resolved": "https://registry.npmjs.org/@types/strip-bom/-/strip-bom-3.0.0.tgz",
213 | "integrity": "sha512-xevGOReSYGM7g/kUBZzPqCrR/KYAo+F0yiPc85WFTJa0MSLtyFTVTU6cJu/aV4mid7IffDIWqo69THF2o4JiEQ==",
214 | "dev": true
215 | },
216 | "node_modules/@types/strip-json-comments": {
217 | "version": "0.0.30",
218 | "resolved": "https://registry.npmjs.org/@types/strip-json-comments/-/strip-json-comments-0.0.30.tgz",
219 | "integrity": "sha512-7NQmHra/JILCd1QqpSzl8+mJRc8ZHz3uDm8YV1Ks9IhK0epEiTw8aIErbvH9PI+6XbqhyIQy3462nEsn7UVzjQ==",
220 | "dev": true
221 | },
222 | "node_modules/@types/uuid": {
223 | "version": "9.0.4",
224 | "resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-9.0.4.tgz",
225 | "integrity": "sha512-zAuJWQflfx6dYJM62vna+Sn5aeSWhh3OB+wfUEACNcqUSc0AGc5JKl+ycL1vrH7frGTXhJchYjE1Hak8L819dA==",
226 | "dev": true
227 | },
228 | "node_modules/@types/ws": {
229 | "version": "8.5.6",
230 | "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.6.tgz",
231 | "integrity": "sha512-8B5EO9jLVCy+B58PLHvLDuOD8DRVMgQzq8d55SjLCOn9kqGyqOvy27exVaTio1q1nX5zLu8/6N0n2ThSxOM6tg==",
232 | "dev": true,
233 | "dependencies": {
234 | "@types/node": "*"
235 | }
236 | },
237 | "node_modules/accepts": {
238 | "version": "1.3.8",
239 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz",
240 | "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==",
241 | "dependencies": {
242 | "mime-types": "~2.1.34",
243 | "negotiator": "0.6.3"
244 | },
245 | "engines": {
246 | "node": ">= 0.6"
247 | }
248 | },
249 | "node_modules/acorn": {
250 | "version": "8.10.0",
251 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.10.0.tgz",
252 | "integrity": "sha512-F0SAmZ8iUtS//m8DmCTA0jlh6TDKkHQyK6xc6V4KDTyZKA9dnvX9/3sRTVQrWm79glUAZbnmmNcdYwUIHWVybw==",
253 | "dev": true,
254 | "bin": {
255 | "acorn": "bin/acorn"
256 | },
257 | "engines": {
258 | "node": ">=0.4.0"
259 | }
260 | },
261 | "node_modules/acorn-walk": {
262 | "version": "8.2.0",
263 | "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.2.0.tgz",
264 | "integrity": "sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==",
265 | "dev": true,
266 | "engines": {
267 | "node": ">=0.4.0"
268 | }
269 | },
270 | "node_modules/ansi-regex": {
271 | "version": "6.0.1",
272 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.0.1.tgz",
273 | "integrity": "sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==",
274 | "dev": true,
275 | "engines": {
276 | "node": ">=12"
277 | },
278 | "funding": {
279 | "url": "https://github.com/chalk/ansi-regex?sponsor=1"
280 | }
281 | },
282 | "node_modules/ansi-styles": {
283 | "version": "6.2.1",
284 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.1.tgz",
285 | "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==",
286 | "dev": true,
287 | "engines": {
288 | "node": ">=12"
289 | },
290 | "funding": {
291 | "url": "https://github.com/chalk/ansi-styles?sponsor=1"
292 | }
293 | },
294 | "node_modules/anymatch": {
295 | "version": "3.1.3",
296 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz",
297 | "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==",
298 | "dev": true,
299 | "dependencies": {
300 | "normalize-path": "^3.0.0",
301 | "picomatch": "^2.0.4"
302 | },
303 | "engines": {
304 | "node": ">= 8"
305 | }
306 | },
307 | "node_modules/arg": {
308 | "version": "4.1.3",
309 | "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz",
310 | "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==",
311 | "dev": true
312 | },
313 | "node_modules/array-flatten": {
314 | "version": "1.1.1",
315 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
316 | "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg=="
317 | },
318 | "node_modules/balanced-match": {
319 | "version": "1.0.2",
320 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
321 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
322 | "dev": true
323 | },
324 | "node_modules/binary-extensions": {
325 | "version": "2.2.0",
326 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz",
327 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==",
328 | "dev": true,
329 | "engines": {
330 | "node": ">=8"
331 | }
332 | },
333 | "node_modules/body-parser": {
334 | "version": "1.20.1",
335 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.1.tgz",
336 | "integrity": "sha512-jWi7abTbYwajOytWCQc37VulmWiRae5RyTpaCyDcS5/lMdtwSz5lOpDE67srw/HYe35f1z3fDQw+3txg7gNtWw==",
337 | "dependencies": {
338 | "bytes": "3.1.2",
339 | "content-type": "~1.0.4",
340 | "debug": "2.6.9",
341 | "depd": "2.0.0",
342 | "destroy": "1.2.0",
343 | "http-errors": "2.0.0",
344 | "iconv-lite": "0.4.24",
345 | "on-finished": "2.4.1",
346 | "qs": "6.11.0",
347 | "raw-body": "2.5.1",
348 | "type-is": "~1.6.18",
349 | "unpipe": "1.0.0"
350 | },
351 | "engines": {
352 | "node": ">= 0.8",
353 | "npm": "1.2.8000 || >= 1.4.16"
354 | }
355 | },
356 | "node_modules/brace-expansion": {
357 | "version": "2.0.1",
358 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz",
359 | "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==",
360 | "dev": true,
361 | "dependencies": {
362 | "balanced-match": "^1.0.0"
363 | }
364 | },
365 | "node_modules/braces": {
366 | "version": "3.0.2",
367 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
368 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
369 | "dev": true,
370 | "dependencies": {
371 | "fill-range": "^7.0.1"
372 | },
373 | "engines": {
374 | "node": ">=8"
375 | }
376 | },
377 | "node_modules/buffer-from": {
378 | "version": "1.1.2",
379 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz",
380 | "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==",
381 | "dev": true
382 | },
383 | "node_modules/bytes": {
384 | "version": "3.1.2",
385 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz",
386 | "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==",
387 | "engines": {
388 | "node": ">= 0.8"
389 | }
390 | },
391 | "node_modules/call-bind": {
392 | "version": "1.0.2",
393 | "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz",
394 | "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==",
395 | "dependencies": {
396 | "function-bind": "^1.1.1",
397 | "get-intrinsic": "^1.0.2"
398 | },
399 | "funding": {
400 | "url": "https://github.com/sponsors/ljharb"
401 | }
402 | },
403 | "node_modules/chokidar": {
404 | "version": "3.5.3",
405 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz",
406 | "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==",
407 | "dev": true,
408 | "funding": [
409 | {
410 | "type": "individual",
411 | "url": "https://paulmillr.com/funding/"
412 | }
413 | ],
414 | "dependencies": {
415 | "anymatch": "~3.1.2",
416 | "braces": "~3.0.2",
417 | "glob-parent": "~5.1.2",
418 | "is-binary-path": "~2.1.0",
419 | "is-glob": "~4.0.1",
420 | "normalize-path": "~3.0.0",
421 | "readdirp": "~3.6.0"
422 | },
423 | "engines": {
424 | "node": ">= 8.10.0"
425 | },
426 | "optionalDependencies": {
427 | "fsevents": "~2.3.2"
428 | }
429 | },
430 | "node_modules/color-convert": {
431 | "version": "2.0.1",
432 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
433 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
434 | "dev": true,
435 | "dependencies": {
436 | "color-name": "~1.1.4"
437 | },
438 | "engines": {
439 | "node": ">=7.0.0"
440 | }
441 | },
442 | "node_modules/color-name": {
443 | "version": "1.1.4",
444 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
445 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
446 | "dev": true
447 | },
448 | "node_modules/concat-map": {
449 | "version": "0.0.1",
450 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
451 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==",
452 | "dev": true
453 | },
454 | "node_modules/content-disposition": {
455 | "version": "0.5.4",
456 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz",
457 | "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==",
458 | "dependencies": {
459 | "safe-buffer": "5.2.1"
460 | },
461 | "engines": {
462 | "node": ">= 0.6"
463 | }
464 | },
465 | "node_modules/content-type": {
466 | "version": "1.0.5",
467 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz",
468 | "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==",
469 | "engines": {
470 | "node": ">= 0.6"
471 | }
472 | },
473 | "node_modules/cookie": {
474 | "version": "0.5.0",
475 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz",
476 | "integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==",
477 | "engines": {
478 | "node": ">= 0.6"
479 | }
480 | },
481 | "node_modules/cookie-signature": {
482 | "version": "1.0.6",
483 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
484 | "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ=="
485 | },
486 | "node_modules/create-require": {
487 | "version": "1.1.1",
488 | "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz",
489 | "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==",
490 | "dev": true
491 | },
492 | "node_modules/cross-spawn": {
493 | "version": "7.0.3",
494 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
495 | "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==",
496 | "dev": true,
497 | "dependencies": {
498 | "path-key": "^3.1.0",
499 | "shebang-command": "^2.0.0",
500 | "which": "^2.0.1"
501 | },
502 | "engines": {
503 | "node": ">= 8"
504 | }
505 | },
506 | "node_modules/debug": {
507 | "version": "2.6.9",
508 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
509 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
510 | "dependencies": {
511 | "ms": "2.0.0"
512 | }
513 | },
514 | "node_modules/depd": {
515 | "version": "2.0.0",
516 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
517 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==",
518 | "engines": {
519 | "node": ">= 0.8"
520 | }
521 | },
522 | "node_modules/destroy": {
523 | "version": "1.2.0",
524 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz",
525 | "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==",
526 | "engines": {
527 | "node": ">= 0.8",
528 | "npm": "1.2.8000 || >= 1.4.16"
529 | }
530 | },
531 | "node_modules/diff": {
532 | "version": "4.0.2",
533 | "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz",
534 | "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==",
535 | "dev": true,
536 | "engines": {
537 | "node": ">=0.3.1"
538 | }
539 | },
540 | "node_modules/dotenv": {
541 | "version": "16.3.1",
542 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.3.1.tgz",
543 | "integrity": "sha512-IPzF4w4/Rd94bA9imS68tZBaYyBWSCE47V1RGuMrB94iyTOIEwRmVL2x/4An+6mETpLrKJ5hQkB8W4kFAadeIQ==",
544 | "engines": {
545 | "node": ">=12"
546 | },
547 | "funding": {
548 | "url": "https://github.com/motdotla/dotenv?sponsor=1"
549 | }
550 | },
551 | "node_modules/dynamic-dedupe": {
552 | "version": "0.3.0",
553 | "resolved": "https://registry.npmjs.org/dynamic-dedupe/-/dynamic-dedupe-0.3.0.tgz",
554 | "integrity": "sha512-ssuANeD+z97meYOqd50e04Ze5qp4bPqo8cCkI4TRjZkzAUgIDTrXV1R8QCdINpiI+hw14+rYazvTRdQrz0/rFQ==",
555 | "dev": true,
556 | "dependencies": {
557 | "xtend": "^4.0.0"
558 | }
559 | },
560 | "node_modules/eastasianwidth": {
561 | "version": "0.2.0",
562 | "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz",
563 | "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==",
564 | "dev": true
565 | },
566 | "node_modules/ee-first": {
567 | "version": "1.1.1",
568 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
569 | "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow=="
570 | },
571 | "node_modules/emoji-regex": {
572 | "version": "9.2.2",
573 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz",
574 | "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==",
575 | "dev": true
576 | },
577 | "node_modules/encodeurl": {
578 | "version": "1.0.2",
579 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
580 | "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==",
581 | "engines": {
582 | "node": ">= 0.8"
583 | }
584 | },
585 | "node_modules/env-var": {
586 | "version": "7.4.1",
587 | "resolved": "https://registry.npmjs.org/env-var/-/env-var-7.4.1.tgz",
588 | "integrity": "sha512-H8Ga2SbXTQwt6MKEawWSvmxoH1+J6bnAXkuyE7eDvbGmrhIL2i+XGjzGM3DFHcJu8GY1zY9/AnBJY8uGQYPHiw==",
589 | "engines": {
590 | "node": ">=10"
591 | }
592 | },
593 | "node_modules/escape-html": {
594 | "version": "1.0.3",
595 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
596 | "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow=="
597 | },
598 | "node_modules/etag": {
599 | "version": "1.8.1",
600 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
601 | "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==",
602 | "engines": {
603 | "node": ">= 0.6"
604 | }
605 | },
606 | "node_modules/express": {
607 | "version": "4.18.2",
608 | "resolved": "https://registry.npmjs.org/express/-/express-4.18.2.tgz",
609 | "integrity": "sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ==",
610 | "dependencies": {
611 | "accepts": "~1.3.8",
612 | "array-flatten": "1.1.1",
613 | "body-parser": "1.20.1",
614 | "content-disposition": "0.5.4",
615 | "content-type": "~1.0.4",
616 | "cookie": "0.5.0",
617 | "cookie-signature": "1.0.6",
618 | "debug": "2.6.9",
619 | "depd": "2.0.0",
620 | "encodeurl": "~1.0.2",
621 | "escape-html": "~1.0.3",
622 | "etag": "~1.8.1",
623 | "finalhandler": "1.2.0",
624 | "fresh": "0.5.2",
625 | "http-errors": "2.0.0",
626 | "merge-descriptors": "1.0.1",
627 | "methods": "~1.1.2",
628 | "on-finished": "2.4.1",
629 | "parseurl": "~1.3.3",
630 | "path-to-regexp": "0.1.7",
631 | "proxy-addr": "~2.0.7",
632 | "qs": "6.11.0",
633 | "range-parser": "~1.2.1",
634 | "safe-buffer": "5.2.1",
635 | "send": "0.18.0",
636 | "serve-static": "1.15.0",
637 | "setprototypeof": "1.2.0",
638 | "statuses": "2.0.1",
639 | "type-is": "~1.6.18",
640 | "utils-merge": "1.0.1",
641 | "vary": "~1.1.2"
642 | },
643 | "engines": {
644 | "node": ">= 0.10.0"
645 | }
646 | },
647 | "node_modules/fill-range": {
648 | "version": "7.0.1",
649 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
650 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
651 | "dev": true,
652 | "dependencies": {
653 | "to-regex-range": "^5.0.1"
654 | },
655 | "engines": {
656 | "node": ">=8"
657 | }
658 | },
659 | "node_modules/finalhandler": {
660 | "version": "1.2.0",
661 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.2.0.tgz",
662 | "integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==",
663 | "dependencies": {
664 | "debug": "2.6.9",
665 | "encodeurl": "~1.0.2",
666 | "escape-html": "~1.0.3",
667 | "on-finished": "2.4.1",
668 | "parseurl": "~1.3.3",
669 | "statuses": "2.0.1",
670 | "unpipe": "~1.0.0"
671 | },
672 | "engines": {
673 | "node": ">= 0.8"
674 | }
675 | },
676 | "node_modules/foreground-child": {
677 | "version": "3.1.1",
678 | "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.1.1.tgz",
679 | "integrity": "sha512-TMKDUnIte6bfb5nWv7V/caI169OHgvwjb7V4WkeUvbQQdjr5rWKqHFiKWb/fcOwB+CzBT+qbWjvj+DVwRskpIg==",
680 | "dev": true,
681 | "dependencies": {
682 | "cross-spawn": "^7.0.0",
683 | "signal-exit": "^4.0.1"
684 | },
685 | "engines": {
686 | "node": ">=14"
687 | },
688 | "funding": {
689 | "url": "https://github.com/sponsors/isaacs"
690 | }
691 | },
692 | "node_modules/forwarded": {
693 | "version": "0.2.0",
694 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
695 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==",
696 | "engines": {
697 | "node": ">= 0.6"
698 | }
699 | },
700 | "node_modules/fresh": {
701 | "version": "0.5.2",
702 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
703 | "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==",
704 | "engines": {
705 | "node": ">= 0.6"
706 | }
707 | },
708 | "node_modules/fs.realpath": {
709 | "version": "1.0.0",
710 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
711 | "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==",
712 | "dev": true
713 | },
714 | "node_modules/fsevents": {
715 | "version": "2.3.3",
716 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
717 | "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
718 | "dev": true,
719 | "hasInstallScript": true,
720 | "optional": true,
721 | "os": [
722 | "darwin"
723 | ],
724 | "engines": {
725 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
726 | }
727 | },
728 | "node_modules/function-bind": {
729 | "version": "1.1.1",
730 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
731 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A=="
732 | },
733 | "node_modules/get-intrinsic": {
734 | "version": "1.2.1",
735 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.1.tgz",
736 | "integrity": "sha512-2DcsyfABl+gVHEfCOaTrWgyt+tb6MSEGmKq+kI5HwLbIYgjgmMcV8KQ41uaKz1xxUcn9tJtgFbQUEVcEbd0FYw==",
737 | "dependencies": {
738 | "function-bind": "^1.1.1",
739 | "has": "^1.0.3",
740 | "has-proto": "^1.0.1",
741 | "has-symbols": "^1.0.3"
742 | },
743 | "funding": {
744 | "url": "https://github.com/sponsors/ljharb"
745 | }
746 | },
747 | "node_modules/glob": {
748 | "version": "10.3.10",
749 | "resolved": "https://registry.npmjs.org/glob/-/glob-10.3.10.tgz",
750 | "integrity": "sha512-fa46+tv1Ak0UPK1TOy/pZrIybNNt4HCv7SDzwyfiOZkvZLEbjsZkJBPtDHVshZjbecAoAGSC20MjLDG/qr679g==",
751 | "dev": true,
752 | "dependencies": {
753 | "foreground-child": "^3.1.0",
754 | "jackspeak": "^2.3.5",
755 | "minimatch": "^9.0.1",
756 | "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0",
757 | "path-scurry": "^1.10.1"
758 | },
759 | "bin": {
760 | "glob": "dist/esm/bin.mjs"
761 | },
762 | "engines": {
763 | "node": ">=16 || 14 >=14.17"
764 | },
765 | "funding": {
766 | "url": "https://github.com/sponsors/isaacs"
767 | }
768 | },
769 | "node_modules/glob-parent": {
770 | "version": "5.1.2",
771 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
772 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
773 | "dev": true,
774 | "dependencies": {
775 | "is-glob": "^4.0.1"
776 | },
777 | "engines": {
778 | "node": ">= 6"
779 | }
780 | },
781 | "node_modules/has": {
782 | "version": "1.0.3",
783 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz",
784 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==",
785 | "dependencies": {
786 | "function-bind": "^1.1.1"
787 | },
788 | "engines": {
789 | "node": ">= 0.4.0"
790 | }
791 | },
792 | "node_modules/has-proto": {
793 | "version": "1.0.1",
794 | "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.1.tgz",
795 | "integrity": "sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==",
796 | "engines": {
797 | "node": ">= 0.4"
798 | },
799 | "funding": {
800 | "url": "https://github.com/sponsors/ljharb"
801 | }
802 | },
803 | "node_modules/has-symbols": {
804 | "version": "1.0.3",
805 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz",
806 | "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==",
807 | "engines": {
808 | "node": ">= 0.4"
809 | },
810 | "funding": {
811 | "url": "https://github.com/sponsors/ljharb"
812 | }
813 | },
814 | "node_modules/http-errors": {
815 | "version": "2.0.0",
816 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz",
817 | "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==",
818 | "dependencies": {
819 | "depd": "2.0.0",
820 | "inherits": "2.0.4",
821 | "setprototypeof": "1.2.0",
822 | "statuses": "2.0.1",
823 | "toidentifier": "1.0.1"
824 | },
825 | "engines": {
826 | "node": ">= 0.8"
827 | }
828 | },
829 | "node_modules/iconv-lite": {
830 | "version": "0.4.24",
831 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
832 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
833 | "dependencies": {
834 | "safer-buffer": ">= 2.1.2 < 3"
835 | },
836 | "engines": {
837 | "node": ">=0.10.0"
838 | }
839 | },
840 | "node_modules/inflight": {
841 | "version": "1.0.6",
842 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
843 | "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==",
844 | "dev": true,
845 | "dependencies": {
846 | "once": "^1.3.0",
847 | "wrappy": "1"
848 | }
849 | },
850 | "node_modules/inherits": {
851 | "version": "2.0.4",
852 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
853 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="
854 | },
855 | "node_modules/ipaddr.js": {
856 | "version": "1.9.1",
857 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
858 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==",
859 | "engines": {
860 | "node": ">= 0.10"
861 | }
862 | },
863 | "node_modules/is-binary-path": {
864 | "version": "2.1.0",
865 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
866 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
867 | "dev": true,
868 | "dependencies": {
869 | "binary-extensions": "^2.0.0"
870 | },
871 | "engines": {
872 | "node": ">=8"
873 | }
874 | },
875 | "node_modules/is-core-module": {
876 | "version": "2.13.0",
877 | "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.13.0.tgz",
878 | "integrity": "sha512-Z7dk6Qo8pOCp3l4tsX2C5ZVas4V+UxwQodwZhLopL91TX8UyyHEXafPcyoeeWuLrwzHcr3igO78wNLwHJHsMCQ==",
879 | "dev": true,
880 | "dependencies": {
881 | "has": "^1.0.3"
882 | },
883 | "funding": {
884 | "url": "https://github.com/sponsors/ljharb"
885 | }
886 | },
887 | "node_modules/is-extglob": {
888 | "version": "2.1.1",
889 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
890 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
891 | "dev": true,
892 | "engines": {
893 | "node": ">=0.10.0"
894 | }
895 | },
896 | "node_modules/is-fullwidth-code-point": {
897 | "version": "3.0.0",
898 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
899 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
900 | "dev": true,
901 | "engines": {
902 | "node": ">=8"
903 | }
904 | },
905 | "node_modules/is-glob": {
906 | "version": "4.0.3",
907 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
908 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
909 | "dev": true,
910 | "dependencies": {
911 | "is-extglob": "^2.1.1"
912 | },
913 | "engines": {
914 | "node": ">=0.10.0"
915 | }
916 | },
917 | "node_modules/is-number": {
918 | "version": "7.0.0",
919 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
920 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
921 | "dev": true,
922 | "engines": {
923 | "node": ">=0.12.0"
924 | }
925 | },
926 | "node_modules/isexe": {
927 | "version": "2.0.0",
928 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
929 | "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==",
930 | "dev": true
931 | },
932 | "node_modules/jackspeak": {
933 | "version": "2.3.6",
934 | "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-2.3.6.tgz",
935 | "integrity": "sha512-N3yCS/NegsOBokc8GAdM8UcmfsKiSS8cipheD/nivzr700H+nsMOxJjQnvwOcRYVuFkdH0wGUvW2WbXGmrZGbQ==",
936 | "dev": true,
937 | "dependencies": {
938 | "@isaacs/cliui": "^8.0.2"
939 | },
940 | "engines": {
941 | "node": ">=14"
942 | },
943 | "funding": {
944 | "url": "https://github.com/sponsors/isaacs"
945 | },
946 | "optionalDependencies": {
947 | "@pkgjs/parseargs": "^0.11.0"
948 | }
949 | },
950 | "node_modules/lru-cache": {
951 | "version": "10.0.1",
952 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.0.1.tgz",
953 | "integrity": "sha512-IJ4uwUTi2qCccrioU6g9g/5rvvVl13bsdczUUcqbciD9iLr095yj8DQKdObriEvuNSx325N1rV1O0sJFszx75g==",
954 | "dev": true,
955 | "engines": {
956 | "node": "14 || >=16.14"
957 | }
958 | },
959 | "node_modules/make-error": {
960 | "version": "1.3.6",
961 | "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz",
962 | "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==",
963 | "dev": true
964 | },
965 | "node_modules/media-typer": {
966 | "version": "0.3.0",
967 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
968 | "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==",
969 | "engines": {
970 | "node": ">= 0.6"
971 | }
972 | },
973 | "node_modules/merge-descriptors": {
974 | "version": "1.0.1",
975 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
976 | "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w=="
977 | },
978 | "node_modules/methods": {
979 | "version": "1.1.2",
980 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
981 | "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==",
982 | "engines": {
983 | "node": ">= 0.6"
984 | }
985 | },
986 | "node_modules/mime": {
987 | "version": "1.6.0",
988 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
989 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==",
990 | "bin": {
991 | "mime": "cli.js"
992 | },
993 | "engines": {
994 | "node": ">=4"
995 | }
996 | },
997 | "node_modules/mime-db": {
998 | "version": "1.52.0",
999 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
1000 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
1001 | "engines": {
1002 | "node": ">= 0.6"
1003 | }
1004 | },
1005 | "node_modules/mime-types": {
1006 | "version": "2.1.35",
1007 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
1008 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
1009 | "dependencies": {
1010 | "mime-db": "1.52.0"
1011 | },
1012 | "engines": {
1013 | "node": ">= 0.6"
1014 | }
1015 | },
1016 | "node_modules/minimatch": {
1017 | "version": "9.0.3",
1018 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz",
1019 | "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==",
1020 | "dev": true,
1021 | "dependencies": {
1022 | "brace-expansion": "^2.0.1"
1023 | },
1024 | "engines": {
1025 | "node": ">=16 || 14 >=14.17"
1026 | },
1027 | "funding": {
1028 | "url": "https://github.com/sponsors/isaacs"
1029 | }
1030 | },
1031 | "node_modules/minimist": {
1032 | "version": "1.2.8",
1033 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz",
1034 | "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==",
1035 | "dev": true,
1036 | "funding": {
1037 | "url": "https://github.com/sponsors/ljharb"
1038 | }
1039 | },
1040 | "node_modules/minipass": {
1041 | "version": "7.0.4",
1042 | "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.0.4.tgz",
1043 | "integrity": "sha512-jYofLM5Dam9279rdkWzqHozUo4ybjdZmCsDHePy5V/PbBcVMiSZR97gmAy45aqi8CK1lG2ECd356FU86avfwUQ==",
1044 | "dev": true,
1045 | "engines": {
1046 | "node": ">=16 || 14 >=14.17"
1047 | }
1048 | },
1049 | "node_modules/mkdirp": {
1050 | "version": "1.0.4",
1051 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz",
1052 | "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==",
1053 | "dev": true,
1054 | "bin": {
1055 | "mkdirp": "bin/cmd.js"
1056 | },
1057 | "engines": {
1058 | "node": ">=10"
1059 | }
1060 | },
1061 | "node_modules/ms": {
1062 | "version": "2.0.0",
1063 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1064 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A=="
1065 | },
1066 | "node_modules/negotiator": {
1067 | "version": "0.6.3",
1068 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz",
1069 | "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==",
1070 | "engines": {
1071 | "node": ">= 0.6"
1072 | }
1073 | },
1074 | "node_modules/normalize-path": {
1075 | "version": "3.0.0",
1076 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
1077 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
1078 | "dev": true,
1079 | "engines": {
1080 | "node": ">=0.10.0"
1081 | }
1082 | },
1083 | "node_modules/object-inspect": {
1084 | "version": "1.12.3",
1085 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.3.tgz",
1086 | "integrity": "sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g==",
1087 | "funding": {
1088 | "url": "https://github.com/sponsors/ljharb"
1089 | }
1090 | },
1091 | "node_modules/on-finished": {
1092 | "version": "2.4.1",
1093 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz",
1094 | "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==",
1095 | "dependencies": {
1096 | "ee-first": "1.1.1"
1097 | },
1098 | "engines": {
1099 | "node": ">= 0.8"
1100 | }
1101 | },
1102 | "node_modules/once": {
1103 | "version": "1.4.0",
1104 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1105 | "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==",
1106 | "dev": true,
1107 | "dependencies": {
1108 | "wrappy": "1"
1109 | }
1110 | },
1111 | "node_modules/parseurl": {
1112 | "version": "1.3.3",
1113 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
1114 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==",
1115 | "engines": {
1116 | "node": ">= 0.8"
1117 | }
1118 | },
1119 | "node_modules/path-is-absolute": {
1120 | "version": "1.0.1",
1121 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
1122 | "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==",
1123 | "dev": true,
1124 | "engines": {
1125 | "node": ">=0.10.0"
1126 | }
1127 | },
1128 | "node_modules/path-key": {
1129 | "version": "3.1.1",
1130 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
1131 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==",
1132 | "dev": true,
1133 | "engines": {
1134 | "node": ">=8"
1135 | }
1136 | },
1137 | "node_modules/path-parse": {
1138 | "version": "1.0.7",
1139 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz",
1140 | "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==",
1141 | "dev": true
1142 | },
1143 | "node_modules/path-scurry": {
1144 | "version": "1.10.1",
1145 | "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.10.1.tgz",
1146 | "integrity": "sha512-MkhCqzzBEpPvxxQ71Md0b1Kk51W01lrYvlMzSUaIzNsODdd7mqhiimSZlr+VegAz5Z6Vzt9Xg2ttE//XBhH3EQ==",
1147 | "dev": true,
1148 | "dependencies": {
1149 | "lru-cache": "^9.1.1 || ^10.0.0",
1150 | "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0"
1151 | },
1152 | "engines": {
1153 | "node": ">=16 || 14 >=14.17"
1154 | },
1155 | "funding": {
1156 | "url": "https://github.com/sponsors/isaacs"
1157 | }
1158 | },
1159 | "node_modules/path-to-regexp": {
1160 | "version": "0.1.7",
1161 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
1162 | "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ=="
1163 | },
1164 | "node_modules/picomatch": {
1165 | "version": "2.3.1",
1166 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz",
1167 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==",
1168 | "dev": true,
1169 | "engines": {
1170 | "node": ">=8.6"
1171 | },
1172 | "funding": {
1173 | "url": "https://github.com/sponsors/jonschlinkert"
1174 | }
1175 | },
1176 | "node_modules/proxy-addr": {
1177 | "version": "2.0.7",
1178 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
1179 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
1180 | "dependencies": {
1181 | "forwarded": "0.2.0",
1182 | "ipaddr.js": "1.9.1"
1183 | },
1184 | "engines": {
1185 | "node": ">= 0.10"
1186 | }
1187 | },
1188 | "node_modules/qs": {
1189 | "version": "6.11.0",
1190 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.11.0.tgz",
1191 | "integrity": "sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==",
1192 | "dependencies": {
1193 | "side-channel": "^1.0.4"
1194 | },
1195 | "engines": {
1196 | "node": ">=0.6"
1197 | },
1198 | "funding": {
1199 | "url": "https://github.com/sponsors/ljharb"
1200 | }
1201 | },
1202 | "node_modules/range-parser": {
1203 | "version": "1.2.1",
1204 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
1205 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
1206 | "engines": {
1207 | "node": ">= 0.6"
1208 | }
1209 | },
1210 | "node_modules/raw-body": {
1211 | "version": "2.5.1",
1212 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.1.tgz",
1213 | "integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==",
1214 | "dependencies": {
1215 | "bytes": "3.1.2",
1216 | "http-errors": "2.0.0",
1217 | "iconv-lite": "0.4.24",
1218 | "unpipe": "1.0.0"
1219 | },
1220 | "engines": {
1221 | "node": ">= 0.8"
1222 | }
1223 | },
1224 | "node_modules/readdirp": {
1225 | "version": "3.6.0",
1226 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz",
1227 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==",
1228 | "dev": true,
1229 | "dependencies": {
1230 | "picomatch": "^2.2.1"
1231 | },
1232 | "engines": {
1233 | "node": ">=8.10.0"
1234 | }
1235 | },
1236 | "node_modules/resolve": {
1237 | "version": "1.22.6",
1238 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.6.tgz",
1239 | "integrity": "sha512-njhxM7mV12JfufShqGy3Rz8j11RPdLy4xi15UurGJeoHLfJpVXKdh3ueuOqbYUcDZnffr6X739JBo5LzyahEsw==",
1240 | "dev": true,
1241 | "dependencies": {
1242 | "is-core-module": "^2.13.0",
1243 | "path-parse": "^1.0.7",
1244 | "supports-preserve-symlinks-flag": "^1.0.0"
1245 | },
1246 | "bin": {
1247 | "resolve": "bin/resolve"
1248 | },
1249 | "funding": {
1250 | "url": "https://github.com/sponsors/ljharb"
1251 | }
1252 | },
1253 | "node_modules/rimraf": {
1254 | "version": "5.0.5",
1255 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-5.0.5.tgz",
1256 | "integrity": "sha512-CqDakW+hMe/Bz202FPEymy68P+G50RfMQK+Qo5YUqc9SPipvbGjCGKd0RSKEelbsfQuw3g5NZDSrlZZAJurH1A==",
1257 | "dev": true,
1258 | "dependencies": {
1259 | "glob": "^10.3.7"
1260 | },
1261 | "bin": {
1262 | "rimraf": "dist/esm/bin.mjs"
1263 | },
1264 | "engines": {
1265 | "node": ">=14"
1266 | },
1267 | "funding": {
1268 | "url": "https://github.com/sponsors/isaacs"
1269 | }
1270 | },
1271 | "node_modules/safe-buffer": {
1272 | "version": "5.2.1",
1273 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
1274 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
1275 | "funding": [
1276 | {
1277 | "type": "github",
1278 | "url": "https://github.com/sponsors/feross"
1279 | },
1280 | {
1281 | "type": "patreon",
1282 | "url": "https://www.patreon.com/feross"
1283 | },
1284 | {
1285 | "type": "consulting",
1286 | "url": "https://feross.org/support"
1287 | }
1288 | ]
1289 | },
1290 | "node_modules/safer-buffer": {
1291 | "version": "2.1.2",
1292 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
1293 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
1294 | },
1295 | "node_modules/send": {
1296 | "version": "0.18.0",
1297 | "resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz",
1298 | "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==",
1299 | "dependencies": {
1300 | "debug": "2.6.9",
1301 | "depd": "2.0.0",
1302 | "destroy": "1.2.0",
1303 | "encodeurl": "~1.0.2",
1304 | "escape-html": "~1.0.3",
1305 | "etag": "~1.8.1",
1306 | "fresh": "0.5.2",
1307 | "http-errors": "2.0.0",
1308 | "mime": "1.6.0",
1309 | "ms": "2.1.3",
1310 | "on-finished": "2.4.1",
1311 | "range-parser": "~1.2.1",
1312 | "statuses": "2.0.1"
1313 | },
1314 | "engines": {
1315 | "node": ">= 0.8.0"
1316 | }
1317 | },
1318 | "node_modules/send/node_modules/ms": {
1319 | "version": "2.1.3",
1320 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
1321 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
1322 | },
1323 | "node_modules/serve-static": {
1324 | "version": "1.15.0",
1325 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz",
1326 | "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==",
1327 | "dependencies": {
1328 | "encodeurl": "~1.0.2",
1329 | "escape-html": "~1.0.3",
1330 | "parseurl": "~1.3.3",
1331 | "send": "0.18.0"
1332 | },
1333 | "engines": {
1334 | "node": ">= 0.8.0"
1335 | }
1336 | },
1337 | "node_modules/setprototypeof": {
1338 | "version": "1.2.0",
1339 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
1340 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw=="
1341 | },
1342 | "node_modules/shebang-command": {
1343 | "version": "2.0.0",
1344 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
1345 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
1346 | "dev": true,
1347 | "dependencies": {
1348 | "shebang-regex": "^3.0.0"
1349 | },
1350 | "engines": {
1351 | "node": ">=8"
1352 | }
1353 | },
1354 | "node_modules/shebang-regex": {
1355 | "version": "3.0.0",
1356 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
1357 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==",
1358 | "dev": true,
1359 | "engines": {
1360 | "node": ">=8"
1361 | }
1362 | },
1363 | "node_modules/side-channel": {
1364 | "version": "1.0.4",
1365 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz",
1366 | "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==",
1367 | "dependencies": {
1368 | "call-bind": "^1.0.0",
1369 | "get-intrinsic": "^1.0.2",
1370 | "object-inspect": "^1.9.0"
1371 | },
1372 | "funding": {
1373 | "url": "https://github.com/sponsors/ljharb"
1374 | }
1375 | },
1376 | "node_modules/signal-exit": {
1377 | "version": "4.1.0",
1378 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz",
1379 | "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==",
1380 | "dev": true,
1381 | "engines": {
1382 | "node": ">=14"
1383 | },
1384 | "funding": {
1385 | "url": "https://github.com/sponsors/isaacs"
1386 | }
1387 | },
1388 | "node_modules/source-map": {
1389 | "version": "0.6.1",
1390 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
1391 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==",
1392 | "dev": true,
1393 | "engines": {
1394 | "node": ">=0.10.0"
1395 | }
1396 | },
1397 | "node_modules/source-map-support": {
1398 | "version": "0.5.21",
1399 | "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz",
1400 | "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==",
1401 | "dev": true,
1402 | "dependencies": {
1403 | "buffer-from": "^1.0.0",
1404 | "source-map": "^0.6.0"
1405 | }
1406 | },
1407 | "node_modules/statuses": {
1408 | "version": "2.0.1",
1409 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz",
1410 | "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==",
1411 | "engines": {
1412 | "node": ">= 0.8"
1413 | }
1414 | },
1415 | "node_modules/string-width": {
1416 | "version": "5.1.2",
1417 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz",
1418 | "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==",
1419 | "dev": true,
1420 | "dependencies": {
1421 | "eastasianwidth": "^0.2.0",
1422 | "emoji-regex": "^9.2.2",
1423 | "strip-ansi": "^7.0.1"
1424 | },
1425 | "engines": {
1426 | "node": ">=12"
1427 | },
1428 | "funding": {
1429 | "url": "https://github.com/sponsors/sindresorhus"
1430 | }
1431 | },
1432 | "node_modules/string-width-cjs": {
1433 | "name": "string-width",
1434 | "version": "4.2.3",
1435 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
1436 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
1437 | "dev": true,
1438 | "dependencies": {
1439 | "emoji-regex": "^8.0.0",
1440 | "is-fullwidth-code-point": "^3.0.0",
1441 | "strip-ansi": "^6.0.1"
1442 | },
1443 | "engines": {
1444 | "node": ">=8"
1445 | }
1446 | },
1447 | "node_modules/string-width-cjs/node_modules/ansi-regex": {
1448 | "version": "5.0.1",
1449 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
1450 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
1451 | "dev": true,
1452 | "engines": {
1453 | "node": ">=8"
1454 | }
1455 | },
1456 | "node_modules/string-width-cjs/node_modules/emoji-regex": {
1457 | "version": "8.0.0",
1458 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
1459 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
1460 | "dev": true
1461 | },
1462 | "node_modules/string-width-cjs/node_modules/strip-ansi": {
1463 | "version": "6.0.1",
1464 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
1465 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
1466 | "dev": true,
1467 | "dependencies": {
1468 | "ansi-regex": "^5.0.1"
1469 | },
1470 | "engines": {
1471 | "node": ">=8"
1472 | }
1473 | },
1474 | "node_modules/strip-ansi": {
1475 | "version": "7.1.0",
1476 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz",
1477 | "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==",
1478 | "dev": true,
1479 | "dependencies": {
1480 | "ansi-regex": "^6.0.1"
1481 | },
1482 | "engines": {
1483 | "node": ">=12"
1484 | },
1485 | "funding": {
1486 | "url": "https://github.com/chalk/strip-ansi?sponsor=1"
1487 | }
1488 | },
1489 | "node_modules/strip-ansi-cjs": {
1490 | "name": "strip-ansi",
1491 | "version": "6.0.1",
1492 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
1493 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
1494 | "dev": true,
1495 | "dependencies": {
1496 | "ansi-regex": "^5.0.1"
1497 | },
1498 | "engines": {
1499 | "node": ">=8"
1500 | }
1501 | },
1502 | "node_modules/strip-ansi-cjs/node_modules/ansi-regex": {
1503 | "version": "5.0.1",
1504 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
1505 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
1506 | "dev": true,
1507 | "engines": {
1508 | "node": ">=8"
1509 | }
1510 | },
1511 | "node_modules/strip-bom": {
1512 | "version": "3.0.0",
1513 | "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz",
1514 | "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==",
1515 | "dev": true,
1516 | "engines": {
1517 | "node": ">=4"
1518 | }
1519 | },
1520 | "node_modules/strip-json-comments": {
1521 | "version": "2.0.1",
1522 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
1523 | "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==",
1524 | "dev": true,
1525 | "engines": {
1526 | "node": ">=0.10.0"
1527 | }
1528 | },
1529 | "node_modules/supports-preserve-symlinks-flag": {
1530 | "version": "1.0.0",
1531 | "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz",
1532 | "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==",
1533 | "dev": true,
1534 | "engines": {
1535 | "node": ">= 0.4"
1536 | },
1537 | "funding": {
1538 | "url": "https://github.com/sponsors/ljharb"
1539 | }
1540 | },
1541 | "node_modules/to-regex-range": {
1542 | "version": "5.0.1",
1543 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
1544 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
1545 | "dev": true,
1546 | "dependencies": {
1547 | "is-number": "^7.0.0"
1548 | },
1549 | "engines": {
1550 | "node": ">=8.0"
1551 | }
1552 | },
1553 | "node_modules/toidentifier": {
1554 | "version": "1.0.1",
1555 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
1556 | "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==",
1557 | "engines": {
1558 | "node": ">=0.6"
1559 | }
1560 | },
1561 | "node_modules/tree-kill": {
1562 | "version": "1.2.2",
1563 | "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz",
1564 | "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==",
1565 | "dev": true,
1566 | "bin": {
1567 | "tree-kill": "cli.js"
1568 | }
1569 | },
1570 | "node_modules/ts-node": {
1571 | "version": "10.9.1",
1572 | "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.9.1.tgz",
1573 | "integrity": "sha512-NtVysVPkxxrwFGUUxGYhfux8k78pQB3JqYBXlLRZgdGUqTO5wU/UyHop5p70iEbGhB7q5KmiZiU0Y3KlJrScEw==",
1574 | "dev": true,
1575 | "dependencies": {
1576 | "@cspotcode/source-map-support": "^0.8.0",
1577 | "@tsconfig/node10": "^1.0.7",
1578 | "@tsconfig/node12": "^1.0.7",
1579 | "@tsconfig/node14": "^1.0.0",
1580 | "@tsconfig/node16": "^1.0.2",
1581 | "acorn": "^8.4.1",
1582 | "acorn-walk": "^8.1.1",
1583 | "arg": "^4.1.0",
1584 | "create-require": "^1.1.0",
1585 | "diff": "^4.0.1",
1586 | "make-error": "^1.1.1",
1587 | "v8-compile-cache-lib": "^3.0.1",
1588 | "yn": "3.1.1"
1589 | },
1590 | "bin": {
1591 | "ts-node": "dist/bin.js",
1592 | "ts-node-cwd": "dist/bin-cwd.js",
1593 | "ts-node-esm": "dist/bin-esm.js",
1594 | "ts-node-script": "dist/bin-script.js",
1595 | "ts-node-transpile-only": "dist/bin-transpile.js",
1596 | "ts-script": "dist/bin-script-deprecated.js"
1597 | },
1598 | "peerDependencies": {
1599 | "@swc/core": ">=1.2.50",
1600 | "@swc/wasm": ">=1.2.50",
1601 | "@types/node": "*",
1602 | "typescript": ">=2.7"
1603 | },
1604 | "peerDependenciesMeta": {
1605 | "@swc/core": {
1606 | "optional": true
1607 | },
1608 | "@swc/wasm": {
1609 | "optional": true
1610 | }
1611 | }
1612 | },
1613 | "node_modules/ts-node-dev": {
1614 | "version": "2.0.0",
1615 | "resolved": "https://registry.npmjs.org/ts-node-dev/-/ts-node-dev-2.0.0.tgz",
1616 | "integrity": "sha512-ywMrhCfH6M75yftYvrvNarLEY+SUXtUvU8/0Z6llrHQVBx12GiFk5sStF8UdfE/yfzk9IAq7O5EEbTQsxlBI8w==",
1617 | "dev": true,
1618 | "dependencies": {
1619 | "chokidar": "^3.5.1",
1620 | "dynamic-dedupe": "^0.3.0",
1621 | "minimist": "^1.2.6",
1622 | "mkdirp": "^1.0.4",
1623 | "resolve": "^1.0.0",
1624 | "rimraf": "^2.6.1",
1625 | "source-map-support": "^0.5.12",
1626 | "tree-kill": "^1.2.2",
1627 | "ts-node": "^10.4.0",
1628 | "tsconfig": "^7.0.0"
1629 | },
1630 | "bin": {
1631 | "ts-node-dev": "lib/bin.js",
1632 | "tsnd": "lib/bin.js"
1633 | },
1634 | "engines": {
1635 | "node": ">=0.8.0"
1636 | },
1637 | "peerDependencies": {
1638 | "node-notifier": "*",
1639 | "typescript": "*"
1640 | },
1641 | "peerDependenciesMeta": {
1642 | "node-notifier": {
1643 | "optional": true
1644 | }
1645 | }
1646 | },
1647 | "node_modules/ts-node-dev/node_modules/brace-expansion": {
1648 | "version": "1.1.11",
1649 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
1650 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
1651 | "dev": true,
1652 | "dependencies": {
1653 | "balanced-match": "^1.0.0",
1654 | "concat-map": "0.0.1"
1655 | }
1656 | },
1657 | "node_modules/ts-node-dev/node_modules/glob": {
1658 | "version": "7.2.3",
1659 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz",
1660 | "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==",
1661 | "dev": true,
1662 | "dependencies": {
1663 | "fs.realpath": "^1.0.0",
1664 | "inflight": "^1.0.4",
1665 | "inherits": "2",
1666 | "minimatch": "^3.1.1",
1667 | "once": "^1.3.0",
1668 | "path-is-absolute": "^1.0.0"
1669 | },
1670 | "engines": {
1671 | "node": "*"
1672 | },
1673 | "funding": {
1674 | "url": "https://github.com/sponsors/isaacs"
1675 | }
1676 | },
1677 | "node_modules/ts-node-dev/node_modules/minimatch": {
1678 | "version": "3.1.2",
1679 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
1680 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
1681 | "dev": true,
1682 | "dependencies": {
1683 | "brace-expansion": "^1.1.7"
1684 | },
1685 | "engines": {
1686 | "node": "*"
1687 | }
1688 | },
1689 | "node_modules/ts-node-dev/node_modules/rimraf": {
1690 | "version": "2.7.1",
1691 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz",
1692 | "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==",
1693 | "dev": true,
1694 | "dependencies": {
1695 | "glob": "^7.1.3"
1696 | },
1697 | "bin": {
1698 | "rimraf": "bin.js"
1699 | }
1700 | },
1701 | "node_modules/tsconfig": {
1702 | "version": "7.0.0",
1703 | "resolved": "https://registry.npmjs.org/tsconfig/-/tsconfig-7.0.0.tgz",
1704 | "integrity": "sha512-vZXmzPrL+EmC4T/4rVlT2jNVMWCi/O4DIiSj3UHg1OE5kCKbk4mfrXc6dZksLgRM/TZlKnousKH9bbTazUWRRw==",
1705 | "dev": true,
1706 | "dependencies": {
1707 | "@types/strip-bom": "^3.0.0",
1708 | "@types/strip-json-comments": "0.0.30",
1709 | "strip-bom": "^3.0.0",
1710 | "strip-json-comments": "^2.0.0"
1711 | }
1712 | },
1713 | "node_modules/type-is": {
1714 | "version": "1.6.18",
1715 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
1716 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
1717 | "dependencies": {
1718 | "media-typer": "0.3.0",
1719 | "mime-types": "~2.1.24"
1720 | },
1721 | "engines": {
1722 | "node": ">= 0.6"
1723 | }
1724 | },
1725 | "node_modules/typescript": {
1726 | "version": "5.2.2",
1727 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz",
1728 | "integrity": "sha512-mI4WrpHsbCIcwT9cF4FZvr80QUeKvsUsUvKDoR+X/7XHQH98xYD8YHZg7ANtz2GtZt/CBq2QJ0thkGJMHfqc1w==",
1729 | "dev": true,
1730 | "bin": {
1731 | "tsc": "bin/tsc",
1732 | "tsserver": "bin/tsserver"
1733 | },
1734 | "engines": {
1735 | "node": ">=14.17"
1736 | }
1737 | },
1738 | "node_modules/unpipe": {
1739 | "version": "1.0.0",
1740 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
1741 | "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==",
1742 | "engines": {
1743 | "node": ">= 0.8"
1744 | }
1745 | },
1746 | "node_modules/utils-merge": {
1747 | "version": "1.0.1",
1748 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
1749 | "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==",
1750 | "engines": {
1751 | "node": ">= 0.4.0"
1752 | }
1753 | },
1754 | "node_modules/uuid": {
1755 | "version": "9.0.1",
1756 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-9.0.1.tgz",
1757 | "integrity": "sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==",
1758 | "funding": [
1759 | "https://github.com/sponsors/broofa",
1760 | "https://github.com/sponsors/ctavan"
1761 | ],
1762 | "bin": {
1763 | "uuid": "dist/bin/uuid"
1764 | }
1765 | },
1766 | "node_modules/v8-compile-cache-lib": {
1767 | "version": "3.0.1",
1768 | "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz",
1769 | "integrity": "sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==",
1770 | "dev": true
1771 | },
1772 | "node_modules/vary": {
1773 | "version": "1.1.2",
1774 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
1775 | "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==",
1776 | "engines": {
1777 | "node": ">= 0.8"
1778 | }
1779 | },
1780 | "node_modules/which": {
1781 | "version": "2.0.2",
1782 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
1783 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
1784 | "dev": true,
1785 | "dependencies": {
1786 | "isexe": "^2.0.0"
1787 | },
1788 | "bin": {
1789 | "node-which": "bin/node-which"
1790 | },
1791 | "engines": {
1792 | "node": ">= 8"
1793 | }
1794 | },
1795 | "node_modules/wrap-ansi": {
1796 | "version": "8.1.0",
1797 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz",
1798 | "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==",
1799 | "dev": true,
1800 | "dependencies": {
1801 | "ansi-styles": "^6.1.0",
1802 | "string-width": "^5.0.1",
1803 | "strip-ansi": "^7.0.1"
1804 | },
1805 | "engines": {
1806 | "node": ">=12"
1807 | },
1808 | "funding": {
1809 | "url": "https://github.com/chalk/wrap-ansi?sponsor=1"
1810 | }
1811 | },
1812 | "node_modules/wrap-ansi-cjs": {
1813 | "name": "wrap-ansi",
1814 | "version": "7.0.0",
1815 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz",
1816 | "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==",
1817 | "dev": true,
1818 | "dependencies": {
1819 | "ansi-styles": "^4.0.0",
1820 | "string-width": "^4.1.0",
1821 | "strip-ansi": "^6.0.0"
1822 | },
1823 | "engines": {
1824 | "node": ">=10"
1825 | },
1826 | "funding": {
1827 | "url": "https://github.com/chalk/wrap-ansi?sponsor=1"
1828 | }
1829 | },
1830 | "node_modules/wrap-ansi-cjs/node_modules/ansi-regex": {
1831 | "version": "5.0.1",
1832 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
1833 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
1834 | "dev": true,
1835 | "engines": {
1836 | "node": ">=8"
1837 | }
1838 | },
1839 | "node_modules/wrap-ansi-cjs/node_modules/ansi-styles": {
1840 | "version": "4.3.0",
1841 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
1842 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
1843 | "dev": true,
1844 | "dependencies": {
1845 | "color-convert": "^2.0.1"
1846 | },
1847 | "engines": {
1848 | "node": ">=8"
1849 | },
1850 | "funding": {
1851 | "url": "https://github.com/chalk/ansi-styles?sponsor=1"
1852 | }
1853 | },
1854 | "node_modules/wrap-ansi-cjs/node_modules/emoji-regex": {
1855 | "version": "8.0.0",
1856 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
1857 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
1858 | "dev": true
1859 | },
1860 | "node_modules/wrap-ansi-cjs/node_modules/string-width": {
1861 | "version": "4.2.3",
1862 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
1863 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
1864 | "dev": true,
1865 | "dependencies": {
1866 | "emoji-regex": "^8.0.0",
1867 | "is-fullwidth-code-point": "^3.0.0",
1868 | "strip-ansi": "^6.0.1"
1869 | },
1870 | "engines": {
1871 | "node": ">=8"
1872 | }
1873 | },
1874 | "node_modules/wrap-ansi-cjs/node_modules/strip-ansi": {
1875 | "version": "6.0.1",
1876 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
1877 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
1878 | "dev": true,
1879 | "dependencies": {
1880 | "ansi-regex": "^5.0.1"
1881 | },
1882 | "engines": {
1883 | "node": ">=8"
1884 | }
1885 | },
1886 | "node_modules/wrappy": {
1887 | "version": "1.0.2",
1888 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
1889 | "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==",
1890 | "dev": true
1891 | },
1892 | "node_modules/ws": {
1893 | "version": "8.14.2",
1894 | "resolved": "https://registry.npmjs.org/ws/-/ws-8.14.2.tgz",
1895 | "integrity": "sha512-wEBG1ftX4jcglPxgFCMJmZ2PLtSbJ2Peg6TmpJFTbe9GZYOQCDPdMYu/Tm0/bGZkw8paZnJY45J4K2PZrLYq8g==",
1896 | "engines": {
1897 | "node": ">=10.0.0"
1898 | },
1899 | "peerDependencies": {
1900 | "bufferutil": "^4.0.1",
1901 | "utf-8-validate": ">=5.0.2"
1902 | },
1903 | "peerDependenciesMeta": {
1904 | "bufferutil": {
1905 | "optional": true
1906 | },
1907 | "utf-8-validate": {
1908 | "optional": true
1909 | }
1910 | }
1911 | },
1912 | "node_modules/xtend": {
1913 | "version": "4.0.2",
1914 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz",
1915 | "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==",
1916 | "dev": true,
1917 | "engines": {
1918 | "node": ">=0.4"
1919 | }
1920 | },
1921 | "node_modules/yn": {
1922 | "version": "3.1.1",
1923 | "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz",
1924 | "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==",
1925 | "dev": true,
1926 | "engines": {
1927 | "node": ">=6"
1928 | }
1929 | }
1930 | }
1931 | }
1932 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "08-user-store",
3 | "version": "1.0.0",
4 | "description": "",
5 | "main": "index.js",
6 | "scripts": {
7 | "dev": "tsnd --respawn --clear src/app.ts",
8 | "build": "rimraf ./dist && tsc",
9 | "start": "npm run build && node dist/app.js"
10 | },
11 | "keywords": [],
12 | "author": "",
13 | "license": "ISC",
14 | "devDependencies": {
15 | "@types/express": "^4.17.18",
16 | "@types/node": "^20.8.0",
17 | "@types/uuid": "^9.0.4",
18 | "@types/ws": "^8.5.6",
19 | "rimraf": "^5.0.5",
20 | "ts-node-dev": "^2.0.0",
21 | "typescript": "^5.2.2"
22 | },
23 | "dependencies": {
24 | "dotenv": "^16.3.1",
25 | "env-var": "^7.4.1",
26 | "express": "^4.18.2",
27 | "uuid": "^9.0.1",
28 | "ws": "^8.14.2"
29 | }
30 | }
31 |
--------------------------------------------------------------------------------
/public/.DS_Store:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/Klerith/ws-rest-api/d626082f649a9d676c78fc2ad831ef6d342c2a33/public/.DS_Store
--------------------------------------------------------------------------------
/public/css/style.css:
--------------------------------------------------------------------------------
1 | html,
2 | body {
3 | height: 100%;
4 | width: 100%;
5 | background-color: #6951F0;
6 | color: white;
7 | }
8 |
9 | table {
10 | width: 100%;
11 | height: 100%;
12 | text-align: center;
13 | }
14 |
15 | .ticket-current {
16 | background-color: #282828;
17 | }
18 |
19 | .ticket-current-number {
20 | font-size: 80px;
21 | }
22 |
23 | .ticket-current-desk {
24 | font-size: 45px;
25 | }
26 |
27 | .ticket-secondary {
28 | font-size: 25px;
29 | }
30 |
31 |
32 | /* Nuevo-ticket */
33 |
34 | #lbl-new-ticket {
35 | font-size: 70px;
36 | }
37 |
--------------------------------------------------------------------------------
/public/desk.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | Programa de Colas
9 |
10 |
11 |
12 |
13 |
14 | Escritorio
15 |
16 |
17 |
18 |
Atendiendo a ....
19 |
20 |
23 |
24 |
27 |
28 |
29 |
30 |
En cola
31 |
32 |
33 | Ya no hay más tickets
34 |
35 |
36 |
Espere...
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
--------------------------------------------------------------------------------
/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | Programa de Colas
9 |
10 |
11 |
12 |
13 |
14 | Programa de Colas
15 |
16 |
17 |
23 |
24 |
25 |
39 |
40 |
41 |
42 |
43 |
44 |
--------------------------------------------------------------------------------
/public/js/.DS_Store:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/Klerith/ws-rest-api/d626082f649a9d676c78fc2ad831ef6d342c2a33/public/js/.DS_Store
--------------------------------------------------------------------------------
/public/js/desk.js:
--------------------------------------------------------------------------------
1 | // Referencias HTML
2 | const lblPending = document.querySelector('#lbl-pending');
3 | const deskHeader = document.querySelector('h1');
4 | const noMoreAlert = document.querySelector('.alert');
5 | const lblCurrentTicket = document.querySelector('small')
6 |
7 | const btnDraw = document.querySelector('#btn-draw');
8 | const btnDone = document.querySelector('#btn-done');
9 |
10 |
11 | const searchParams = new URLSearchParams( window.location.search );
12 |
13 | if ( !searchParams.has('escritorio') ) {
14 | window.location = 'index.html';
15 | throw new Error('Escritorio es requerido');
16 | }
17 |
18 | const deskNumber = searchParams.get('escritorio');
19 | let workingTicket = null;
20 |
21 | deskHeader.innerText = deskNumber;
22 |
23 |
24 | function checkTicketCount( currentCount = 0 ) {
25 | // noMoreAlert.classList.toggle('d-none');
26 | if ( currentCount === 0 ) {
27 | noMoreAlert.classList.remove('d-none');
28 | } else {
29 | noMoreAlert.classList.add('d-none');
30 | }
31 |
32 | lblPending.innerHTML = currentCount;
33 | }
34 |
35 |
36 | async function loadInitialCount() {
37 | const pendingTickets = await fetch('/api/ticket/pending').then(resp => resp.json());
38 | checkTicketCount(pendingTickets.length);
39 | }
40 |
41 | async function getTicket() {
42 | await finishTicket();
43 |
44 | const { status, ticket, message } = await fetch(`/api/ticket/draw/${ deskNumber }`)
45 | .then( resp =>resp.json() );
46 |
47 | if ( status === 'error' ) {
48 | lblCurrentTicket.innerText = message;
49 | return;
50 | }
51 |
52 | workingTicket = ticket;
53 | lblCurrentTicket.innerText = ticket.number;
54 | }
55 |
56 | async function finishTicket() {
57 | if ( !workingTicket ) return;
58 |
59 | const { status, message } = await fetch(`/api/ticket/done/${ workingTicket.id }`, {
60 | method: 'PUT'
61 | }).then( resp => resp.json() );
62 |
63 | console.log({ status, message });
64 |
65 | if ( status === 'ok' ) {
66 | workingTicket = null;
67 | lblCurrentTicket.innerText = 'Nadie';
68 | }
69 |
70 | }
71 |
72 |
73 |
74 |
75 |
76 | function connectToWebSockets() {
77 |
78 | const socket = new WebSocket( 'ws://localhost:3000/ws' );
79 |
80 | socket.onmessage = ( event ) => {
81 | // console.log(event.data); // on-ticket-count-changed
82 | const { type, payload } = JSON.parse( event.data );
83 | if ( type !== 'on-ticket-count-changed' ) return;
84 | checkTicketCount(payload);
85 | };
86 |
87 | socket.onclose = ( event ) => {
88 | console.log( 'Connection closed' );
89 | setTimeout( () => {
90 | console.log( 'retrying to connect' );
91 | connectToWebSockets();
92 | }, 1500 );
93 |
94 | };
95 |
96 | socket.onopen = ( event ) => {
97 | console.log( 'Connected' );
98 | };
99 |
100 | }
101 |
102 |
103 |
104 | // Listeners
105 | btnDraw.addEventListener('click', getTicket );
106 | btnDone.addEventListener('click', finishTicket );
107 |
108 |
109 |
110 | // Init
111 | loadInitialCount();
112 | connectToWebSockets();
--------------------------------------------------------------------------------
/public/js/new-ticket.js:
--------------------------------------------------------------------------------
1 |
2 | const currentTicketLbl = document.querySelector('span');
3 | const createTicketBtn = document.querySelector('button');
4 |
5 |
6 |
7 | async function getLastTicket() {
8 | const lastTicket = await fetch('/api/ticket/last').then( resp => resp.json() );
9 | currentTicketLbl.innerText = lastTicket;
10 | }
11 |
12 | async function createTicket() {
13 | const newTicket = await fetch('/api/ticket',{
14 | method: 'POST'
15 | }).then( resp => resp.json());
16 |
17 | currentTicketLbl.innerText = newTicket.number;
18 | }
19 |
20 |
21 |
22 | createTicketBtn.addEventListener('click', createTicket );
23 |
24 |
25 | getLastTicket();
26 |
27 |
28 |
29 |
--------------------------------------------------------------------------------
/public/js/public.js:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | function renderTickets( tickets = []) {
5 |
6 |
7 | for( let i = 0; i< tickets.length; i++ ) {
8 | if ( i >=4 ) break;
9 |
10 | const ticket = tickets[i];
11 | if ( !ticket ) continue;
12 |
13 | const lblTicket = document.querySelector(`#lbl-ticket-0${ i + 1 }`);
14 | const lblDesk = document.querySelector(`#lbl-desk-0${ i + 1 }`);
15 |
16 | lblTicket.innerText = `Ticket ${ ticket.number }`;
17 | lblDesk.innerText = ticket.handleAtDesk
18 |
19 |
20 | }
21 |
22 |
23 | }
24 | async function loadCurrentTickets() {
25 | const tickets = await fetch('/api/ticket/working-on').then(resp => resp.json());
26 | renderTickets(tickets);
27 |
28 | }
29 |
30 |
31 |
32 | function connectToWebSockets() {
33 |
34 | const socket = new WebSocket( 'ws://localhost:3000/ws' );
35 |
36 | socket.onmessage = ( event ) => {
37 | const { type, payload } = JSON.parse( event.data );
38 | if ( type !== 'on-working-changed' ) return;
39 | renderTickets(payload);
40 | };
41 |
42 | socket.onclose = ( event ) => {
43 |
44 | setTimeout( () => {
45 | connectToWebSockets();
46 | }, 1500 );
47 |
48 | };
49 |
50 | socket.onopen = ( event ) => {
51 | console.log( 'Connected' );
52 | };
53 |
54 | }
55 |
56 |
57 |
58 |
59 | loadCurrentTickets();
60 | connectToWebSockets();
--------------------------------------------------------------------------------
/public/js/socket-client.js:
--------------------------------------------------------------------------------
1 |
2 |
3 | function connectToWebSockets() {
4 |
5 | const socket = new WebSocket( 'ws://localhost:3000/ws' );
6 |
7 | socket.onmessage = ( event ) => {
8 | console.log(event.data);
9 | };
10 |
11 | socket.onclose = ( event ) => {
12 | console.log( 'Connection closed' );
13 | setTimeout( () => {
14 | console.log( 'retrying to connect' );
15 | connectToWebSockets();
16 | }, 1500 );
17 |
18 | };
19 |
20 | socket.onopen = ( event ) => {
21 | console.log( 'Connected' );
22 | };
23 |
24 | }
25 |
26 | connectToWebSockets();
27 |
28 |
--------------------------------------------------------------------------------
/public/new-ticket.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | Nuevo Ticket
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 | Cargando...
21 |
22 | |
23 |
24 |
25 |
26 |
29 | |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
--------------------------------------------------------------------------------
/public/public.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | Sistema de colas
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 | Ticket W
21 |
22 | Escritorio W
23 | |
24 |
25 |
26 |
27 |
28 | Ticket X
29 |
30 | Escritorio X
31 | |
32 |
33 |
34 |
35 | Ticket Y
36 |
37 | Escritorio Y
38 | |
39 |
40 |
41 |
42 | Ticket Z
43 |
44 | Escritorio Z
45 | |
46 |
47 |
48 | |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
--------------------------------------------------------------------------------
/src/app.ts:
--------------------------------------------------------------------------------
1 | import { createServer } from 'http';
2 | import { envs } from './config/envs';
3 | import { AppRoutes } from './presentation/routes';
4 | import { Server } from './presentation/server';
5 | import { WssService } from './presentation/services/wss.service';
6 |
7 |
8 | (async()=> {
9 | main();
10 | })();
11 |
12 |
13 | function main() {
14 |
15 | const server = new Server({
16 | port: envs.PORT,
17 | });
18 |
19 | const httpServer = createServer( server.app );
20 | WssService.initWss({ server: httpServer });
21 |
22 |
23 | server.setRoutes( AppRoutes.routes );
24 |
25 |
26 |
27 | httpServer.listen( envs.PORT, () => {
28 | console.log(`Server running on port: ${ envs.PORT }`);
29 | })
30 | }
--------------------------------------------------------------------------------
/src/config/envs.ts:
--------------------------------------------------------------------------------
1 | import 'dotenv/config';
2 | import { get } from 'env-var';
3 |
4 |
5 | export const envs = {
6 |
7 | PORT: get('PORT').required().asPortNumber(),
8 |
9 | }
10 |
11 |
12 |
13 |
--------------------------------------------------------------------------------
/src/config/uuid.adapter.ts:
--------------------------------------------------------------------------------
1 | import { v4 as uuidv4 } from 'uuid';
2 |
3 | export class UuidAdapter {
4 |
5 | public static v4() {
6 | return uuidv4();
7 | }
8 |
9 |
10 | }
--------------------------------------------------------------------------------
/src/domain/interfaces/ticket.ts:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | export interface Ticket {
5 | id: string;
6 | number: number;
7 | createdAt: Date;
8 | handleAtDesk?: string; // Escritorio 1,
9 | handleAt?: Date;
10 | done: boolean;
11 | }
--------------------------------------------------------------------------------
/src/presentation/routes.ts:
--------------------------------------------------------------------------------
1 | import { Router } from 'express';
2 | import { TicketRoutes } from './tickets/routes';
3 |
4 |
5 |
6 |
7 | export class AppRoutes {
8 |
9 |
10 | static get routes(): Router {
11 |
12 | const router = Router();
13 |
14 | // Definir las rutas
15 | // router.use('/api/todos', /*TodoRoutes.routes */ );
16 |
17 | router.use('/api/ticket', TicketRoutes.routes );
18 |
19 |
20 |
21 | return router;
22 | }
23 |
24 |
25 | }
26 |
27 |
--------------------------------------------------------------------------------
/src/presentation/server.ts:
--------------------------------------------------------------------------------
1 | import express, { Router } from 'express';
2 | import path from 'path';
3 |
4 | interface Options {
5 | port: number;
6 | // routes: Router;
7 | public_path?: string;
8 | }
9 |
10 |
11 | export class Server {
12 |
13 | public readonly app = express();
14 | private serverListener?: any;
15 | private readonly port: number;
16 | private readonly publicPath: string;
17 | // private readonly routes: Router;
18 |
19 | constructor(options: Options) {
20 | const { port, public_path = 'public' } = options;
21 | this.port = port;
22 | this.publicPath = public_path;
23 |
24 | this.configure();
25 | }
26 |
27 | private configure() {
28 |
29 |
30 | //* Middlewares
31 | this.app.use( express.json() ); // raw
32 | this.app.use( express.urlencoded({ extended: true }) ); // x-www-form-urlencoded
33 |
34 | //* Public Folder
35 | this.app.use( express.static( this.publicPath ) );
36 |
37 | //* Routes
38 | // this.app.use( this.routes );
39 |
40 | //* SPA
41 | this.app.get(/^\/(?!api).*/, (req, res) => {
42 | const indexPath = path.join( __dirname + `../../../${ this.publicPath }/index.html` );
43 | res.sendFile(indexPath);
44 | });
45 |
46 | }
47 |
48 | public setRoutes( router: Router ) {
49 | this.app.use(router);
50 | }
51 |
52 |
53 | async start() {
54 |
55 | this.serverListener = this.app.listen(this.port, () => {
56 | console.log(`Server running on port ${ this.port }`);
57 | });
58 |
59 | }
60 |
61 | public close() {
62 | this.serverListener?.close();
63 | }
64 |
65 | }
--------------------------------------------------------------------------------
/src/presentation/services/ticket.service.ts:
--------------------------------------------------------------------------------
1 | import { UuidAdapter } from '../../config/uuid.adapter';
2 | import { Ticket } from '../../domain/interfaces/ticket';
3 | import { WssService } from './wss.service';
4 |
5 |
6 |
7 | export class TicketService {
8 |
9 | constructor(
10 | private readonly wssService = WssService.instance,
11 | ) {}
12 |
13 |
14 | public tickets: Ticket [] = [
15 | { id: UuidAdapter.v4(), number: 1, createdAt: new Date(), done: false },
16 | { id: UuidAdapter.v4(), number: 2, createdAt: new Date(), done: false },
17 | { id: UuidAdapter.v4(), number: 3, createdAt: new Date(), done: false },
18 | { id: UuidAdapter.v4(), number: 4, createdAt: new Date(), done: false },
19 | { id: UuidAdapter.v4(), number: 5, createdAt: new Date(), done: false },
20 | { id: UuidAdapter.v4(), number: 6, createdAt: new Date(), done: false },
21 | ];
22 |
23 | private readonly workingOnTickets: Ticket[] = [];
24 |
25 | public get pendingTickets():Ticket[] {
26 | return this.tickets.filter( ticket => !ticket.handleAtDesk );
27 | }
28 |
29 | public get lastWorkingOnTickets():Ticket[] {
30 | return this.workingOnTickets.slice(0,4);
31 | }
32 |
33 | public get lastTicketNumber(): number {
34 | return this.tickets.length > 0 ? this.tickets.at(-1)!.number : 0;
35 | }
36 |
37 | public createTicket() {
38 |
39 | const ticket: Ticket = {
40 | id: UuidAdapter.v4(),
41 | number: this.lastTicketNumber + 1,
42 | createdAt: new Date(),
43 | done: false,
44 | handleAt: undefined,
45 | handleAtDesk: undefined,
46 | }
47 |
48 | this.tickets.push(ticket);
49 | this.onTicketNumberChanged();
50 |
51 | return ticket;
52 | }
53 |
54 | public drawTicket(desk: string) {
55 |
56 | const ticket = this.tickets.find( t => !t.handleAtDesk );
57 | if ( !ticket ) return { status: 'error', message: 'No hay tickets pendientes' };
58 |
59 | ticket.handleAtDesk = desk;
60 | ticket.handleAt = new Date();
61 |
62 |
63 | this.workingOnTickets.unshift({...ticket});
64 | this.onTicketNumberChanged();
65 | this.onWorkingOnChanged();
66 |
67 |
68 | return { status: 'ok', ticket }
69 |
70 | }
71 |
72 | public onFinishedTicket( id: string ) {
73 | const ticket = this.tickets.find( t => t.id === id );
74 | if ( !ticket ) return { status: 'error', message: 'Ticket no encontrado' };
75 |
76 | this.tickets = this.tickets.map( ticket => {
77 |
78 | if ( ticket.id === id ) {
79 | ticket.done = true;
80 | }
81 |
82 | return ticket;
83 | });
84 |
85 | return { status: 'ok' }
86 | }
87 |
88 | private onTicketNumberChanged() {
89 | this.wssService.sendMessage('on-ticket-count-changed', this.pendingTickets.length );
90 | }
91 |
92 | private onWorkingOnChanged() {
93 | this.wssService.sendMessage('on-working-changed', this.lastWorkingOnTickets );
94 | }
95 |
96 |
97 | }
--------------------------------------------------------------------------------
/src/presentation/services/wss.service.ts:
--------------------------------------------------------------------------------
1 | import { Server } from 'http';
2 | import { WebSocket, WebSocketServer} from 'ws';
3 |
4 | interface Options {
5 | server: Server;
6 | path?: string; // ws
7 | }
8 |
9 |
10 | export class WssService {
11 |
12 | private static _instance: WssService;
13 | private wss: WebSocketServer;
14 |
15 | private constructor( options: Options ) {
16 | const { server, path = '/ws' } = options; /// ws://localhost:3000/ws
17 |
18 | this.wss = new WebSocketServer({ server, path });
19 | this.start();
20 | }
21 |
22 | static get instance(): WssService {
23 | if ( !WssService._instance ) {
24 | throw 'WssService is not initialized';
25 | }
26 |
27 | return WssService._instance;
28 | }
29 |
30 | static initWss( options: Options ) {
31 | WssService._instance = new WssService(options);
32 | }
33 |
34 |
35 | public sendMessage( type: string, payload: Object ) {
36 | this.wss.clients.forEach( client => {
37 | if ( client.readyState === WebSocket.OPEN ) {
38 | client.send( JSON.stringify({ type, payload }) );
39 | }
40 | })
41 | }
42 |
43 |
44 | public start() {
45 |
46 | this.wss.on('connection', (ws: WebSocket ) => {
47 |
48 | console.log('Client connected');
49 |
50 | ws.on('close', () => console.log('Client disconnected') )
51 |
52 | });
53 |
54 | }
55 |
56 | }
--------------------------------------------------------------------------------
/src/presentation/tickets/controller.ts:
--------------------------------------------------------------------------------
1 | import { Request, Response } from 'express';
2 | import { TicketService } from '../services/ticket.service';
3 |
4 |
5 |
6 | export class TicketController {
7 |
8 | // DI - WssService
9 | constructor(
10 | private readonly ticketService = new TicketService(),
11 | ) {}
12 |
13 |
14 | public getTickets = async (req: Request, res: Response) => {
15 | res.json( this.ticketService.tickets );
16 | }
17 |
18 | public getLastTicketNumber = async (req: Request, res: Response) => {
19 | res.json( this.ticketService.lastTicketNumber );
20 | }
21 |
22 | public pendingTickets = async (req: Request, res: Response) => {
23 | res.json( this.ticketService.pendingTickets );
24 | }
25 |
26 | public createTicket = async (req: Request, res: Response) => {
27 | res.status(201).json( this.ticketService.createTicket() );
28 | }
29 |
30 | public drawTicket = async (req: Request, res: Response) => {
31 | const { desk } = req.params;
32 | res.json( this.ticketService.drawTicket(desk) );
33 | }
34 |
35 | public ticketFinished = async (req: Request, res: Response) => {
36 | const { ticketId } = req.params;
37 |
38 | res.json( this.ticketService.onFinishedTicket(ticketId) );
39 | }
40 |
41 | public workingOn = async (req: Request, res: Response) => {
42 | res.json( this.ticketService.lastWorkingOnTickets );
43 | }
44 |
45 |
46 |
47 | }
--------------------------------------------------------------------------------
/src/presentation/tickets/routes.ts:
--------------------------------------------------------------------------------
1 | import { Router } from 'express';
2 | import { TicketController } from './controller';
3 |
4 |
5 |
6 |
7 | export class TicketRoutes {
8 |
9 |
10 | static get routes() {
11 |
12 | const router = Router();
13 | const ticketController = new TicketController();
14 |
15 |
16 | router.get('/', ticketController.getTickets );
17 | router.get('/last', ticketController.getLastTicketNumber );
18 | router.get('/pending', ticketController.pendingTickets );
19 |
20 |
21 | router.post('/', ticketController.createTicket);
22 |
23 | router.get('/draw/:desk', ticketController.drawTicket);
24 | router.put('/done/:ticketId', ticketController.ticketFinished);
25 |
26 | router.get('/working-on', ticketController.workingOn );
27 |
28 |
29 |
30 |
31 |
32 | return router;
33 | }
34 |
35 |
36 | }
--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | /* Visit https://aka.ms/tsconfig to read more about this file */
4 |
5 | /* Projects */
6 | // "incremental": true, /* Save .tsbuildinfo files to allow for incremental compilation of projects. */
7 | // "composite": true, /* Enable constraints that allow a TypeScript project to be used with project references. */
8 | // "tsBuildInfoFile": "./.tsbuildinfo", /* Specify the path to .tsbuildinfo incremental compilation file. */
9 | // "disableSourceOfProjectReferenceRedirect": true, /* Disable preferring source files instead of declaration files when referencing composite projects. */
10 | // "disableSolutionSearching": true, /* Opt a project out of multi-project reference checking when editing. */
11 | // "disableReferencedProjectLoad": true, /* Reduce the number of projects loaded automatically by TypeScript. */
12 |
13 | /* Language and Environment */
14 | "target": "es2016", /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */
15 | // "lib": [], /* Specify a set of bundled library declaration files that describe the target runtime environment. */
16 | // "jsx": "preserve", /* Specify what JSX code is generated. */
17 | // "experimentalDecorators": true, /* Enable experimental support for legacy experimental decorators. */
18 | // "emitDecoratorMetadata": true, /* Emit design-type metadata for decorated declarations in source files. */
19 | // "jsxFactory": "", /* Specify the JSX factory function used when targeting React JSX emit, e.g. 'React.createElement' or 'h'. */
20 | // "jsxFragmentFactory": "", /* Specify the JSX Fragment reference used for fragments when targeting React JSX emit e.g. 'React.Fragment' or 'Fragment'. */
21 | // "jsxImportSource": "", /* Specify module specifier used to import the JSX factory functions when using 'jsx: react-jsx*'. */
22 | // "reactNamespace": "", /* Specify the object invoked for 'createElement'. This only applies when targeting 'react' JSX emit. */
23 | // "noLib": true, /* Disable including any library files, including the default lib.d.ts. */
24 | // "useDefineForClassFields": true, /* Emit ECMAScript-standard-compliant class fields. */
25 | // "moduleDetection": "auto", /* Control what method is used to detect module-format JS files. */
26 |
27 | /* Modules */
28 | "module": "commonjs", /* Specify what module code is generated. */
29 | "rootDir": "src", /* Specify the root folder within your source files. */
30 | // "moduleResolution": "node10", /* Specify how TypeScript looks up a file from a given module specifier. */
31 | // "baseUrl": "./", /* Specify the base directory to resolve non-relative module names. */
32 | // "paths": {}, /* Specify a set of entries that re-map imports to additional lookup locations. */
33 | // "rootDirs": [], /* Allow multiple folders to be treated as one when resolving modules. */
34 | // "typeRoots": [], /* Specify multiple folders that act like './node_modules/@types'. */
35 | // "types": [], /* Specify type package names to be included without being referenced in a source file. */
36 | // "allowUmdGlobalAccess": true, /* Allow accessing UMD globals from modules. */
37 | // "moduleSuffixes": [], /* List of file name suffixes to search when resolving a module. */
38 | // "allowImportingTsExtensions": true, /* Allow imports to include TypeScript file extensions. Requires '--moduleResolution bundler' and either '--noEmit' or '--emitDeclarationOnly' to be set. */
39 | // "resolvePackageJsonExports": true, /* Use the package.json 'exports' field when resolving package imports. */
40 | // "resolvePackageJsonImports": true, /* Use the package.json 'imports' field when resolving imports. */
41 | // "customConditions": [], /* Conditions to set in addition to the resolver-specific defaults when resolving imports. */
42 | // "resolveJsonModule": true, /* Enable importing .json files. */
43 | // "allowArbitraryExtensions": true, /* Enable importing files with any extension, provided a declaration file is present. */
44 | // "noResolve": true, /* Disallow 'import's, 'require's or ''s from expanding the number of files TypeScript should add to a project. */
45 |
46 | /* JavaScript Support */
47 | // "allowJs": true, /* Allow JavaScript files to be a part of your program. Use the 'checkJS' option to get errors from these files. */
48 | // "checkJs": true, /* Enable error reporting in type-checked JavaScript files. */
49 | // "maxNodeModuleJsDepth": 1, /* Specify the maximum folder depth used for checking JavaScript files from 'node_modules'. Only applicable with 'allowJs'. */
50 |
51 | /* Emit */
52 | // "declaration": true, /* Generate .d.ts files from TypeScript and JavaScript files in your project. */
53 | // "declarationMap": true, /* Create sourcemaps for d.ts files. */
54 | // "emitDeclarationOnly": true, /* Only output d.ts files and not JavaScript files. */
55 | // "sourceMap": true, /* Create source map files for emitted JavaScript files. */
56 | // "inlineSourceMap": true, /* Include sourcemap files inside the emitted JavaScript. */
57 | // "outFile": "./", /* Specify a file that bundles all outputs into one JavaScript file. If 'declaration' is true, also designates a file that bundles all .d.ts output. */
58 | "outDir": "dist/", /* Specify an output folder for all emitted files. */
59 | // "removeComments": true, /* Disable emitting comments. */
60 | // "noEmit": true, /* Disable emitting files from a compilation. */
61 | // "importHelpers": true, /* Allow importing helper functions from tslib once per project, instead of including them per-file. */
62 | // "importsNotUsedAsValues": "remove", /* Specify emit/checking behavior for imports that are only used for types. */
63 | // "downlevelIteration": true, /* Emit more compliant, but verbose and less performant JavaScript for iteration. */
64 | // "sourceRoot": "", /* Specify the root path for debuggers to find the reference source code. */
65 | // "mapRoot": "", /* Specify the location where debugger should locate map files instead of generated locations. */
66 | // "inlineSources": true, /* Include source code in the sourcemaps inside the emitted JavaScript. */
67 | // "emitBOM": true, /* Emit a UTF-8 Byte Order Mark (BOM) in the beginning of output files. */
68 | // "newLine": "crlf", /* Set the newline character for emitting files. */
69 | // "stripInternal": true, /* Disable emitting declarations that have '@internal' in their JSDoc comments. */
70 | // "noEmitHelpers": true, /* Disable generating custom helper functions like '__extends' in compiled output. */
71 | // "noEmitOnError": true, /* Disable emitting files if any type checking errors are reported. */
72 | // "preserveConstEnums": true, /* Disable erasing 'const enum' declarations in generated code. */
73 | // "declarationDir": "./", /* Specify the output directory for generated declaration files. */
74 | // "preserveValueImports": true, /* Preserve unused imported values in the JavaScript output that would otherwise be removed. */
75 |
76 | /* Interop Constraints */
77 | // "isolatedModules": true, /* Ensure that each file can be safely transpiled without relying on other imports. */
78 | // "verbatimModuleSyntax": true, /* Do not transform or elide any imports or exports not marked as type-only, ensuring they are written in the output file's format based on the 'module' setting. */
79 | // "allowSyntheticDefaultImports": true, /* Allow 'import x from y' when a module doesn't have a default export. */
80 | "esModuleInterop": true, /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */
81 | // "preserveSymlinks": true, /* Disable resolving symlinks to their realpath. This correlates to the same flag in node. */
82 | "forceConsistentCasingInFileNames": true, /* Ensure that casing is correct in imports. */
83 |
84 | /* Type Checking */
85 | "strict": true, /* Enable all strict type-checking options. */
86 | // "noImplicitAny": true, /* Enable error reporting for expressions and declarations with an implied 'any' type. */
87 | // "strictNullChecks": true, /* When type checking, take into account 'null' and 'undefined'. */
88 | // "strictFunctionTypes": true, /* When assigning functions, check to ensure parameters and the return values are subtype-compatible. */
89 | // "strictBindCallApply": true, /* Check that the arguments for 'bind', 'call', and 'apply' methods match the original function. */
90 | // "strictPropertyInitialization": true, /* Check for class properties that are declared but not set in the constructor. */
91 | // "noImplicitThis": true, /* Enable error reporting when 'this' is given the type 'any'. */
92 | // "useUnknownInCatchVariables": true, /* Default catch clause variables as 'unknown' instead of 'any'. */
93 | // "alwaysStrict": true, /* Ensure 'use strict' is always emitted. */
94 | // "noUnusedLocals": true, /* Enable error reporting when local variables aren't read. */
95 | // "noUnusedParameters": true, /* Raise an error when a function parameter isn't read. */
96 | // "exactOptionalPropertyTypes": true, /* Interpret optional property types as written, rather than adding 'undefined'. */
97 | // "noImplicitReturns": true, /* Enable error reporting for codepaths that do not explicitly return in a function. */
98 | // "noFallthroughCasesInSwitch": true, /* Enable error reporting for fallthrough cases in switch statements. */
99 | // "noUncheckedIndexedAccess": true, /* Add 'undefined' to a type when accessed using an index. */
100 | // "noImplicitOverride": true, /* Ensure overriding members in derived classes are marked with an override modifier. */
101 | // "noPropertyAccessFromIndexSignature": true, /* Enforces using indexed accessors for keys declared using an indexed type. */
102 | // "allowUnusedLabels": true, /* Disable error reporting for unused labels. */
103 | // "allowUnreachableCode": true, /* Disable error reporting for unreachable code. */
104 |
105 | /* Completeness */
106 | // "skipDefaultLibCheck": true, /* Skip type checking .d.ts files that are included with TypeScript. */
107 | "skipLibCheck": true /* Skip type checking all .d.ts files. */
108 | }
109 | }
110 |
--------------------------------------------------------------------------------