├── .dockerignore ├── .editorconfig ├── .eslintrc.cjs ├── .gitignore ├── Dockerfile ├── LICENSE ├── README.md ├── compose-demo.yaml ├── compose-dev.yaml ├── compose.yaml ├── config.ts ├── package-lock.json ├── package.json ├── pool.ts ├── public └── demo-kuma │ ├── main.css │ └── main.scss ├── server.ts ├── tsconfig.json ├── util.ts └── views └── index.ejs /.dockerignore: -------------------------------------------------------------------------------- 1 | # Should be identical to .gitignore 2 | .env 3 | node_modules 4 | .idea 5 | tmp 6 | /private 7 | 8 | # Docker extra 9 | .editorconfig 10 | .env.sample 11 | .eslintrc.cjs 12 | .git 13 | .gitignore 14 | README.md 15 | compose.yaml 16 | Dockerfile 17 | -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [*] 4 | indent_style = space 5 | indent_size = 4 6 | end_of_line = lf 7 | charset = utf-8 8 | trim_trailing_whitespace = true 9 | insert_final_newline = true 10 | 11 | [*.md] 12 | trim_trailing_whitespace = false 13 | 14 | [*.yaml] 15 | indent_size = 2 16 | 17 | [*.yml] 18 | indent_size = 2 19 | 20 | [*.vue] 21 | trim_trailing_whitespace = false 22 | 23 | [*.go] 24 | indent_style = tab 25 | -------------------------------------------------------------------------------- /.eslintrc.cjs: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | root: true, 3 | env: { 4 | browser: true, 5 | node: true, 6 | }, 7 | extends: [ 8 | "eslint:recommended", 9 | "plugin:@typescript-eslint/recommended", 10 | ], 11 | parser: "@typescript-eslint/parser", 12 | plugins: [ 13 | "@typescript-eslint", 14 | "jsdoc" 15 | ], 16 | rules: { 17 | "yoda": "error", 18 | "linebreak-style": [ "error", "unix" ], 19 | "camelcase": [ "warn", { 20 | "properties": "never", 21 | "ignoreImports": true 22 | }], 23 | "no-unused-vars": [ "warn", { 24 | "args": "none" 25 | }], 26 | indent: [ 27 | "error", 28 | 4, 29 | { 30 | ignoredNodes: [ "TemplateLiteral" ], 31 | SwitchCase: 1, 32 | }, 33 | ], 34 | quotes: [ "error", "double" ], 35 | semi: "error", 36 | "no-multi-spaces": [ "error", { 37 | ignoreEOLComments: true, 38 | }], 39 | "array-bracket-spacing": [ "warn", "always", { 40 | "singleValue": true, 41 | "objectsInArrays": false, 42 | "arraysInArrays": false 43 | }], 44 | "space-before-function-paren": [ "error", { 45 | "anonymous": "always", 46 | "named": "never", 47 | "asyncArrow": "always" 48 | }], 49 | "curly": "error", 50 | "object-curly-spacing": [ "error", "always" ], 51 | "object-curly-newline": "off", 52 | "object-property-newline": "error", 53 | "comma-spacing": "error", 54 | "brace-style": "error", 55 | "no-var": "error", 56 | "key-spacing": "warn", 57 | "keyword-spacing": "warn", 58 | "space-infix-ops": "error", 59 | "arrow-spacing": "warn", 60 | "no-trailing-spaces": "error", 61 | "no-constant-condition": [ "error", { 62 | "checkLoops": false, 63 | }], 64 | "space-before-blocks": "warn", 65 | "no-extra-boolean-cast": "off", 66 | "no-multiple-empty-lines": [ "warn", { 67 | "max": 1, 68 | "maxBOF": 0, 69 | }], 70 | "lines-between-class-members": [ "warn", "always", { 71 | exceptAfterSingleLine: true, 72 | }], 73 | "no-unneeded-ternary": "error", 74 | "array-bracket-newline": [ "error", "consistent" ], 75 | "eol-last": [ "error", "always" ], 76 | "comma-dangle": [ "warn", "only-multiline" ], 77 | "no-empty": [ "error", { 78 | "allowEmptyCatch": true 79 | }], 80 | "no-control-regex": "off", 81 | "one-var": [ "error", "never" ], 82 | "max-statements-per-line": [ "error", { "max": 1 }], 83 | "@typescript-eslint/ban-ts-comment": "off", 84 | "@typescript-eslint/no-unused-vars": [ "warn", { 85 | "args": "none" 86 | }], 87 | "prefer-const" : "off", 88 | }, 89 | }; 90 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .idea 2 | .env 3 | 4 | # Logs 5 | logs 6 | *.log 7 | npm-debug.log* 8 | yarn-debug.log* 9 | yarn-error.log* 10 | lerna-debug.log* 11 | 12 | # Diagnostic reports (https://nodejs.org/api/report.html) 13 | report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json 14 | 15 | # Runtime data 16 | pids 17 | *.pid 18 | *.seed 19 | *.pid.lock 20 | 21 | # Directory for instrumented libs generated by jscoverage/JSCover 22 | lib-cov 23 | 24 | # Coverage directory used by tools like istanbul 25 | coverage 26 | *.lcov 27 | 28 | # nyc test coverage 29 | .nyc_output 30 | 31 | # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) 32 | .grunt 33 | 34 | # Bower dependency directory (https://bower.io/) 35 | bower_components 36 | 37 | # node-waf configuration 38 | .lock-wscript 39 | 40 | # Compiled binary addons (https://nodejs.org/api/addons.html) 41 | build/Release 42 | 43 | # Dependency directories 44 | node_modules/ 45 | jspm_packages/ 46 | 47 | # TypeScript v1 declaration files 48 | typings/ 49 | 50 | # TypeScript cache 51 | *.tsbuildinfo 52 | 53 | # Optional npm cache directory 54 | .npm 55 | 56 | # Optional eslint cache 57 | .eslintcache 58 | 59 | # Microbundle cache 60 | .rpt2_cache/ 61 | .rts2_cache_cjs/ 62 | .rts2_cache_es/ 63 | .rts2_cache_umd/ 64 | 65 | # Optional REPL history 66 | .node_repl_history 67 | 68 | # Output of 'npm pack' 69 | *.tgz 70 | 71 | # Yarn Integrity file 72 | .yarn-integrity 73 | 74 | # dotenv environment variables file 75 | .env 76 | .env.test 77 | 78 | # parcel-bundler cache (https://parceljs.org/) 79 | .cache 80 | 81 | # Next.js build output 82 | .next 83 | 84 | # Nuxt.js build / generate output 85 | .nuxt 86 | dist 87 | 88 | # Gatsby files 89 | .cache/ 90 | # Comment in the public line in if your project uses Gatsby and *not* Next.js 91 | # https://nextjs.org/blog/next-9-1#public-directory-support 92 | # public 93 | 94 | # vuepress build output 95 | .vuepress/dist 96 | 97 | # Serverless directories 98 | .serverless/ 99 | 100 | # FuseBox cache 101 | .fusebox/ 102 | 103 | # DynamoDB Local files 104 | .dynamodb/ 105 | 106 | # TernJS port file 107 | .tern-port 108 | 109 | /private 110 | -------------------------------------------------------------------------------- /Dockerfile: -------------------------------------------------------------------------------- 1 | ############################################ 2 | # ⭐ Main Image 3 | ############################################ 4 | FROM louislam/dockge:base AS release 5 | WORKDIR /app 6 | COPY --chown=node:node . . 7 | RUN npm ci --production 8 | EXPOSE 80 9 | ENTRYPOINT ["/usr/bin/dumb-init", "--"] 10 | CMD ["tsx", "./server.ts"] 11 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2021 Louis Lam 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 | # Demo Kuma 2 | 3 | Create a demo site for your Docker based web application. 4 | 5 | Live Demo (Uptime Kuma): 6 | https://demo.kuma.pet/start-demo 7 | 8 | ExerciseDiary (2 mins demo) - Another selfhosted app which I found on Reddit: 9 | https://exercisediary-demo.kuma.pet/start-demo 10 | 11 | ![image](https://github.com/louislam/demo-kuma/assets/1336778/f15b5e46-5c98-419e-98e4-a2d52b91780c) 12 | 13 | 14 | ## Features 15 | 16 | - Quickly create a demo for your project. 17 | - Spin up a docker stack when requested, shut down the stack when time is up. 18 | - A countdown timer at the bottom right corner. 19 | - Custom demo duration. 20 | - Portless demo instances design in v2, you just need one port for Demo Kuma. 21 | 22 | ## How to use 23 | 24 | 1. Create a stack directory `demo-kuma`. 25 | 1. Download `compose.yaml` and `compose-demo.yaml` into the directory. 26 | - `compose.yaml` = Demo Kuma stack 27 | - You should go through all variables in the `environment:` section. 28 | - `compose-demo.yaml` = The stack that you want to provide a demo 29 | - By default, the main service should be `main` and the network name should be `demo-kuma_default`. 30 | - If your stack name is not `demo-kuma`, you should change the network name to `{your-name}_default` 31 | 1. `docker compose up -d`. 32 | 1. Go to http://localhost:3003/start-demo to test the demo. 33 | 34 | ## How it works? 35 | 36 | 1. Demo Kuma takes control of your Docker 37 | 1. User requests a demo via a browser 38 | 1. Demo Kuma assign a session ID for this request and spin up the stack `compose-demo.yaml` 39 | 1. Once the demo stack is started, Demo Kuma will act as a reverse proxy to communicate between the browser and the demo stack. 40 | 1. The timer will be created at the same time. When time is up, Demo Kuma will shut down the stack. 41 | 42 | # More screenshots 43 | 44 | ![image](https://github.com/louislam/demo-kuma/assets/1336778/c264c86a-0718-42af-a91b-20db061af7db) 45 | 46 | -------------------------------------------------------------------------------- /compose-demo.yaml: -------------------------------------------------------------------------------- 1 | services: 2 | # Main Service that you want to create a demo 3 | main: 4 | image: louislam/uptime-kuma:1 5 | restart: no 6 | 7 | # The network name should be the same as the one in compose.yaml, so that demo-kuma can find it and access it 8 | # You should keep this 9 | networks: 10 | - demo-kuma_default 11 | 12 | networks: 13 | # The network name should be the same as the one in compose.yaml, so that demo-kuma can find it and access it 14 | # You should keep this 15 | demo-kuma_default: 16 | external: true 17 | -------------------------------------------------------------------------------- /compose-dev.yaml: -------------------------------------------------------------------------------- 1 | services: 2 | demo-kuma: 3 | image: louislam/demo-kuma:nightly 4 | restart: unless-stopped 5 | ports: 6 | - 3003:80 7 | command: tsx watch --inspect ./server.ts 8 | volumes: 9 | - ./:/app 10 | - /var/run/docker.sock:/var/run/docker.sock 11 | environment: 12 | - NODE_ENV=development 13 | 14 | # Demo Website Name 15 | - WEBSITE_NAME=Uptime Kuma Demo 16 | 17 | # Stack Prefix (Should a special prefix, because demo-kuma will manage (start/shutdown) all stacks with this) 18 | - STACK_PREFIX=uptime-kuma-demo 19 | 20 | # Specify the service name of the stack 21 | - STACK_MAIN_SERVICE_NAME=main 22 | 23 | # Specify the port of the service 24 | - STACK_MAIN_SERVICE_PORT=3001 25 | 26 | # The entry path of the demo 27 | # Should be start with / 28 | - STACK_MAIN_SERVICE_ENTRY_PATH=/ 29 | 30 | # Session Time (Seconds) 31 | - SESSION_TIME=60 32 | 33 | # If the stack has not started in time, it will be considered as failed (Seconds) 34 | - START_TIMEOUT=10 35 | 36 | # Show Entry 37 | # If true, the entry will be shown in the demo page and the user has to click "Start Demo" to start the demo 38 | # true / false 39 | - SHOW_ENTRY=false 40 | 41 | # Install URL for the "Install now" button 42 | - INSTALL_URL=https://github.com/louislam/uptime-kuma 43 | 44 | # Docker Network Name 45 | # The demo instance must be in the same network as the demo-kuma 46 | - DOCKER_NETWORK_NAME=demo-kuma_default 47 | -------------------------------------------------------------------------------- /compose.yaml: -------------------------------------------------------------------------------- 1 | services: 2 | demo-kuma: 3 | image: louislam/demo-kuma:2 4 | restart: unless-stopped 5 | ports: 6 | - 3003:80 7 | volumes: 8 | - /var/run/docker.sock:/var/run/docker.sock 9 | - ./compose-demo.yaml:/app/compose-demo.yaml 10 | environment: 11 | # Demo Website Name 12 | - WEBSITE_NAME=Uptime Kuma Demo 13 | 14 | # Stack Prefix (Should a special prefix, because demo-kuma will manage (start/shutdown) all stacks with this) 15 | - STACK_PREFIX=uptime-kuma-demo 16 | 17 | # Specify the service name of the stack 18 | - STACK_MAIN_SERVICE_NAME=main 19 | 20 | # Specify the port of the service 21 | - STACK_MAIN_SERVICE_PORT=3001 22 | 23 | # The entry path of the demo 24 | # Should be start with / 25 | - STACK_MAIN_SERVICE_ENTRY_PATH=/ 26 | 27 | # Session Time (Seconds) 28 | - SESSION_TIME=600 29 | 30 | # If the stack has not started in time, it will be considered as failed (Seconds) 31 | - START_TIMEOUT=60 32 | 33 | # Show Entry 34 | # If true, the entry will be shown in the demo page and the user has to click "Start Demo" to start the demo 35 | # true / false 36 | - SHOW_ENTRY=false 37 | 38 | # Install URL for the "Install now" button 39 | - INSTALL_URL=https://github.com/louislam/uptime-kuma 40 | 41 | # Docker Network Name 42 | # The demo instance must be in the same network as the demo-kuma 43 | - DOCKER_NETWORK_NAME=demo-kuma_default 44 | 45 | # (Optional) Cloudflare Tunnel as a reverse proxy 46 | # Remove this if you don't need it 47 | # cloudflared: 48 | # image: cloudflare/cloudflared:latest 49 | # command: tunnel --no-autoupdate run --token ${TUNNEL_TOKEN} 50 | # restart: unless-stopped 51 | 52 | 53 | -------------------------------------------------------------------------------- /config.ts: -------------------------------------------------------------------------------- 1 | export const serverPort = parseInt(process.env.SERVER_PORT) || 80; 2 | export const stackPrefix = process.env.STACK_PREFIX || "demo"; 3 | export const serviceName = process.env.STACK_MAIN_SERVICE_NAME || "main"; 4 | export const servicePort = parseInt(process.env.STACK_MAIN_SERVICE_PORT) || 80; 5 | export const entryPath = process.env.STACK_MAIN_SERVICE_ENTRY_PATH || "/"; 6 | export const dockerNetwork = process.env.DOCKER_NETWORK_NAME || "demo-kuma"; 7 | export const websiteName = process.env.WEBSITE_NAME || "Demo"; 8 | export const sessionTime = parseInt(process.env.SESSION_TIME) || 600; 9 | export const startTimeout = parseInt(process.env.START_TIMEOUT) || 60; 10 | export const installURL = process.env.INSTALL_URL; 11 | export const showEntry = (process.env.SHOW_ENTRY === "true"); 12 | 13 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "demo-kuma", 3 | "version": "2.0.0", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "demo-kuma", 9 | "version": "2.0.0", 10 | "license": "MIT", 11 | "dependencies": { 12 | "dotenv": "~10.0.0", 13 | "ejs": "~3.1.9", 14 | "http-graceful-shutdown": "~3.1.13", 15 | "http-proxy": "~1.18.1", 16 | "promisify-child-process": "~4.1.2", 17 | "tsx": "~4.7.0" 18 | }, 19 | "devDependencies": { 20 | "@types/ejs": "^3.1.5", 21 | "@types/http-proxy": "^1.17.14", 22 | "@types/node": "~20.10.6", 23 | "@typescript-eslint/eslint-plugin": "~6.8.0", 24 | "@typescript-eslint/parser": "~6.8.0", 25 | "concurrently": "^8.2.2", 26 | "cross-env": "~7.0.3", 27 | "eslint": "~8.50.0", 28 | "eslint-plugin-jsdoc": "~46.8.2", 29 | "typescript": "~5.2.2" 30 | } 31 | }, 32 | "node_modules/@aashutoshrathi/word-wrap": { 33 | "version": "1.2.6", 34 | "resolved": "https://registry.npmjs.org/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz", 35 | "integrity": "sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==", 36 | "dev": true, 37 | "engines": { 38 | "node": ">=0.10.0" 39 | } 40 | }, 41 | "node_modules/@babel/runtime": { 42 | "version": "7.23.7", 43 | "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.23.7.tgz", 44 | "integrity": "sha512-w06OXVOFso7LcbzMiDGt+3X7Rh7Ho8MmgPoWU3rarH+8upf+wSU/grlGbWzQyr3DkdN6ZeuMFjpdwW0Q+HxobA==", 45 | "dev": true, 46 | "dependencies": { 47 | "regenerator-runtime": "^0.14.0" 48 | }, 49 | "engines": { 50 | "node": ">=6.9.0" 51 | } 52 | }, 53 | "node_modules/@es-joy/jsdoccomment": { 54 | "version": "0.40.1", 55 | "resolved": "https://registry.npmjs.org/@es-joy/jsdoccomment/-/jsdoccomment-0.40.1.tgz", 56 | "integrity": "sha512-YORCdZSusAlBrFpZ77pJjc5r1bQs5caPWtAu+WWmiSo+8XaUzseapVrfAtiRFbQWnrBxxLLEwF6f6ZG/UgCQCg==", 57 | "dev": true, 58 | "dependencies": { 59 | "comment-parser": "1.4.0", 60 | "esquery": "^1.5.0", 61 | "jsdoc-type-pratt-parser": "~4.0.0" 62 | }, 63 | "engines": { 64 | "node": ">=16" 65 | } 66 | }, 67 | "node_modules/@esbuild/aix-ppc64": { 68 | "version": "0.19.11", 69 | "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.19.11.tgz", 70 | "integrity": "sha512-FnzU0LyE3ySQk7UntJO4+qIiQgI7KoODnZg5xzXIrFJlKd2P2gwHsHY4927xj9y5PJmJSzULiUCWmv7iWnNa7g==", 71 | "cpu": [ 72 | "ppc64" 73 | ], 74 | "optional": true, 75 | "os": [ 76 | "aix" 77 | ], 78 | "engines": { 79 | "node": ">=12" 80 | } 81 | }, 82 | "node_modules/@esbuild/android-arm": { 83 | "version": "0.19.11", 84 | "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.19.11.tgz", 85 | "integrity": "sha512-5OVapq0ClabvKvQ58Bws8+wkLCV+Rxg7tUVbo9xu034Nm536QTII4YzhaFriQ7rMrorfnFKUsArD2lqKbFY4vw==", 86 | "cpu": [ 87 | "arm" 88 | ], 89 | "optional": true, 90 | "os": [ 91 | "android" 92 | ], 93 | "engines": { 94 | "node": ">=12" 95 | } 96 | }, 97 | "node_modules/@esbuild/android-arm64": { 98 | "version": "0.19.11", 99 | "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.19.11.tgz", 100 | "integrity": "sha512-aiu7K/5JnLj//KOnOfEZ0D90obUkRzDMyqd/wNAUQ34m4YUPVhRZpnqKV9uqDGxT7cToSDnIHsGooyIczu9T+Q==", 101 | "cpu": [ 102 | "arm64" 103 | ], 104 | "optional": true, 105 | "os": [ 106 | "android" 107 | ], 108 | "engines": { 109 | "node": ">=12" 110 | } 111 | }, 112 | "node_modules/@esbuild/android-x64": { 113 | "version": "0.19.11", 114 | "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.19.11.tgz", 115 | "integrity": "sha512-eccxjlfGw43WYoY9QgB82SgGgDbibcqyDTlk3l3C0jOVHKxrjdc9CTwDUQd0vkvYg5um0OH+GpxYvp39r+IPOg==", 116 | "cpu": [ 117 | "x64" 118 | ], 119 | "optional": true, 120 | "os": [ 121 | "android" 122 | ], 123 | "engines": { 124 | "node": ">=12" 125 | } 126 | }, 127 | "node_modules/@esbuild/darwin-arm64": { 128 | "version": "0.19.11", 129 | "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.19.11.tgz", 130 | "integrity": "sha512-ETp87DRWuSt9KdDVkqSoKoLFHYTrkyz2+65fj9nfXsaV3bMhTCjtQfw3y+um88vGRKRiF7erPrh/ZuIdLUIVxQ==", 131 | "cpu": [ 132 | "arm64" 133 | ], 134 | "optional": true, 135 | "os": [ 136 | "darwin" 137 | ], 138 | "engines": { 139 | "node": ">=12" 140 | } 141 | }, 142 | "node_modules/@esbuild/darwin-x64": { 143 | "version": "0.19.11", 144 | "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.19.11.tgz", 145 | "integrity": "sha512-fkFUiS6IUK9WYUO/+22omwetaSNl5/A8giXvQlcinLIjVkxwTLSktbF5f/kJMftM2MJp9+fXqZ5ezS7+SALp4g==", 146 | "cpu": [ 147 | "x64" 148 | ], 149 | "optional": true, 150 | "os": [ 151 | "darwin" 152 | ], 153 | "engines": { 154 | "node": ">=12" 155 | } 156 | }, 157 | "node_modules/@esbuild/freebsd-arm64": { 158 | "version": "0.19.11", 159 | "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.19.11.tgz", 160 | "integrity": "sha512-lhoSp5K6bxKRNdXUtHoNc5HhbXVCS8V0iZmDvyWvYq9S5WSfTIHU2UGjcGt7UeS6iEYp9eeymIl5mJBn0yiuxA==", 161 | "cpu": [ 162 | "arm64" 163 | ], 164 | "optional": true, 165 | "os": [ 166 | "freebsd" 167 | ], 168 | "engines": { 169 | "node": ">=12" 170 | } 171 | }, 172 | "node_modules/@esbuild/freebsd-x64": { 173 | "version": "0.19.11", 174 | "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.19.11.tgz", 175 | "integrity": "sha512-JkUqn44AffGXitVI6/AbQdoYAq0TEullFdqcMY/PCUZ36xJ9ZJRtQabzMA+Vi7r78+25ZIBosLTOKnUXBSi1Kw==", 176 | "cpu": [ 177 | "x64" 178 | ], 179 | "optional": true, 180 | "os": [ 181 | "freebsd" 182 | ], 183 | "engines": { 184 | "node": ">=12" 185 | } 186 | }, 187 | "node_modules/@esbuild/linux-arm": { 188 | "version": "0.19.11", 189 | "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.19.11.tgz", 190 | "integrity": "sha512-3CRkr9+vCV2XJbjwgzjPtO8T0SZUmRZla+UL1jw+XqHZPkPgZiyWvbDvl9rqAN8Zl7qJF0O/9ycMtjU67HN9/Q==", 191 | "cpu": [ 192 | "arm" 193 | ], 194 | "optional": true, 195 | "os": [ 196 | "linux" 197 | ], 198 | "engines": { 199 | "node": ">=12" 200 | } 201 | }, 202 | "node_modules/@esbuild/linux-arm64": { 203 | "version": "0.19.11", 204 | "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.19.11.tgz", 205 | "integrity": "sha512-LneLg3ypEeveBSMuoa0kwMpCGmpu8XQUh+mL8XXwoYZ6Be2qBnVtcDI5azSvh7vioMDhoJFZzp9GWp9IWpYoUg==", 206 | "cpu": [ 207 | "arm64" 208 | ], 209 | "optional": true, 210 | "os": [ 211 | "linux" 212 | ], 213 | "engines": { 214 | "node": ">=12" 215 | } 216 | }, 217 | "node_modules/@esbuild/linux-ia32": { 218 | "version": "0.19.11", 219 | "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.19.11.tgz", 220 | "integrity": "sha512-caHy++CsD8Bgq2V5CodbJjFPEiDPq8JJmBdeyZ8GWVQMjRD0sU548nNdwPNvKjVpamYYVL40AORekgfIubwHoA==", 221 | "cpu": [ 222 | "ia32" 223 | ], 224 | "optional": true, 225 | "os": [ 226 | "linux" 227 | ], 228 | "engines": { 229 | "node": ">=12" 230 | } 231 | }, 232 | "node_modules/@esbuild/linux-loong64": { 233 | "version": "0.19.11", 234 | "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.19.11.tgz", 235 | "integrity": "sha512-ppZSSLVpPrwHccvC6nQVZaSHlFsvCQyjnvirnVjbKSHuE5N24Yl8F3UwYUUR1UEPaFObGD2tSvVKbvR+uT1Nrg==", 236 | "cpu": [ 237 | "loong64" 238 | ], 239 | "optional": true, 240 | "os": [ 241 | "linux" 242 | ], 243 | "engines": { 244 | "node": ">=12" 245 | } 246 | }, 247 | "node_modules/@esbuild/linux-mips64el": { 248 | "version": "0.19.11", 249 | "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.19.11.tgz", 250 | "integrity": "sha512-B5x9j0OgjG+v1dF2DkH34lr+7Gmv0kzX6/V0afF41FkPMMqaQ77pH7CrhWeR22aEeHKaeZVtZ6yFwlxOKPVFyg==", 251 | "cpu": [ 252 | "mips64el" 253 | ], 254 | "optional": true, 255 | "os": [ 256 | "linux" 257 | ], 258 | "engines": { 259 | "node": ">=12" 260 | } 261 | }, 262 | "node_modules/@esbuild/linux-ppc64": { 263 | "version": "0.19.11", 264 | "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.19.11.tgz", 265 | "integrity": "sha512-MHrZYLeCG8vXblMetWyttkdVRjQlQUb/oMgBNurVEnhj4YWOr4G5lmBfZjHYQHHN0g6yDmCAQRR8MUHldvvRDA==", 266 | "cpu": [ 267 | "ppc64" 268 | ], 269 | "optional": true, 270 | "os": [ 271 | "linux" 272 | ], 273 | "engines": { 274 | "node": ">=12" 275 | } 276 | }, 277 | "node_modules/@esbuild/linux-riscv64": { 278 | "version": "0.19.11", 279 | "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.19.11.tgz", 280 | "integrity": "sha512-f3DY++t94uVg141dozDu4CCUkYW+09rWtaWfnb3bqe4w5NqmZd6nPVBm+qbz7WaHZCoqXqHz5p6CM6qv3qnSSQ==", 281 | "cpu": [ 282 | "riscv64" 283 | ], 284 | "optional": true, 285 | "os": [ 286 | "linux" 287 | ], 288 | "engines": { 289 | "node": ">=12" 290 | } 291 | }, 292 | "node_modules/@esbuild/linux-s390x": { 293 | "version": "0.19.11", 294 | "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.19.11.tgz", 295 | "integrity": "sha512-A5xdUoyWJHMMlcSMcPGVLzYzpcY8QP1RtYzX5/bS4dvjBGVxdhuiYyFwp7z74ocV7WDc0n1harxmpq2ePOjI0Q==", 296 | "cpu": [ 297 | "s390x" 298 | ], 299 | "optional": true, 300 | "os": [ 301 | "linux" 302 | ], 303 | "engines": { 304 | "node": ">=12" 305 | } 306 | }, 307 | "node_modules/@esbuild/linux-x64": { 308 | "version": "0.19.11", 309 | "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.19.11.tgz", 310 | "integrity": "sha512-grbyMlVCvJSfxFQUndw5mCtWs5LO1gUlwP4CDi4iJBbVpZcqLVT29FxgGuBJGSzyOxotFG4LoO5X+M1350zmPA==", 311 | "cpu": [ 312 | "x64" 313 | ], 314 | "optional": true, 315 | "os": [ 316 | "linux" 317 | ], 318 | "engines": { 319 | "node": ">=12" 320 | } 321 | }, 322 | "node_modules/@esbuild/netbsd-x64": { 323 | "version": "0.19.11", 324 | "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.19.11.tgz", 325 | "integrity": "sha512-13jvrQZJc3P230OhU8xgwUnDeuC/9egsjTkXN49b3GcS5BKvJqZn86aGM8W9pd14Kd+u7HuFBMVtrNGhh6fHEQ==", 326 | "cpu": [ 327 | "x64" 328 | ], 329 | "optional": true, 330 | "os": [ 331 | "netbsd" 332 | ], 333 | "engines": { 334 | "node": ">=12" 335 | } 336 | }, 337 | "node_modules/@esbuild/openbsd-x64": { 338 | "version": "0.19.11", 339 | "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.19.11.tgz", 340 | "integrity": "sha512-ysyOGZuTp6SNKPE11INDUeFVVQFrhcNDVUgSQVDzqsqX38DjhPEPATpid04LCoUr2WXhQTEZ8ct/EgJCUDpyNw==", 341 | "cpu": [ 342 | "x64" 343 | ], 344 | "optional": true, 345 | "os": [ 346 | "openbsd" 347 | ], 348 | "engines": { 349 | "node": ">=12" 350 | } 351 | }, 352 | "node_modules/@esbuild/sunos-x64": { 353 | "version": "0.19.11", 354 | "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.19.11.tgz", 355 | "integrity": "sha512-Hf+Sad9nVwvtxy4DXCZQqLpgmRTQqyFyhT3bZ4F2XlJCjxGmRFF0Shwn9rzhOYRB61w9VMXUkxlBy56dk9JJiQ==", 356 | "cpu": [ 357 | "x64" 358 | ], 359 | "optional": true, 360 | "os": [ 361 | "sunos" 362 | ], 363 | "engines": { 364 | "node": ">=12" 365 | } 366 | }, 367 | "node_modules/@esbuild/win32-arm64": { 368 | "version": "0.19.11", 369 | "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.19.11.tgz", 370 | "integrity": "sha512-0P58Sbi0LctOMOQbpEOvOL44Ne0sqbS0XWHMvvrg6NE5jQ1xguCSSw9jQeUk2lfrXYsKDdOe6K+oZiwKPilYPQ==", 371 | "cpu": [ 372 | "arm64" 373 | ], 374 | "optional": true, 375 | "os": [ 376 | "win32" 377 | ], 378 | "engines": { 379 | "node": ">=12" 380 | } 381 | }, 382 | "node_modules/@esbuild/win32-ia32": { 383 | "version": "0.19.11", 384 | "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.19.11.tgz", 385 | "integrity": "sha512-6YOrWS+sDJDmshdBIQU+Uoyh7pQKrdykdefC1avn76ss5c+RN6gut3LZA4E2cH5xUEp5/cA0+YxRaVtRAb0xBg==", 386 | "cpu": [ 387 | "ia32" 388 | ], 389 | "optional": true, 390 | "os": [ 391 | "win32" 392 | ], 393 | "engines": { 394 | "node": ">=12" 395 | } 396 | }, 397 | "node_modules/@esbuild/win32-x64": { 398 | "version": "0.19.11", 399 | "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.19.11.tgz", 400 | "integrity": "sha512-vfkhltrjCAb603XaFhqhAF4LGDi2M4OrCRrFusyQ+iTLQ/o60QQXxc9cZC/FFpihBI9N1Grn6SMKVJ4KP7Fuiw==", 401 | "cpu": [ 402 | "x64" 403 | ], 404 | "optional": true, 405 | "os": [ 406 | "win32" 407 | ], 408 | "engines": { 409 | "node": ">=12" 410 | } 411 | }, 412 | "node_modules/@eslint-community/eslint-utils": { 413 | "version": "4.4.0", 414 | "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", 415 | "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", 416 | "dev": true, 417 | "dependencies": { 418 | "eslint-visitor-keys": "^3.3.0" 419 | }, 420 | "engines": { 421 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 422 | }, 423 | "peerDependencies": { 424 | "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" 425 | } 426 | }, 427 | "node_modules/@eslint-community/regexpp": { 428 | "version": "4.10.0", 429 | "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.10.0.tgz", 430 | "integrity": "sha512-Cu96Sd2By9mCNTx2iyKOmq10v22jUVQv0lQnlGNy16oE9589yE+QADPbrMGCkA51cKZSg3Pu/aTJVTGfL/qjUA==", 431 | "dev": true, 432 | "engines": { 433 | "node": "^12.0.0 || ^14.0.0 || >=16.0.0" 434 | } 435 | }, 436 | "node_modules/@eslint/eslintrc": { 437 | "version": "2.1.4", 438 | "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", 439 | "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", 440 | "dev": true, 441 | "dependencies": { 442 | "ajv": "^6.12.4", 443 | "debug": "^4.3.2", 444 | "espree": "^9.6.0", 445 | "globals": "^13.19.0", 446 | "ignore": "^5.2.0", 447 | "import-fresh": "^3.2.1", 448 | "js-yaml": "^4.1.0", 449 | "minimatch": "^3.1.2", 450 | "strip-json-comments": "^3.1.1" 451 | }, 452 | "engines": { 453 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 454 | }, 455 | "funding": { 456 | "url": "https://opencollective.com/eslint" 457 | } 458 | }, 459 | "node_modules/@eslint/js": { 460 | "version": "8.50.0", 461 | "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.50.0.tgz", 462 | "integrity": "sha512-NCC3zz2+nvYd+Ckfh87rA47zfu2QsQpvc6k1yzTk+b9KzRj0wkGa8LSoGOXN6Zv4lRf/EIoZ80biDh9HOI+RNQ==", 463 | "dev": true, 464 | "engines": { 465 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 466 | } 467 | }, 468 | "node_modules/@humanwhocodes/config-array": { 469 | "version": "0.11.13", 470 | "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.13.tgz", 471 | "integrity": "sha512-JSBDMiDKSzQVngfRjOdFXgFfklaXI4K9nLF49Auh21lmBWRLIK3+xTErTWD4KU54pb6coM6ESE7Awz/FNU3zgQ==", 472 | "dev": true, 473 | "dependencies": { 474 | "@humanwhocodes/object-schema": "^2.0.1", 475 | "debug": "^4.1.1", 476 | "minimatch": "^3.0.5" 477 | }, 478 | "engines": { 479 | "node": ">=10.10.0" 480 | } 481 | }, 482 | "node_modules/@humanwhocodes/module-importer": { 483 | "version": "1.0.1", 484 | "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", 485 | "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", 486 | "dev": true, 487 | "engines": { 488 | "node": ">=12.22" 489 | }, 490 | "funding": { 491 | "type": "github", 492 | "url": "https://github.com/sponsors/nzakas" 493 | } 494 | }, 495 | "node_modules/@humanwhocodes/object-schema": { 496 | "version": "2.0.1", 497 | "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.1.tgz", 498 | "integrity": "sha512-dvuCeX5fC9dXgJn9t+X5atfmgQAzUOWqS1254Gh0m6i8wKd10ebXkfNKiRK+1GWi/yTvvLDHpoxLr0xxxeslWw==", 499 | "dev": true 500 | }, 501 | "node_modules/@nodelib/fs.scandir": { 502 | "version": "2.1.5", 503 | "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", 504 | "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", 505 | "dev": true, 506 | "dependencies": { 507 | "@nodelib/fs.stat": "2.0.5", 508 | "run-parallel": "^1.1.9" 509 | }, 510 | "engines": { 511 | "node": ">= 8" 512 | } 513 | }, 514 | "node_modules/@nodelib/fs.stat": { 515 | "version": "2.0.5", 516 | "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", 517 | "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", 518 | "dev": true, 519 | "engines": { 520 | "node": ">= 8" 521 | } 522 | }, 523 | "node_modules/@nodelib/fs.walk": { 524 | "version": "1.2.8", 525 | "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", 526 | "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", 527 | "dev": true, 528 | "dependencies": { 529 | "@nodelib/fs.scandir": "2.1.5", 530 | "fastq": "^1.6.0" 531 | }, 532 | "engines": { 533 | "node": ">= 8" 534 | } 535 | }, 536 | "node_modules/@types/ejs": { 537 | "version": "3.1.5", 538 | "resolved": "https://registry.npmjs.org/@types/ejs/-/ejs-3.1.5.tgz", 539 | "integrity": "sha512-nv+GSx77ZtXiJzwKdsASqi+YQ5Z7vwHsTP0JY2SiQgjGckkBRKZnk8nIM+7oUZ1VCtuTz0+By4qVR7fqzp/Dfg==", 540 | "dev": true 541 | }, 542 | "node_modules/@types/http-proxy": { 543 | "version": "1.17.14", 544 | "resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.14.tgz", 545 | "integrity": "sha512-SSrD0c1OQzlFX7pGu1eXxSEjemej64aaNPRhhVYUGqXh0BtldAAx37MG8btcumvpgKyZp1F5Gn3JkktdxiFv6w==", 546 | "dev": true, 547 | "dependencies": { 548 | "@types/node": "*" 549 | } 550 | }, 551 | "node_modules/@types/json-schema": { 552 | "version": "7.0.15", 553 | "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", 554 | "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", 555 | "dev": true 556 | }, 557 | "node_modules/@types/node": { 558 | "version": "20.10.6", 559 | "resolved": "https://registry.npmjs.org/@types/node/-/node-20.10.6.tgz", 560 | "integrity": "sha512-Vac8H+NlRNNlAmDfGUP7b5h/KA+AtWIzuXy0E6OyP8f1tCLYAtPvKRRDJjAPqhpCb0t6U2j7/xqAuLEebW2kiw==", 561 | "dev": true, 562 | "dependencies": { 563 | "undici-types": "~5.26.4" 564 | } 565 | }, 566 | "node_modules/@types/semver": { 567 | "version": "7.5.6", 568 | "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.6.tgz", 569 | "integrity": "sha512-dn1l8LaMea/IjDoHNd9J52uBbInB796CDffS6VdIxvqYCPSG0V0DzHp76GpaWnlhg88uYyPbXCDIowa86ybd5A==", 570 | "dev": true 571 | }, 572 | "node_modules/@typescript-eslint/eslint-plugin": { 573 | "version": "6.8.0", 574 | "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.8.0.tgz", 575 | "integrity": "sha512-GosF4238Tkes2SHPQ1i8f6rMtG6zlKwMEB0abqSJ3Npvos+doIlc/ATG+vX1G9coDF3Ex78zM3heXHLyWEwLUw==", 576 | "dev": true, 577 | "dependencies": { 578 | "@eslint-community/regexpp": "^4.5.1", 579 | "@typescript-eslint/scope-manager": "6.8.0", 580 | "@typescript-eslint/type-utils": "6.8.0", 581 | "@typescript-eslint/utils": "6.8.0", 582 | "@typescript-eslint/visitor-keys": "6.8.0", 583 | "debug": "^4.3.4", 584 | "graphemer": "^1.4.0", 585 | "ignore": "^5.2.4", 586 | "natural-compare": "^1.4.0", 587 | "semver": "^7.5.4", 588 | "ts-api-utils": "^1.0.1" 589 | }, 590 | "engines": { 591 | "node": "^16.0.0 || >=18.0.0" 592 | }, 593 | "funding": { 594 | "type": "opencollective", 595 | "url": "https://opencollective.com/typescript-eslint" 596 | }, 597 | "peerDependencies": { 598 | "@typescript-eslint/parser": "^6.0.0 || ^6.0.0-alpha", 599 | "eslint": "^7.0.0 || ^8.0.0" 600 | }, 601 | "peerDependenciesMeta": { 602 | "typescript": { 603 | "optional": true 604 | } 605 | } 606 | }, 607 | "node_modules/@typescript-eslint/parser": { 608 | "version": "6.8.0", 609 | "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-6.8.0.tgz", 610 | "integrity": "sha512-5tNs6Bw0j6BdWuP8Fx+VH4G9fEPDxnVI7yH1IAPkQH5RUtvKwRoqdecAPdQXv4rSOADAaz1LFBZvZG7VbXivSg==", 611 | "dev": true, 612 | "dependencies": { 613 | "@typescript-eslint/scope-manager": "6.8.0", 614 | "@typescript-eslint/types": "6.8.0", 615 | "@typescript-eslint/typescript-estree": "6.8.0", 616 | "@typescript-eslint/visitor-keys": "6.8.0", 617 | "debug": "^4.3.4" 618 | }, 619 | "engines": { 620 | "node": "^16.0.0 || >=18.0.0" 621 | }, 622 | "funding": { 623 | "type": "opencollective", 624 | "url": "https://opencollective.com/typescript-eslint" 625 | }, 626 | "peerDependencies": { 627 | "eslint": "^7.0.0 || ^8.0.0" 628 | }, 629 | "peerDependenciesMeta": { 630 | "typescript": { 631 | "optional": true 632 | } 633 | } 634 | }, 635 | "node_modules/@typescript-eslint/scope-manager": { 636 | "version": "6.8.0", 637 | "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.8.0.tgz", 638 | "integrity": "sha512-xe0HNBVwCph7rak+ZHcFD6A+q50SMsFwcmfdjs9Kz4qDh5hWhaPhFjRs/SODEhroBI5Ruyvyz9LfwUJ624O40g==", 639 | "dev": true, 640 | "dependencies": { 641 | "@typescript-eslint/types": "6.8.0", 642 | "@typescript-eslint/visitor-keys": "6.8.0" 643 | }, 644 | "engines": { 645 | "node": "^16.0.0 || >=18.0.0" 646 | }, 647 | "funding": { 648 | "type": "opencollective", 649 | "url": "https://opencollective.com/typescript-eslint" 650 | } 651 | }, 652 | "node_modules/@typescript-eslint/type-utils": { 653 | "version": "6.8.0", 654 | "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.8.0.tgz", 655 | "integrity": "sha512-RYOJdlkTJIXW7GSldUIHqc/Hkto8E+fZN96dMIFhuTJcQwdRoGN2rEWA8U6oXbLo0qufH7NPElUb+MceHtz54g==", 656 | "dev": true, 657 | "dependencies": { 658 | "@typescript-eslint/typescript-estree": "6.8.0", 659 | "@typescript-eslint/utils": "6.8.0", 660 | "debug": "^4.3.4", 661 | "ts-api-utils": "^1.0.1" 662 | }, 663 | "engines": { 664 | "node": "^16.0.0 || >=18.0.0" 665 | }, 666 | "funding": { 667 | "type": "opencollective", 668 | "url": "https://opencollective.com/typescript-eslint" 669 | }, 670 | "peerDependencies": { 671 | "eslint": "^7.0.0 || ^8.0.0" 672 | }, 673 | "peerDependenciesMeta": { 674 | "typescript": { 675 | "optional": true 676 | } 677 | } 678 | }, 679 | "node_modules/@typescript-eslint/types": { 680 | "version": "6.8.0", 681 | "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.8.0.tgz", 682 | "integrity": "sha512-p5qOxSum7W3k+llc7owEStXlGmSl8FcGvhYt8Vjy7FqEnmkCVlM3P57XQEGj58oqaBWDQXbJDZxwUWMS/EAPNQ==", 683 | "dev": true, 684 | "engines": { 685 | "node": "^16.0.0 || >=18.0.0" 686 | }, 687 | "funding": { 688 | "type": "opencollective", 689 | "url": "https://opencollective.com/typescript-eslint" 690 | } 691 | }, 692 | "node_modules/@typescript-eslint/typescript-estree": { 693 | "version": "6.8.0", 694 | "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.8.0.tgz", 695 | "integrity": "sha512-ISgV0lQ8XgW+mvv5My/+iTUdRmGspducmQcDw5JxznasXNnZn3SKNrTRuMsEXv+V/O+Lw9AGcQCfVaOPCAk/Zg==", 696 | "dev": true, 697 | "dependencies": { 698 | "@typescript-eslint/types": "6.8.0", 699 | "@typescript-eslint/visitor-keys": "6.8.0", 700 | "debug": "^4.3.4", 701 | "globby": "^11.1.0", 702 | "is-glob": "^4.0.3", 703 | "semver": "^7.5.4", 704 | "ts-api-utils": "^1.0.1" 705 | }, 706 | "engines": { 707 | "node": "^16.0.0 || >=18.0.0" 708 | }, 709 | "funding": { 710 | "type": "opencollective", 711 | "url": "https://opencollective.com/typescript-eslint" 712 | }, 713 | "peerDependenciesMeta": { 714 | "typescript": { 715 | "optional": true 716 | } 717 | } 718 | }, 719 | "node_modules/@typescript-eslint/utils": { 720 | "version": "6.8.0", 721 | "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.8.0.tgz", 722 | "integrity": "sha512-dKs1itdE2qFG4jr0dlYLQVppqTE+Itt7GmIf/vX6CSvsW+3ov8PbWauVKyyfNngokhIO9sKZeRGCUo1+N7U98Q==", 723 | "dev": true, 724 | "dependencies": { 725 | "@eslint-community/eslint-utils": "^4.4.0", 726 | "@types/json-schema": "^7.0.12", 727 | "@types/semver": "^7.5.0", 728 | "@typescript-eslint/scope-manager": "6.8.0", 729 | "@typescript-eslint/types": "6.8.0", 730 | "@typescript-eslint/typescript-estree": "6.8.0", 731 | "semver": "^7.5.4" 732 | }, 733 | "engines": { 734 | "node": "^16.0.0 || >=18.0.0" 735 | }, 736 | "funding": { 737 | "type": "opencollective", 738 | "url": "https://opencollective.com/typescript-eslint" 739 | }, 740 | "peerDependencies": { 741 | "eslint": "^7.0.0 || ^8.0.0" 742 | } 743 | }, 744 | "node_modules/@typescript-eslint/visitor-keys": { 745 | "version": "6.8.0", 746 | "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.8.0.tgz", 747 | "integrity": "sha512-oqAnbA7c+pgOhW2OhGvxm0t1BULX5peQI/rLsNDpGM78EebV3C9IGbX5HNZabuZ6UQrYveCLjKo8Iy/lLlBkkg==", 748 | "dev": true, 749 | "dependencies": { 750 | "@typescript-eslint/types": "6.8.0", 751 | "eslint-visitor-keys": "^3.4.1" 752 | }, 753 | "engines": { 754 | "node": "^16.0.0 || >=18.0.0" 755 | }, 756 | "funding": { 757 | "type": "opencollective", 758 | "url": "https://opencollective.com/typescript-eslint" 759 | } 760 | }, 761 | "node_modules/acorn": { 762 | "version": "8.11.3", 763 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.11.3.tgz", 764 | "integrity": "sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg==", 765 | "dev": true, 766 | "bin": { 767 | "acorn": "bin/acorn" 768 | }, 769 | "engines": { 770 | "node": ">=0.4.0" 771 | } 772 | }, 773 | "node_modules/acorn-jsx": { 774 | "version": "5.3.2", 775 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", 776 | "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", 777 | "dev": true, 778 | "peerDependencies": { 779 | "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" 780 | } 781 | }, 782 | "node_modules/ajv": { 783 | "version": "6.12.6", 784 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", 785 | "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", 786 | "dev": true, 787 | "dependencies": { 788 | "fast-deep-equal": "^3.1.1", 789 | "fast-json-stable-stringify": "^2.0.0", 790 | "json-schema-traverse": "^0.4.1", 791 | "uri-js": "^4.2.2" 792 | }, 793 | "funding": { 794 | "type": "github", 795 | "url": "https://github.com/sponsors/epoberezkin" 796 | } 797 | }, 798 | "node_modules/ansi-regex": { 799 | "version": "5.0.1", 800 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 801 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 802 | "dev": true, 803 | "engines": { 804 | "node": ">=8" 805 | } 806 | }, 807 | "node_modules/ansi-styles": { 808 | "version": "4.3.0", 809 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 810 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 811 | "dependencies": { 812 | "color-convert": "^2.0.1" 813 | }, 814 | "engines": { 815 | "node": ">=8" 816 | }, 817 | "funding": { 818 | "url": "https://github.com/chalk/ansi-styles?sponsor=1" 819 | } 820 | }, 821 | "node_modules/are-docs-informative": { 822 | "version": "0.0.2", 823 | "resolved": "https://registry.npmjs.org/are-docs-informative/-/are-docs-informative-0.0.2.tgz", 824 | "integrity": "sha512-ixiS0nLNNG5jNQzgZJNoUpBKdo9yTYZMGJ+QgT2jmjR7G7+QHRCc4v6LQ3NgE7EBJq+o0ams3waJwkrlBom8Ig==", 825 | "dev": true, 826 | "engines": { 827 | "node": ">=14" 828 | } 829 | }, 830 | "node_modules/argparse": { 831 | "version": "2.0.1", 832 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", 833 | "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", 834 | "dev": true 835 | }, 836 | "node_modules/array-union": { 837 | "version": "2.1.0", 838 | "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", 839 | "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", 840 | "dev": true, 841 | "engines": { 842 | "node": ">=8" 843 | } 844 | }, 845 | "node_modules/async": { 846 | "version": "3.2.5", 847 | "resolved": "https://registry.npmjs.org/async/-/async-3.2.5.tgz", 848 | "integrity": "sha512-baNZyqaaLhyLVKm/DlvdW051MSgO6b8eVfIezl9E5PqWxFgzLm/wQntEW4zOytVburDEr0JlALEpdOFwvErLsg==" 849 | }, 850 | "node_modules/balanced-match": { 851 | "version": "1.0.2", 852 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 853 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" 854 | }, 855 | "node_modules/brace-expansion": { 856 | "version": "1.1.11", 857 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 858 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 859 | "dependencies": { 860 | "balanced-match": "^1.0.0", 861 | "concat-map": "0.0.1" 862 | } 863 | }, 864 | "node_modules/braces": { 865 | "version": "3.0.2", 866 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 867 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 868 | "dev": true, 869 | "dependencies": { 870 | "fill-range": "^7.0.1" 871 | }, 872 | "engines": { 873 | "node": ">=8" 874 | } 875 | }, 876 | "node_modules/builtin-modules": { 877 | "version": "3.3.0", 878 | "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-3.3.0.tgz", 879 | "integrity": "sha512-zhaCDicdLuWN5UbN5IMnFqNMhNfo919sH85y2/ea+5Yg9TsTkeZxpL+JLbp6cgYFS4sRLp3YV4S6yDuqVWHYOw==", 880 | "dev": true, 881 | "engines": { 882 | "node": ">=6" 883 | }, 884 | "funding": { 885 | "url": "https://github.com/sponsors/sindresorhus" 886 | } 887 | }, 888 | "node_modules/callsites": { 889 | "version": "3.1.0", 890 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", 891 | "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", 892 | "dev": true, 893 | "engines": { 894 | "node": ">=6" 895 | } 896 | }, 897 | "node_modules/chalk": { 898 | "version": "4.1.2", 899 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", 900 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", 901 | "dependencies": { 902 | "ansi-styles": "^4.1.0", 903 | "supports-color": "^7.1.0" 904 | }, 905 | "engines": { 906 | "node": ">=10" 907 | }, 908 | "funding": { 909 | "url": "https://github.com/chalk/chalk?sponsor=1" 910 | } 911 | }, 912 | "node_modules/chalk/node_modules/supports-color": { 913 | "version": "7.2.0", 914 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 915 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 916 | "dependencies": { 917 | "has-flag": "^4.0.0" 918 | }, 919 | "engines": { 920 | "node": ">=8" 921 | } 922 | }, 923 | "node_modules/cliui": { 924 | "version": "8.0.1", 925 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", 926 | "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", 927 | "dev": true, 928 | "dependencies": { 929 | "string-width": "^4.2.0", 930 | "strip-ansi": "^6.0.1", 931 | "wrap-ansi": "^7.0.0" 932 | }, 933 | "engines": { 934 | "node": ">=12" 935 | } 936 | }, 937 | "node_modules/color-convert": { 938 | "version": "2.0.1", 939 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 940 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 941 | "dependencies": { 942 | "color-name": "~1.1.4" 943 | }, 944 | "engines": { 945 | "node": ">=7.0.0" 946 | } 947 | }, 948 | "node_modules/color-name": { 949 | "version": "1.1.4", 950 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 951 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" 952 | }, 953 | "node_modules/comment-parser": { 954 | "version": "1.4.0", 955 | "resolved": "https://registry.npmjs.org/comment-parser/-/comment-parser-1.4.0.tgz", 956 | "integrity": "sha512-QLyTNiZ2KDOibvFPlZ6ZngVsZ/0gYnE6uTXi5aoDg8ed3AkJAz4sEje3Y8a29hQ1s6A99MZXe47fLAXQ1rTqaw==", 957 | "dev": true, 958 | "engines": { 959 | "node": ">= 12.0.0" 960 | } 961 | }, 962 | "node_modules/concat-map": { 963 | "version": "0.0.1", 964 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 965 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==" 966 | }, 967 | "node_modules/concurrently": { 968 | "version": "8.2.2", 969 | "resolved": "https://registry.npmjs.org/concurrently/-/concurrently-8.2.2.tgz", 970 | "integrity": "sha512-1dP4gpXFhei8IOtlXRE/T/4H88ElHgTiUzh71YUmtjTEHMSRS2Z/fgOxHSxxusGHogsRfxNq1vyAwxSC+EVyDg==", 971 | "dev": true, 972 | "dependencies": { 973 | "chalk": "^4.1.2", 974 | "date-fns": "^2.30.0", 975 | "lodash": "^4.17.21", 976 | "rxjs": "^7.8.1", 977 | "shell-quote": "^1.8.1", 978 | "spawn-command": "0.0.2", 979 | "supports-color": "^8.1.1", 980 | "tree-kill": "^1.2.2", 981 | "yargs": "^17.7.2" 982 | }, 983 | "bin": { 984 | "conc": "dist/bin/concurrently.js", 985 | "concurrently": "dist/bin/concurrently.js" 986 | }, 987 | "engines": { 988 | "node": "^14.13.0 || >=16.0.0" 989 | }, 990 | "funding": { 991 | "url": "https://github.com/open-cli-tools/concurrently?sponsor=1" 992 | } 993 | }, 994 | "node_modules/cross-env": { 995 | "version": "7.0.3", 996 | "resolved": "https://registry.npmjs.org/cross-env/-/cross-env-7.0.3.tgz", 997 | "integrity": "sha512-+/HKd6EgcQCJGh2PSjZuUitQBQynKor4wrFbRg4DtAgS1aWO+gU52xpH7M9ScGgXSYmAVS9bIJ8EzuaGw0oNAw==", 998 | "dev": true, 999 | "dependencies": { 1000 | "cross-spawn": "^7.0.1" 1001 | }, 1002 | "bin": { 1003 | "cross-env": "src/bin/cross-env.js", 1004 | "cross-env-shell": "src/bin/cross-env-shell.js" 1005 | }, 1006 | "engines": { 1007 | "node": ">=10.14", 1008 | "npm": ">=6", 1009 | "yarn": ">=1" 1010 | } 1011 | }, 1012 | "node_modules/cross-spawn": { 1013 | "version": "7.0.3", 1014 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", 1015 | "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", 1016 | "dev": true, 1017 | "dependencies": { 1018 | "path-key": "^3.1.0", 1019 | "shebang-command": "^2.0.0", 1020 | "which": "^2.0.1" 1021 | }, 1022 | "engines": { 1023 | "node": ">= 8" 1024 | } 1025 | }, 1026 | "node_modules/date-fns": { 1027 | "version": "2.30.0", 1028 | "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-2.30.0.tgz", 1029 | "integrity": "sha512-fnULvOpxnC5/Vg3NCiWelDsLiUc9bRwAPs/+LfTLNvetFCtCTN+yQz15C/fs4AwX1R9K5GLtLfn8QW+dWisaAw==", 1030 | "dev": true, 1031 | "dependencies": { 1032 | "@babel/runtime": "^7.21.0" 1033 | }, 1034 | "engines": { 1035 | "node": ">=0.11" 1036 | }, 1037 | "funding": { 1038 | "type": "opencollective", 1039 | "url": "https://opencollective.com/date-fns" 1040 | } 1041 | }, 1042 | "node_modules/debug": { 1043 | "version": "4.3.4", 1044 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 1045 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 1046 | "dependencies": { 1047 | "ms": "2.1.2" 1048 | }, 1049 | "engines": { 1050 | "node": ">=6.0" 1051 | }, 1052 | "peerDependenciesMeta": { 1053 | "supports-color": { 1054 | "optional": true 1055 | } 1056 | } 1057 | }, 1058 | "node_modules/deep-is": { 1059 | "version": "0.1.4", 1060 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", 1061 | "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", 1062 | "dev": true 1063 | }, 1064 | "node_modules/dir-glob": { 1065 | "version": "3.0.1", 1066 | "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", 1067 | "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", 1068 | "dev": true, 1069 | "dependencies": { 1070 | "path-type": "^4.0.0" 1071 | }, 1072 | "engines": { 1073 | "node": ">=8" 1074 | } 1075 | }, 1076 | "node_modules/doctrine": { 1077 | "version": "3.0.0", 1078 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", 1079 | "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", 1080 | "dev": true, 1081 | "dependencies": { 1082 | "esutils": "^2.0.2" 1083 | }, 1084 | "engines": { 1085 | "node": ">=6.0.0" 1086 | } 1087 | }, 1088 | "node_modules/dotenv": { 1089 | "version": "10.0.0", 1090 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-10.0.0.tgz", 1091 | "integrity": "sha512-rlBi9d8jpv9Sf1klPjNfFAuWDjKLwTIJJ/VxtoTwIR6hnZxcEOQCZg2oIL3MWBYw5GpUDKOEnND7LXTbIpQ03Q==", 1092 | "engines": { 1093 | "node": ">=10" 1094 | } 1095 | }, 1096 | "node_modules/ejs": { 1097 | "version": "3.1.9", 1098 | "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.9.tgz", 1099 | "integrity": "sha512-rC+QVNMJWv+MtPgkt0y+0rVEIdbtxVADApW9JXrUVlzHetgcyczP/E7DJmWJ4fJCZF2cPcBk0laWO9ZHMG3DmQ==", 1100 | "dependencies": { 1101 | "jake": "^10.8.5" 1102 | }, 1103 | "bin": { 1104 | "ejs": "bin/cli.js" 1105 | }, 1106 | "engines": { 1107 | "node": ">=0.10.0" 1108 | } 1109 | }, 1110 | "node_modules/emoji-regex": { 1111 | "version": "8.0.0", 1112 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 1113 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 1114 | "dev": true 1115 | }, 1116 | "node_modules/esbuild": { 1117 | "version": "0.19.11", 1118 | "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.19.11.tgz", 1119 | "integrity": "sha512-HJ96Hev2hX/6i5cDVwcqiJBBtuo9+FeIJOtZ9W1kA5M6AMJRHUZlpYZ1/SbEwtO0ioNAW8rUooVpC/WehY2SfA==", 1120 | "hasInstallScript": true, 1121 | "bin": { 1122 | "esbuild": "bin/esbuild" 1123 | }, 1124 | "engines": { 1125 | "node": ">=12" 1126 | }, 1127 | "optionalDependencies": { 1128 | "@esbuild/aix-ppc64": "0.19.11", 1129 | "@esbuild/android-arm": "0.19.11", 1130 | "@esbuild/android-arm64": "0.19.11", 1131 | "@esbuild/android-x64": "0.19.11", 1132 | "@esbuild/darwin-arm64": "0.19.11", 1133 | "@esbuild/darwin-x64": "0.19.11", 1134 | "@esbuild/freebsd-arm64": "0.19.11", 1135 | "@esbuild/freebsd-x64": "0.19.11", 1136 | "@esbuild/linux-arm": "0.19.11", 1137 | "@esbuild/linux-arm64": "0.19.11", 1138 | "@esbuild/linux-ia32": "0.19.11", 1139 | "@esbuild/linux-loong64": "0.19.11", 1140 | "@esbuild/linux-mips64el": "0.19.11", 1141 | "@esbuild/linux-ppc64": "0.19.11", 1142 | "@esbuild/linux-riscv64": "0.19.11", 1143 | "@esbuild/linux-s390x": "0.19.11", 1144 | "@esbuild/linux-x64": "0.19.11", 1145 | "@esbuild/netbsd-x64": "0.19.11", 1146 | "@esbuild/openbsd-x64": "0.19.11", 1147 | "@esbuild/sunos-x64": "0.19.11", 1148 | "@esbuild/win32-arm64": "0.19.11", 1149 | "@esbuild/win32-ia32": "0.19.11", 1150 | "@esbuild/win32-x64": "0.19.11" 1151 | } 1152 | }, 1153 | "node_modules/escalade": { 1154 | "version": "3.1.1", 1155 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", 1156 | "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", 1157 | "dev": true, 1158 | "engines": { 1159 | "node": ">=6" 1160 | } 1161 | }, 1162 | "node_modules/escape-string-regexp": { 1163 | "version": "4.0.0", 1164 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", 1165 | "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", 1166 | "dev": true, 1167 | "engines": { 1168 | "node": ">=10" 1169 | }, 1170 | "funding": { 1171 | "url": "https://github.com/sponsors/sindresorhus" 1172 | } 1173 | }, 1174 | "node_modules/eslint": { 1175 | "version": "8.50.0", 1176 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.50.0.tgz", 1177 | "integrity": "sha512-FOnOGSuFuFLv/Sa+FDVRZl4GGVAAFFi8LecRsI5a1tMO5HIE8nCm4ivAlzt4dT3ol/PaaGC0rJEEXQmHJBGoOg==", 1178 | "dev": true, 1179 | "dependencies": { 1180 | "@eslint-community/eslint-utils": "^4.2.0", 1181 | "@eslint-community/regexpp": "^4.6.1", 1182 | "@eslint/eslintrc": "^2.1.2", 1183 | "@eslint/js": "8.50.0", 1184 | "@humanwhocodes/config-array": "^0.11.11", 1185 | "@humanwhocodes/module-importer": "^1.0.1", 1186 | "@nodelib/fs.walk": "^1.2.8", 1187 | "ajv": "^6.12.4", 1188 | "chalk": "^4.0.0", 1189 | "cross-spawn": "^7.0.2", 1190 | "debug": "^4.3.2", 1191 | "doctrine": "^3.0.0", 1192 | "escape-string-regexp": "^4.0.0", 1193 | "eslint-scope": "^7.2.2", 1194 | "eslint-visitor-keys": "^3.4.3", 1195 | "espree": "^9.6.1", 1196 | "esquery": "^1.4.2", 1197 | "esutils": "^2.0.2", 1198 | "fast-deep-equal": "^3.1.3", 1199 | "file-entry-cache": "^6.0.1", 1200 | "find-up": "^5.0.0", 1201 | "glob-parent": "^6.0.2", 1202 | "globals": "^13.19.0", 1203 | "graphemer": "^1.4.0", 1204 | "ignore": "^5.2.0", 1205 | "imurmurhash": "^0.1.4", 1206 | "is-glob": "^4.0.0", 1207 | "is-path-inside": "^3.0.3", 1208 | "js-yaml": "^4.1.0", 1209 | "json-stable-stringify-without-jsonify": "^1.0.1", 1210 | "levn": "^0.4.1", 1211 | "lodash.merge": "^4.6.2", 1212 | "minimatch": "^3.1.2", 1213 | "natural-compare": "^1.4.0", 1214 | "optionator": "^0.9.3", 1215 | "strip-ansi": "^6.0.1", 1216 | "text-table": "^0.2.0" 1217 | }, 1218 | "bin": { 1219 | "eslint": "bin/eslint.js" 1220 | }, 1221 | "engines": { 1222 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1223 | }, 1224 | "funding": { 1225 | "url": "https://opencollective.com/eslint" 1226 | } 1227 | }, 1228 | "node_modules/eslint-plugin-jsdoc": { 1229 | "version": "46.8.2", 1230 | "resolved": "https://registry.npmjs.org/eslint-plugin-jsdoc/-/eslint-plugin-jsdoc-46.8.2.tgz", 1231 | "integrity": "sha512-5TSnD018f3tUJNne4s4gDWQflbsgOycIKEUBoCLn6XtBMgNHxQFmV8vVxUtiPxAQq8lrX85OaSG/2gnctxw9uQ==", 1232 | "dev": true, 1233 | "dependencies": { 1234 | "@es-joy/jsdoccomment": "~0.40.1", 1235 | "are-docs-informative": "^0.0.2", 1236 | "comment-parser": "1.4.0", 1237 | "debug": "^4.3.4", 1238 | "escape-string-regexp": "^4.0.0", 1239 | "esquery": "^1.5.0", 1240 | "is-builtin-module": "^3.2.1", 1241 | "semver": "^7.5.4", 1242 | "spdx-expression-parse": "^3.0.1" 1243 | }, 1244 | "engines": { 1245 | "node": ">=16" 1246 | }, 1247 | "peerDependencies": { 1248 | "eslint": "^7.0.0 || ^8.0.0" 1249 | } 1250 | }, 1251 | "node_modules/eslint-scope": { 1252 | "version": "7.2.2", 1253 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", 1254 | "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", 1255 | "dev": true, 1256 | "dependencies": { 1257 | "esrecurse": "^4.3.0", 1258 | "estraverse": "^5.2.0" 1259 | }, 1260 | "engines": { 1261 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1262 | }, 1263 | "funding": { 1264 | "url": "https://opencollective.com/eslint" 1265 | } 1266 | }, 1267 | "node_modules/eslint-visitor-keys": { 1268 | "version": "3.4.3", 1269 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", 1270 | "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", 1271 | "dev": true, 1272 | "engines": { 1273 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1274 | }, 1275 | "funding": { 1276 | "url": "https://opencollective.com/eslint" 1277 | } 1278 | }, 1279 | "node_modules/espree": { 1280 | "version": "9.6.1", 1281 | "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", 1282 | "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", 1283 | "dev": true, 1284 | "dependencies": { 1285 | "acorn": "^8.9.0", 1286 | "acorn-jsx": "^5.3.2", 1287 | "eslint-visitor-keys": "^3.4.1" 1288 | }, 1289 | "engines": { 1290 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1291 | }, 1292 | "funding": { 1293 | "url": "https://opencollective.com/eslint" 1294 | } 1295 | }, 1296 | "node_modules/esquery": { 1297 | "version": "1.5.0", 1298 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", 1299 | "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", 1300 | "dev": true, 1301 | "dependencies": { 1302 | "estraverse": "^5.1.0" 1303 | }, 1304 | "engines": { 1305 | "node": ">=0.10" 1306 | } 1307 | }, 1308 | "node_modules/esrecurse": { 1309 | "version": "4.3.0", 1310 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", 1311 | "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", 1312 | "dev": true, 1313 | "dependencies": { 1314 | "estraverse": "^5.2.0" 1315 | }, 1316 | "engines": { 1317 | "node": ">=4.0" 1318 | } 1319 | }, 1320 | "node_modules/estraverse": { 1321 | "version": "5.3.0", 1322 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", 1323 | "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", 1324 | "dev": true, 1325 | "engines": { 1326 | "node": ">=4.0" 1327 | } 1328 | }, 1329 | "node_modules/esutils": { 1330 | "version": "2.0.3", 1331 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", 1332 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", 1333 | "dev": true, 1334 | "engines": { 1335 | "node": ">=0.10.0" 1336 | } 1337 | }, 1338 | "node_modules/eventemitter3": { 1339 | "version": "4.0.7", 1340 | "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz", 1341 | "integrity": "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==" 1342 | }, 1343 | "node_modules/fast-deep-equal": { 1344 | "version": "3.1.3", 1345 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", 1346 | "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", 1347 | "dev": true 1348 | }, 1349 | "node_modules/fast-glob": { 1350 | "version": "3.3.2", 1351 | "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz", 1352 | "integrity": "sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==", 1353 | "dev": true, 1354 | "dependencies": { 1355 | "@nodelib/fs.stat": "^2.0.2", 1356 | "@nodelib/fs.walk": "^1.2.3", 1357 | "glob-parent": "^5.1.2", 1358 | "merge2": "^1.3.0", 1359 | "micromatch": "^4.0.4" 1360 | }, 1361 | "engines": { 1362 | "node": ">=8.6.0" 1363 | } 1364 | }, 1365 | "node_modules/fast-glob/node_modules/glob-parent": { 1366 | "version": "5.1.2", 1367 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 1368 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 1369 | "dev": true, 1370 | "dependencies": { 1371 | "is-glob": "^4.0.1" 1372 | }, 1373 | "engines": { 1374 | "node": ">= 6" 1375 | } 1376 | }, 1377 | "node_modules/fast-json-stable-stringify": { 1378 | "version": "2.1.0", 1379 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", 1380 | "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", 1381 | "dev": true 1382 | }, 1383 | "node_modules/fast-levenshtein": { 1384 | "version": "2.0.6", 1385 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", 1386 | "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", 1387 | "dev": true 1388 | }, 1389 | "node_modules/fastq": { 1390 | "version": "1.16.0", 1391 | "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.16.0.tgz", 1392 | "integrity": "sha512-ifCoaXsDrsdkWTtiNJX5uzHDsrck5TzfKKDcuFFTIrrc/BS076qgEIfoIy1VeZqViznfKiysPYTh/QeHtnIsYA==", 1393 | "dev": true, 1394 | "dependencies": { 1395 | "reusify": "^1.0.4" 1396 | } 1397 | }, 1398 | "node_modules/file-entry-cache": { 1399 | "version": "6.0.1", 1400 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", 1401 | "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", 1402 | "dev": true, 1403 | "dependencies": { 1404 | "flat-cache": "^3.0.4" 1405 | }, 1406 | "engines": { 1407 | "node": "^10.12.0 || >=12.0.0" 1408 | } 1409 | }, 1410 | "node_modules/filelist": { 1411 | "version": "1.0.4", 1412 | "resolved": "https://registry.npmjs.org/filelist/-/filelist-1.0.4.tgz", 1413 | "integrity": "sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q==", 1414 | "dependencies": { 1415 | "minimatch": "^5.0.1" 1416 | } 1417 | }, 1418 | "node_modules/filelist/node_modules/brace-expansion": { 1419 | "version": "2.0.1", 1420 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", 1421 | "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", 1422 | "dependencies": { 1423 | "balanced-match": "^1.0.0" 1424 | } 1425 | }, 1426 | "node_modules/filelist/node_modules/minimatch": { 1427 | "version": "5.1.6", 1428 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz", 1429 | "integrity": "sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==", 1430 | "dependencies": { 1431 | "brace-expansion": "^2.0.1" 1432 | }, 1433 | "engines": { 1434 | "node": ">=10" 1435 | } 1436 | }, 1437 | "node_modules/fill-range": { 1438 | "version": "7.0.1", 1439 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 1440 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 1441 | "dev": true, 1442 | "dependencies": { 1443 | "to-regex-range": "^5.0.1" 1444 | }, 1445 | "engines": { 1446 | "node": ">=8" 1447 | } 1448 | }, 1449 | "node_modules/find-up": { 1450 | "version": "5.0.0", 1451 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", 1452 | "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", 1453 | "dev": true, 1454 | "dependencies": { 1455 | "locate-path": "^6.0.0", 1456 | "path-exists": "^4.0.0" 1457 | }, 1458 | "engines": { 1459 | "node": ">=10" 1460 | }, 1461 | "funding": { 1462 | "url": "https://github.com/sponsors/sindresorhus" 1463 | } 1464 | }, 1465 | "node_modules/flat-cache": { 1466 | "version": "3.2.0", 1467 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", 1468 | "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", 1469 | "dev": true, 1470 | "dependencies": { 1471 | "flatted": "^3.2.9", 1472 | "keyv": "^4.5.3", 1473 | "rimraf": "^3.0.2" 1474 | }, 1475 | "engines": { 1476 | "node": "^10.12.0 || >=12.0.0" 1477 | } 1478 | }, 1479 | "node_modules/flatted": { 1480 | "version": "3.2.9", 1481 | "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.9.tgz", 1482 | "integrity": "sha512-36yxDn5H7OFZQla0/jFJmbIKTdZAQHngCedGxiMmpNfEZM0sdEeT+WczLQrjK6D7o2aiyLYDnkw0R3JK0Qv1RQ==", 1483 | "dev": true 1484 | }, 1485 | "node_modules/follow-redirects": { 1486 | "version": "1.15.4", 1487 | "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.4.tgz", 1488 | "integrity": "sha512-Cr4D/5wlrb0z9dgERpUL3LrmPKVDsETIJhaCMeDfuFYcqa5bldGV6wBsAN6X/vxlXQtFBMrXdXxdL8CbDTGniw==", 1489 | "funding": [ 1490 | { 1491 | "type": "individual", 1492 | "url": "https://github.com/sponsors/RubenVerborgh" 1493 | } 1494 | ], 1495 | "engines": { 1496 | "node": ">=4.0" 1497 | }, 1498 | "peerDependenciesMeta": { 1499 | "debug": { 1500 | "optional": true 1501 | } 1502 | } 1503 | }, 1504 | "node_modules/fs.realpath": { 1505 | "version": "1.0.0", 1506 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 1507 | "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", 1508 | "dev": true 1509 | }, 1510 | "node_modules/fsevents": { 1511 | "version": "2.3.3", 1512 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", 1513 | "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", 1514 | "hasInstallScript": true, 1515 | "optional": true, 1516 | "os": [ 1517 | "darwin" 1518 | ], 1519 | "engines": { 1520 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 1521 | } 1522 | }, 1523 | "node_modules/get-caller-file": { 1524 | "version": "2.0.5", 1525 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 1526 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 1527 | "dev": true, 1528 | "engines": { 1529 | "node": "6.* || 8.* || >= 10.*" 1530 | } 1531 | }, 1532 | "node_modules/get-tsconfig": { 1533 | "version": "4.7.2", 1534 | "resolved": "https://registry.npmjs.org/get-tsconfig/-/get-tsconfig-4.7.2.tgz", 1535 | "integrity": "sha512-wuMsz4leaj5hbGgg4IvDU0bqJagpftG5l5cXIAvo8uZrqn0NJqwtfupTN00VnkQJPcIRrxYrm1Ue24btpCha2A==", 1536 | "dependencies": { 1537 | "resolve-pkg-maps": "^1.0.0" 1538 | }, 1539 | "funding": { 1540 | "url": "https://github.com/privatenumber/get-tsconfig?sponsor=1" 1541 | } 1542 | }, 1543 | "node_modules/glob": { 1544 | "version": "7.2.3", 1545 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", 1546 | "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", 1547 | "dev": true, 1548 | "dependencies": { 1549 | "fs.realpath": "^1.0.0", 1550 | "inflight": "^1.0.4", 1551 | "inherits": "2", 1552 | "minimatch": "^3.1.1", 1553 | "once": "^1.3.0", 1554 | "path-is-absolute": "^1.0.0" 1555 | }, 1556 | "engines": { 1557 | "node": "*" 1558 | }, 1559 | "funding": { 1560 | "url": "https://github.com/sponsors/isaacs" 1561 | } 1562 | }, 1563 | "node_modules/glob-parent": { 1564 | "version": "6.0.2", 1565 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", 1566 | "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", 1567 | "dev": true, 1568 | "dependencies": { 1569 | "is-glob": "^4.0.3" 1570 | }, 1571 | "engines": { 1572 | "node": ">=10.13.0" 1573 | } 1574 | }, 1575 | "node_modules/globals": { 1576 | "version": "13.24.0", 1577 | "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", 1578 | "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", 1579 | "dev": true, 1580 | "dependencies": { 1581 | "type-fest": "^0.20.2" 1582 | }, 1583 | "engines": { 1584 | "node": ">=8" 1585 | }, 1586 | "funding": { 1587 | "url": "https://github.com/sponsors/sindresorhus" 1588 | } 1589 | }, 1590 | "node_modules/globby": { 1591 | "version": "11.1.0", 1592 | "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz", 1593 | "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", 1594 | "dev": true, 1595 | "dependencies": { 1596 | "array-union": "^2.1.0", 1597 | "dir-glob": "^3.0.1", 1598 | "fast-glob": "^3.2.9", 1599 | "ignore": "^5.2.0", 1600 | "merge2": "^1.4.1", 1601 | "slash": "^3.0.0" 1602 | }, 1603 | "engines": { 1604 | "node": ">=10" 1605 | }, 1606 | "funding": { 1607 | "url": "https://github.com/sponsors/sindresorhus" 1608 | } 1609 | }, 1610 | "node_modules/graphemer": { 1611 | "version": "1.4.0", 1612 | "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", 1613 | "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", 1614 | "dev": true 1615 | }, 1616 | "node_modules/has-flag": { 1617 | "version": "4.0.0", 1618 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 1619 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 1620 | "engines": { 1621 | "node": ">=8" 1622 | } 1623 | }, 1624 | "node_modules/http-graceful-shutdown": { 1625 | "version": "3.1.13", 1626 | "resolved": "https://registry.npmjs.org/http-graceful-shutdown/-/http-graceful-shutdown-3.1.13.tgz", 1627 | "integrity": "sha512-Ci5LRufQ8AtrQ1U26AevS8QoMXDOhnAHCJI3eZu1com7mZGHxREmw3dNj85ftpQokQCvak8nI2pnFS8zyM1M+Q==", 1628 | "dependencies": { 1629 | "debug": "^4.3.4" 1630 | }, 1631 | "engines": { 1632 | "node": ">=4.0.0" 1633 | } 1634 | }, 1635 | "node_modules/http-proxy": { 1636 | "version": "1.18.1", 1637 | "resolved": "https://registry.npmjs.org/http-proxy/-/http-proxy-1.18.1.tgz", 1638 | "integrity": "sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ==", 1639 | "dependencies": { 1640 | "eventemitter3": "^4.0.0", 1641 | "follow-redirects": "^1.0.0", 1642 | "requires-port": "^1.0.0" 1643 | }, 1644 | "engines": { 1645 | "node": ">=8.0.0" 1646 | } 1647 | }, 1648 | "node_modules/ignore": { 1649 | "version": "5.3.0", 1650 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.0.tgz", 1651 | "integrity": "sha512-g7dmpshy+gD7mh88OC9NwSGTKoc3kyLAZQRU1mt53Aw/vnvfXnbC+F/7F7QoYVKbV+KNvJx8wArewKy1vXMtlg==", 1652 | "dev": true, 1653 | "engines": { 1654 | "node": ">= 4" 1655 | } 1656 | }, 1657 | "node_modules/import-fresh": { 1658 | "version": "3.3.0", 1659 | "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", 1660 | "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", 1661 | "dev": true, 1662 | "dependencies": { 1663 | "parent-module": "^1.0.0", 1664 | "resolve-from": "^4.0.0" 1665 | }, 1666 | "engines": { 1667 | "node": ">=6" 1668 | }, 1669 | "funding": { 1670 | "url": "https://github.com/sponsors/sindresorhus" 1671 | } 1672 | }, 1673 | "node_modules/imurmurhash": { 1674 | "version": "0.1.4", 1675 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 1676 | "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", 1677 | "dev": true, 1678 | "engines": { 1679 | "node": ">=0.8.19" 1680 | } 1681 | }, 1682 | "node_modules/inflight": { 1683 | "version": "1.0.6", 1684 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 1685 | "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", 1686 | "dev": true, 1687 | "dependencies": { 1688 | "once": "^1.3.0", 1689 | "wrappy": "1" 1690 | } 1691 | }, 1692 | "node_modules/inherits": { 1693 | "version": "2.0.4", 1694 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 1695 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 1696 | "dev": true 1697 | }, 1698 | "node_modules/is-builtin-module": { 1699 | "version": "3.2.1", 1700 | "resolved": "https://registry.npmjs.org/is-builtin-module/-/is-builtin-module-3.2.1.tgz", 1701 | "integrity": "sha512-BSLE3HnV2syZ0FK0iMA/yUGplUeMmNz4AW5fnTunbCIqZi4vG3WjJT9FHMy5D69xmAYBHXQhJdALdpwVxV501A==", 1702 | "dev": true, 1703 | "dependencies": { 1704 | "builtin-modules": "^3.3.0" 1705 | }, 1706 | "engines": { 1707 | "node": ">=6" 1708 | }, 1709 | "funding": { 1710 | "url": "https://github.com/sponsors/sindresorhus" 1711 | } 1712 | }, 1713 | "node_modules/is-extglob": { 1714 | "version": "2.1.1", 1715 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 1716 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", 1717 | "dev": true, 1718 | "engines": { 1719 | "node": ">=0.10.0" 1720 | } 1721 | }, 1722 | "node_modules/is-fullwidth-code-point": { 1723 | "version": "3.0.0", 1724 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 1725 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 1726 | "dev": true, 1727 | "engines": { 1728 | "node": ">=8" 1729 | } 1730 | }, 1731 | "node_modules/is-glob": { 1732 | "version": "4.0.3", 1733 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 1734 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 1735 | "dev": true, 1736 | "dependencies": { 1737 | "is-extglob": "^2.1.1" 1738 | }, 1739 | "engines": { 1740 | "node": ">=0.10.0" 1741 | } 1742 | }, 1743 | "node_modules/is-number": { 1744 | "version": "7.0.0", 1745 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 1746 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 1747 | "dev": true, 1748 | "engines": { 1749 | "node": ">=0.12.0" 1750 | } 1751 | }, 1752 | "node_modules/is-path-inside": { 1753 | "version": "3.0.3", 1754 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", 1755 | "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", 1756 | "dev": true, 1757 | "engines": { 1758 | "node": ">=8" 1759 | } 1760 | }, 1761 | "node_modules/isexe": { 1762 | "version": "2.0.0", 1763 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 1764 | "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", 1765 | "dev": true 1766 | }, 1767 | "node_modules/jake": { 1768 | "version": "10.8.7", 1769 | "resolved": "https://registry.npmjs.org/jake/-/jake-10.8.7.tgz", 1770 | "integrity": "sha512-ZDi3aP+fG/LchyBzUM804VjddnwfSfsdeYkwt8NcbKRvo4rFkjhs456iLFn3k2ZUWvNe4i48WACDbza8fhq2+w==", 1771 | "dependencies": { 1772 | "async": "^3.2.3", 1773 | "chalk": "^4.0.2", 1774 | "filelist": "^1.0.4", 1775 | "minimatch": "^3.1.2" 1776 | }, 1777 | "bin": { 1778 | "jake": "bin/cli.js" 1779 | }, 1780 | "engines": { 1781 | "node": ">=10" 1782 | } 1783 | }, 1784 | "node_modules/js-yaml": { 1785 | "version": "4.1.0", 1786 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", 1787 | "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", 1788 | "dev": true, 1789 | "dependencies": { 1790 | "argparse": "^2.0.1" 1791 | }, 1792 | "bin": { 1793 | "js-yaml": "bin/js-yaml.js" 1794 | } 1795 | }, 1796 | "node_modules/jsdoc-type-pratt-parser": { 1797 | "version": "4.0.0", 1798 | "resolved": "https://registry.npmjs.org/jsdoc-type-pratt-parser/-/jsdoc-type-pratt-parser-4.0.0.tgz", 1799 | "integrity": "sha512-YtOli5Cmzy3q4dP26GraSOeAhqecewG04hoO8DY56CH4KJ9Fvv5qKWUCCo3HZob7esJQHCv6/+bnTy72xZZaVQ==", 1800 | "dev": true, 1801 | "engines": { 1802 | "node": ">=12.0.0" 1803 | } 1804 | }, 1805 | "node_modules/json-buffer": { 1806 | "version": "3.0.1", 1807 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", 1808 | "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", 1809 | "dev": true 1810 | }, 1811 | "node_modules/json-schema-traverse": { 1812 | "version": "0.4.1", 1813 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", 1814 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", 1815 | "dev": true 1816 | }, 1817 | "node_modules/json-stable-stringify-without-jsonify": { 1818 | "version": "1.0.1", 1819 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", 1820 | "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", 1821 | "dev": true 1822 | }, 1823 | "node_modules/keyv": { 1824 | "version": "4.5.4", 1825 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", 1826 | "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", 1827 | "dev": true, 1828 | "dependencies": { 1829 | "json-buffer": "3.0.1" 1830 | } 1831 | }, 1832 | "node_modules/levn": { 1833 | "version": "0.4.1", 1834 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", 1835 | "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", 1836 | "dev": true, 1837 | "dependencies": { 1838 | "prelude-ls": "^1.2.1", 1839 | "type-check": "~0.4.0" 1840 | }, 1841 | "engines": { 1842 | "node": ">= 0.8.0" 1843 | } 1844 | }, 1845 | "node_modules/locate-path": { 1846 | "version": "6.0.0", 1847 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", 1848 | "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", 1849 | "dev": true, 1850 | "dependencies": { 1851 | "p-locate": "^5.0.0" 1852 | }, 1853 | "engines": { 1854 | "node": ">=10" 1855 | }, 1856 | "funding": { 1857 | "url": "https://github.com/sponsors/sindresorhus" 1858 | } 1859 | }, 1860 | "node_modules/lodash": { 1861 | "version": "4.17.21", 1862 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", 1863 | "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", 1864 | "dev": true 1865 | }, 1866 | "node_modules/lodash.merge": { 1867 | "version": "4.6.2", 1868 | "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", 1869 | "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", 1870 | "dev": true 1871 | }, 1872 | "node_modules/lru-cache": { 1873 | "version": "6.0.0", 1874 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", 1875 | "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", 1876 | "dev": true, 1877 | "dependencies": { 1878 | "yallist": "^4.0.0" 1879 | }, 1880 | "engines": { 1881 | "node": ">=10" 1882 | } 1883 | }, 1884 | "node_modules/merge2": { 1885 | "version": "1.4.1", 1886 | "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", 1887 | "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", 1888 | "dev": true, 1889 | "engines": { 1890 | "node": ">= 8" 1891 | } 1892 | }, 1893 | "node_modules/micromatch": { 1894 | "version": "4.0.5", 1895 | "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", 1896 | "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", 1897 | "dev": true, 1898 | "dependencies": { 1899 | "braces": "^3.0.2", 1900 | "picomatch": "^2.3.1" 1901 | }, 1902 | "engines": { 1903 | "node": ">=8.6" 1904 | } 1905 | }, 1906 | "node_modules/minimatch": { 1907 | "version": "3.1.2", 1908 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 1909 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 1910 | "dependencies": { 1911 | "brace-expansion": "^1.1.7" 1912 | }, 1913 | "engines": { 1914 | "node": "*" 1915 | } 1916 | }, 1917 | "node_modules/ms": { 1918 | "version": "2.1.2", 1919 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1920 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1921 | }, 1922 | "node_modules/natural-compare": { 1923 | "version": "1.4.0", 1924 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", 1925 | "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", 1926 | "dev": true 1927 | }, 1928 | "node_modules/once": { 1929 | "version": "1.4.0", 1930 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1931 | "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", 1932 | "dev": true, 1933 | "dependencies": { 1934 | "wrappy": "1" 1935 | } 1936 | }, 1937 | "node_modules/optionator": { 1938 | "version": "0.9.3", 1939 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", 1940 | "integrity": "sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==", 1941 | "dev": true, 1942 | "dependencies": { 1943 | "@aashutoshrathi/word-wrap": "^1.2.3", 1944 | "deep-is": "^0.1.3", 1945 | "fast-levenshtein": "^2.0.6", 1946 | "levn": "^0.4.1", 1947 | "prelude-ls": "^1.2.1", 1948 | "type-check": "^0.4.0" 1949 | }, 1950 | "engines": { 1951 | "node": ">= 0.8.0" 1952 | } 1953 | }, 1954 | "node_modules/p-limit": { 1955 | "version": "3.1.0", 1956 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", 1957 | "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", 1958 | "dev": true, 1959 | "dependencies": { 1960 | "yocto-queue": "^0.1.0" 1961 | }, 1962 | "engines": { 1963 | "node": ">=10" 1964 | }, 1965 | "funding": { 1966 | "url": "https://github.com/sponsors/sindresorhus" 1967 | } 1968 | }, 1969 | "node_modules/p-locate": { 1970 | "version": "5.0.0", 1971 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", 1972 | "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", 1973 | "dev": true, 1974 | "dependencies": { 1975 | "p-limit": "^3.0.2" 1976 | }, 1977 | "engines": { 1978 | "node": ">=10" 1979 | }, 1980 | "funding": { 1981 | "url": "https://github.com/sponsors/sindresorhus" 1982 | } 1983 | }, 1984 | "node_modules/parent-module": { 1985 | "version": "1.0.1", 1986 | "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", 1987 | "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", 1988 | "dev": true, 1989 | "dependencies": { 1990 | "callsites": "^3.0.0" 1991 | }, 1992 | "engines": { 1993 | "node": ">=6" 1994 | } 1995 | }, 1996 | "node_modules/path-exists": { 1997 | "version": "4.0.0", 1998 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 1999 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 2000 | "dev": true, 2001 | "engines": { 2002 | "node": ">=8" 2003 | } 2004 | }, 2005 | "node_modules/path-is-absolute": { 2006 | "version": "1.0.1", 2007 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 2008 | "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", 2009 | "dev": true, 2010 | "engines": { 2011 | "node": ">=0.10.0" 2012 | } 2013 | }, 2014 | "node_modules/path-key": { 2015 | "version": "3.1.1", 2016 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", 2017 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", 2018 | "dev": true, 2019 | "engines": { 2020 | "node": ">=8" 2021 | } 2022 | }, 2023 | "node_modules/path-type": { 2024 | "version": "4.0.0", 2025 | "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", 2026 | "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", 2027 | "dev": true, 2028 | "engines": { 2029 | "node": ">=8" 2030 | } 2031 | }, 2032 | "node_modules/picomatch": { 2033 | "version": "2.3.1", 2034 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 2035 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 2036 | "dev": true, 2037 | "engines": { 2038 | "node": ">=8.6" 2039 | }, 2040 | "funding": { 2041 | "url": "https://github.com/sponsors/jonschlinkert" 2042 | } 2043 | }, 2044 | "node_modules/prelude-ls": { 2045 | "version": "1.2.1", 2046 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", 2047 | "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", 2048 | "dev": true, 2049 | "engines": { 2050 | "node": ">= 0.8.0" 2051 | } 2052 | }, 2053 | "node_modules/promisify-child-process": { 2054 | "version": "4.1.2", 2055 | "resolved": "https://registry.npmjs.org/promisify-child-process/-/promisify-child-process-4.1.2.tgz", 2056 | "integrity": "sha512-APnkIgmaHNJpkAn7k+CrJSi9WMuff5ctYFbD0CO2XIPkM8yO7d/ShouU2clywbpHV/DUsyc4bpJCsNgddNtx4g==", 2057 | "engines": { 2058 | "node": ">=8" 2059 | } 2060 | }, 2061 | "node_modules/punycode": { 2062 | "version": "2.3.1", 2063 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", 2064 | "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", 2065 | "dev": true, 2066 | "engines": { 2067 | "node": ">=6" 2068 | } 2069 | }, 2070 | "node_modules/queue-microtask": { 2071 | "version": "1.2.3", 2072 | "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", 2073 | "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", 2074 | "dev": true, 2075 | "funding": [ 2076 | { 2077 | "type": "github", 2078 | "url": "https://github.com/sponsors/feross" 2079 | }, 2080 | { 2081 | "type": "patreon", 2082 | "url": "https://www.patreon.com/feross" 2083 | }, 2084 | { 2085 | "type": "consulting", 2086 | "url": "https://feross.org/support" 2087 | } 2088 | ] 2089 | }, 2090 | "node_modules/regenerator-runtime": { 2091 | "version": "0.14.1", 2092 | "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.14.1.tgz", 2093 | "integrity": "sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==", 2094 | "dev": true 2095 | }, 2096 | "node_modules/require-directory": { 2097 | "version": "2.1.1", 2098 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 2099 | "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", 2100 | "dev": true, 2101 | "engines": { 2102 | "node": ">=0.10.0" 2103 | } 2104 | }, 2105 | "node_modules/requires-port": { 2106 | "version": "1.0.0", 2107 | "resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz", 2108 | "integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==" 2109 | }, 2110 | "node_modules/resolve-from": { 2111 | "version": "4.0.0", 2112 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", 2113 | "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", 2114 | "dev": true, 2115 | "engines": { 2116 | "node": ">=4" 2117 | } 2118 | }, 2119 | "node_modules/resolve-pkg-maps": { 2120 | "version": "1.0.0", 2121 | "resolved": "https://registry.npmjs.org/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz", 2122 | "integrity": "sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==", 2123 | "funding": { 2124 | "url": "https://github.com/privatenumber/resolve-pkg-maps?sponsor=1" 2125 | } 2126 | }, 2127 | "node_modules/reusify": { 2128 | "version": "1.0.4", 2129 | "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", 2130 | "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", 2131 | "dev": true, 2132 | "engines": { 2133 | "iojs": ">=1.0.0", 2134 | "node": ">=0.10.0" 2135 | } 2136 | }, 2137 | "node_modules/rimraf": { 2138 | "version": "3.0.2", 2139 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", 2140 | "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", 2141 | "dev": true, 2142 | "dependencies": { 2143 | "glob": "^7.1.3" 2144 | }, 2145 | "bin": { 2146 | "rimraf": "bin.js" 2147 | }, 2148 | "funding": { 2149 | "url": "https://github.com/sponsors/isaacs" 2150 | } 2151 | }, 2152 | "node_modules/run-parallel": { 2153 | "version": "1.2.0", 2154 | "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", 2155 | "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", 2156 | "dev": true, 2157 | "funding": [ 2158 | { 2159 | "type": "github", 2160 | "url": "https://github.com/sponsors/feross" 2161 | }, 2162 | { 2163 | "type": "patreon", 2164 | "url": "https://www.patreon.com/feross" 2165 | }, 2166 | { 2167 | "type": "consulting", 2168 | "url": "https://feross.org/support" 2169 | } 2170 | ], 2171 | "dependencies": { 2172 | "queue-microtask": "^1.2.2" 2173 | } 2174 | }, 2175 | "node_modules/rxjs": { 2176 | "version": "7.8.1", 2177 | "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.8.1.tgz", 2178 | "integrity": "sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==", 2179 | "dev": true, 2180 | "dependencies": { 2181 | "tslib": "^2.1.0" 2182 | } 2183 | }, 2184 | "node_modules/semver": { 2185 | "version": "7.5.4", 2186 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", 2187 | "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", 2188 | "dev": true, 2189 | "dependencies": { 2190 | "lru-cache": "^6.0.0" 2191 | }, 2192 | "bin": { 2193 | "semver": "bin/semver.js" 2194 | }, 2195 | "engines": { 2196 | "node": ">=10" 2197 | } 2198 | }, 2199 | "node_modules/shebang-command": { 2200 | "version": "2.0.0", 2201 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", 2202 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", 2203 | "dev": true, 2204 | "dependencies": { 2205 | "shebang-regex": "^3.0.0" 2206 | }, 2207 | "engines": { 2208 | "node": ">=8" 2209 | } 2210 | }, 2211 | "node_modules/shebang-regex": { 2212 | "version": "3.0.0", 2213 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", 2214 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", 2215 | "dev": true, 2216 | "engines": { 2217 | "node": ">=8" 2218 | } 2219 | }, 2220 | "node_modules/shell-quote": { 2221 | "version": "1.8.1", 2222 | "resolved": "https://registry.npmjs.org/shell-quote/-/shell-quote-1.8.1.tgz", 2223 | "integrity": "sha512-6j1W9l1iAs/4xYBI1SYOVZyFcCis9b4KCLQ8fgAGG07QvzaRLVVRQvAy85yNmmZSjYjg4MWh4gNvlPujU/5LpA==", 2224 | "dev": true, 2225 | "funding": { 2226 | "url": "https://github.com/sponsors/ljharb" 2227 | } 2228 | }, 2229 | "node_modules/slash": { 2230 | "version": "3.0.0", 2231 | "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", 2232 | "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", 2233 | "dev": true, 2234 | "engines": { 2235 | "node": ">=8" 2236 | } 2237 | }, 2238 | "node_modules/spawn-command": { 2239 | "version": "0.0.2", 2240 | "resolved": "https://registry.npmjs.org/spawn-command/-/spawn-command-0.0.2.tgz", 2241 | "integrity": "sha512-zC8zGoGkmc8J9ndvml8Xksr1Amk9qBujgbF0JAIWO7kXr43w0h/0GJNM/Vustixu+YE8N/MTrQ7N31FvHUACxQ==", 2242 | "dev": true 2243 | }, 2244 | "node_modules/spdx-exceptions": { 2245 | "version": "2.3.0", 2246 | "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz", 2247 | "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==", 2248 | "dev": true 2249 | }, 2250 | "node_modules/spdx-expression-parse": { 2251 | "version": "3.0.1", 2252 | "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz", 2253 | "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==", 2254 | "dev": true, 2255 | "dependencies": { 2256 | "spdx-exceptions": "^2.1.0", 2257 | "spdx-license-ids": "^3.0.0" 2258 | } 2259 | }, 2260 | "node_modules/spdx-license-ids": { 2261 | "version": "3.0.16", 2262 | "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.16.tgz", 2263 | "integrity": "sha512-eWN+LnM3GR6gPu35WxNgbGl8rmY1AEmoMDvL/QD6zYmPWgywxWqJWNdLGT+ke8dKNWrcYgYjPpG5gbTfghP8rw==", 2264 | "dev": true 2265 | }, 2266 | "node_modules/string-width": { 2267 | "version": "4.2.3", 2268 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 2269 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 2270 | "dev": true, 2271 | "dependencies": { 2272 | "emoji-regex": "^8.0.0", 2273 | "is-fullwidth-code-point": "^3.0.0", 2274 | "strip-ansi": "^6.0.1" 2275 | }, 2276 | "engines": { 2277 | "node": ">=8" 2278 | } 2279 | }, 2280 | "node_modules/strip-ansi": { 2281 | "version": "6.0.1", 2282 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 2283 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 2284 | "dev": true, 2285 | "dependencies": { 2286 | "ansi-regex": "^5.0.1" 2287 | }, 2288 | "engines": { 2289 | "node": ">=8" 2290 | } 2291 | }, 2292 | "node_modules/strip-json-comments": { 2293 | "version": "3.1.1", 2294 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", 2295 | "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", 2296 | "dev": true, 2297 | "engines": { 2298 | "node": ">=8" 2299 | }, 2300 | "funding": { 2301 | "url": "https://github.com/sponsors/sindresorhus" 2302 | } 2303 | }, 2304 | "node_modules/supports-color": { 2305 | "version": "8.1.1", 2306 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", 2307 | "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", 2308 | "dev": true, 2309 | "dependencies": { 2310 | "has-flag": "^4.0.0" 2311 | }, 2312 | "engines": { 2313 | "node": ">=10" 2314 | }, 2315 | "funding": { 2316 | "url": "https://github.com/chalk/supports-color?sponsor=1" 2317 | } 2318 | }, 2319 | "node_modules/text-table": { 2320 | "version": "0.2.0", 2321 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", 2322 | "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", 2323 | "dev": true 2324 | }, 2325 | "node_modules/to-regex-range": { 2326 | "version": "5.0.1", 2327 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 2328 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 2329 | "dev": true, 2330 | "dependencies": { 2331 | "is-number": "^7.0.0" 2332 | }, 2333 | "engines": { 2334 | "node": ">=8.0" 2335 | } 2336 | }, 2337 | "node_modules/tree-kill": { 2338 | "version": "1.2.2", 2339 | "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", 2340 | "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", 2341 | "dev": true, 2342 | "bin": { 2343 | "tree-kill": "cli.js" 2344 | } 2345 | }, 2346 | "node_modules/ts-api-utils": { 2347 | "version": "1.0.3", 2348 | "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.0.3.tgz", 2349 | "integrity": "sha512-wNMeqtMz5NtwpT/UZGY5alT+VoKdSsOOP/kqHFcUW1P/VRhH2wJ48+DN2WwUliNbQ976ETwDL0Ifd2VVvgonvg==", 2350 | "dev": true, 2351 | "engines": { 2352 | "node": ">=16.13.0" 2353 | }, 2354 | "peerDependencies": { 2355 | "typescript": ">=4.2.0" 2356 | } 2357 | }, 2358 | "node_modules/tslib": { 2359 | "version": "2.6.2", 2360 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", 2361 | "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==", 2362 | "dev": true 2363 | }, 2364 | "node_modules/tsx": { 2365 | "version": "4.7.0", 2366 | "resolved": "https://registry.npmjs.org/tsx/-/tsx-4.7.0.tgz", 2367 | "integrity": "sha512-I+t79RYPlEYlHn9a+KzwrvEwhJg35h/1zHsLC2JXvhC2mdynMv6Zxzvhv5EMV6VF5qJlLlkSnMVvdZV3PSIGcg==", 2368 | "dependencies": { 2369 | "esbuild": "~0.19.10", 2370 | "get-tsconfig": "^4.7.2" 2371 | }, 2372 | "bin": { 2373 | "tsx": "dist/cli.mjs" 2374 | }, 2375 | "engines": { 2376 | "node": ">=18.0.0" 2377 | }, 2378 | "optionalDependencies": { 2379 | "fsevents": "~2.3.3" 2380 | } 2381 | }, 2382 | "node_modules/type-check": { 2383 | "version": "0.4.0", 2384 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", 2385 | "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", 2386 | "dev": true, 2387 | "dependencies": { 2388 | "prelude-ls": "^1.2.1" 2389 | }, 2390 | "engines": { 2391 | "node": ">= 0.8.0" 2392 | } 2393 | }, 2394 | "node_modules/type-fest": { 2395 | "version": "0.20.2", 2396 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", 2397 | "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", 2398 | "dev": true, 2399 | "engines": { 2400 | "node": ">=10" 2401 | }, 2402 | "funding": { 2403 | "url": "https://github.com/sponsors/sindresorhus" 2404 | } 2405 | }, 2406 | "node_modules/typescript": { 2407 | "version": "5.2.2", 2408 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz", 2409 | "integrity": "sha512-mI4WrpHsbCIcwT9cF4FZvr80QUeKvsUsUvKDoR+X/7XHQH98xYD8YHZg7ANtz2GtZt/CBq2QJ0thkGJMHfqc1w==", 2410 | "dev": true, 2411 | "bin": { 2412 | "tsc": "bin/tsc", 2413 | "tsserver": "bin/tsserver" 2414 | }, 2415 | "engines": { 2416 | "node": ">=14.17" 2417 | } 2418 | }, 2419 | "node_modules/undici-types": { 2420 | "version": "5.26.5", 2421 | "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", 2422 | "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==", 2423 | "dev": true 2424 | }, 2425 | "node_modules/uri-js": { 2426 | "version": "4.4.1", 2427 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", 2428 | "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", 2429 | "dev": true, 2430 | "dependencies": { 2431 | "punycode": "^2.1.0" 2432 | } 2433 | }, 2434 | "node_modules/which": { 2435 | "version": "2.0.2", 2436 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 2437 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 2438 | "dev": true, 2439 | "dependencies": { 2440 | "isexe": "^2.0.0" 2441 | }, 2442 | "bin": { 2443 | "node-which": "bin/node-which" 2444 | }, 2445 | "engines": { 2446 | "node": ">= 8" 2447 | } 2448 | }, 2449 | "node_modules/wrap-ansi": { 2450 | "version": "7.0.0", 2451 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", 2452 | "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", 2453 | "dev": true, 2454 | "dependencies": { 2455 | "ansi-styles": "^4.0.0", 2456 | "string-width": "^4.1.0", 2457 | "strip-ansi": "^6.0.0" 2458 | }, 2459 | "engines": { 2460 | "node": ">=10" 2461 | }, 2462 | "funding": { 2463 | "url": "https://github.com/chalk/wrap-ansi?sponsor=1" 2464 | } 2465 | }, 2466 | "node_modules/wrappy": { 2467 | "version": "1.0.2", 2468 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 2469 | "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", 2470 | "dev": true 2471 | }, 2472 | "node_modules/y18n": { 2473 | "version": "5.0.8", 2474 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", 2475 | "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", 2476 | "dev": true, 2477 | "engines": { 2478 | "node": ">=10" 2479 | } 2480 | }, 2481 | "node_modules/yallist": { 2482 | "version": "4.0.0", 2483 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", 2484 | "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", 2485 | "dev": true 2486 | }, 2487 | "node_modules/yargs": { 2488 | "version": "17.7.2", 2489 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", 2490 | "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", 2491 | "dev": true, 2492 | "dependencies": { 2493 | "cliui": "^8.0.1", 2494 | "escalade": "^3.1.1", 2495 | "get-caller-file": "^2.0.5", 2496 | "require-directory": "^2.1.1", 2497 | "string-width": "^4.2.3", 2498 | "y18n": "^5.0.5", 2499 | "yargs-parser": "^21.1.1" 2500 | }, 2501 | "engines": { 2502 | "node": ">=12" 2503 | } 2504 | }, 2505 | "node_modules/yargs-parser": { 2506 | "version": "21.1.1", 2507 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", 2508 | "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", 2509 | "dev": true, 2510 | "engines": { 2511 | "node": ">=12" 2512 | } 2513 | }, 2514 | "node_modules/yocto-queue": { 2515 | "version": "0.1.0", 2516 | "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", 2517 | "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", 2518 | "dev": true, 2519 | "engines": { 2520 | "node": ">=10" 2521 | }, 2522 | "funding": { 2523 | "url": "https://github.com/sponsors/sindresorhus" 2524 | } 2525 | } 2526 | } 2527 | } 2528 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "demo-kuma", 3 | "version": "2.0.0", 4 | "main": "server.js", 5 | "type": "module", 6 | "scripts": { 7 | "dev": "docker-compose -f compose-dev.yaml up", 8 | "start": "tsx server.ts", 9 | "build:docker-nightly": "docker buildx build --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/demo-kuma:nightly --target release -f Dockerfile . --push", 10 | "build:docker": "docker buildx build --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/demo-kuma:2 --target release -f Dockerfile . --push" 11 | }, 12 | "repository": { 13 | "type": "git", 14 | "url": "git+https://github.com/louislam/demo-kuma.git" 15 | }, 16 | "author": "Louis Lam", 17 | "license": "MIT", 18 | "bugs": { 19 | "url": "https://github.com/louislam/demo-kuma/issues" 20 | }, 21 | "homepage": "https://github.com/louislam/demo-kuma#readme", 22 | "dependencies": { 23 | "dotenv": "~10.0.0", 24 | "ejs": "~3.1.9", 25 | "http-graceful-shutdown": "~3.1.13", 26 | "http-proxy": "~1.18.1", 27 | "promisify-child-process": "~4.1.2", 28 | "tsx": "~4.7.0" 29 | }, 30 | "devDependencies": { 31 | "@types/ejs": "^3.1.5", 32 | "@types/http-proxy": "^1.17.14", 33 | "@types/node": "~20.10.6", 34 | "@typescript-eslint/eslint-plugin": "~6.8.0", 35 | "@typescript-eslint/parser": "~6.8.0", 36 | "concurrently": "^8.2.2", 37 | "cross-env": "~7.0.3", 38 | "eslint": "~8.50.0", 39 | "eslint-plugin-jsdoc": "~46.8.2", 40 | "typescript": "~5.2.2" 41 | } 42 | } 43 | -------------------------------------------------------------------------------- /pool.ts: -------------------------------------------------------------------------------- 1 | import childProcess from "child_process"; 2 | import childProcessAsync from "promisify-child-process"; 3 | import { sleep } from "./util"; 4 | import crypto from "crypto"; 5 | import { sessionTime, stackPrefix, startTimeout, servicePort, entryPath, dockerNetwork, serviceName } from "./config"; 6 | 7 | export class Pool { 8 | /** 9 | * sessionList[sessionID] = serviceURL 10 | */ 11 | sessionList: Record = {}; 12 | 13 | async startInstance() { 14 | let sessionID : string = ""; 15 | 16 | while (true) { 17 | sessionID = crypto.randomUUID().substring(0, 8); 18 | if (this.sessionList[sessionID] === undefined) { 19 | break; 20 | } 21 | } 22 | 23 | let timeout : NodeJS.Timeout; 24 | 25 | console.log(`[${sessionID}] Start a session`); 26 | 27 | await childProcessAsync.spawn("docker", [ 28 | "compose", 29 | "--file", "compose-demo.yaml", 30 | "-p", `${stackPrefix}-${sessionID}`, 31 | "up", 32 | "-d", 33 | ], { 34 | encoding: "utf-8", 35 | }); 36 | 37 | let startStackTime = Date.now(); 38 | let baseURL = ""; 39 | 40 | // Wait until the service is opened 41 | while (true) { 42 | try { 43 | let ip = await this.getServiceIP(sessionID); 44 | baseURL = `http://${ip}:${servicePort}`; 45 | let entryURL = baseURL + entryPath; 46 | 47 | console.log("Checking entry: " + entryURL); 48 | 49 | // Try to access the index page 50 | let res = await fetch(entryURL); 51 | await res.text(); 52 | 53 | if (res.status === 200) { 54 | break; 55 | } 56 | } catch (e) { 57 | } 58 | 59 | await sleep(2000); 60 | if (Date.now() - startStackTime > startTimeout * 1000) { 61 | throw new Error("Start instance timeout"); 62 | } 63 | } 64 | 65 | let endSessionTime = Date.now() + sessionTime * 1000; 66 | 67 | // Timer for closing the session 68 | setTimeout(async () => { 69 | console.log(`[${sessionID}] Time's up`); 70 | await this.stopInstance(sessionID); 71 | delete this.sessionList[sessionID]; 72 | }, (sessionTime) * 1000); 73 | 74 | this.sessionList[sessionID] = baseURL; 75 | console.log(`[${sessionID}] Session started`); 76 | 77 | return { 78 | sessionID, 79 | endSessionTime, 80 | }; 81 | } 82 | 83 | async stopInstance(sessionID : string) { 84 | await childProcessAsync.spawn("docker", [ 85 | "compose", 86 | "-f", "compose-demo.yaml", 87 | "-p", `${stackPrefix}-${sessionID}`, 88 | "down", 89 | "--volumes", 90 | "--remove-orphans", 91 | ], { 92 | encoding: "utf-8", 93 | env: { 94 | ...process.env, 95 | }, 96 | }); 97 | } 98 | 99 | getServiceURL(sessionID : string) : string | undefined { 100 | return this.sessionList[sessionID]; 101 | } 102 | 103 | async getServiceIP(sessionID : string) : Promise { 104 | let response = await childProcessAsync.spawn("docker", [ 105 | "inspect", 106 | `${stackPrefix}-${sessionID}-${serviceName}-1`, 107 | "--format", "json" 108 | ], { 109 | encoding: "utf-8", 110 | }); 111 | 112 | if (typeof response.stdout !== "string") { 113 | throw new Error("No output"); 114 | } 115 | 116 | let array = JSON.parse(response.stdout); 117 | 118 | if (!Array.isArray(array)) { 119 | throw new Error("Not an array"); 120 | } 121 | 122 | if (array.length === 0) { 123 | throw new Error("Array is empty"); 124 | } 125 | 126 | let obj = array[0]; 127 | 128 | // Check if the object is valid 129 | if (!obj || typeof obj !== "object") { 130 | throw new Error("Not an object"); 131 | } 132 | 133 | let networkSettings = obj.NetworkSettings; 134 | 135 | if (!networkSettings) { 136 | throw new Error("No network settings"); 137 | } 138 | 139 | let networks = obj.NetworkSettings.Networks; 140 | 141 | if (!networks) { 142 | throw new Error("No networks"); 143 | } 144 | 145 | // Find the target network 146 | let network = networks[dockerNetwork]; 147 | 148 | if (!network) { 149 | throw new Error("Network not found"); 150 | } 151 | 152 | let ip = network.IPAddress; 153 | 154 | if (!ip) { 155 | throw new Error("IP not found"); 156 | } 157 | 158 | return ip; 159 | } 160 | 161 | async clearInstance() { 162 | let result = await childProcessAsync.spawn("docker", [ 163 | "compose", 164 | "ls", 165 | "--format", "json", 166 | ], { 167 | encoding: "utf-8", 168 | }); 169 | 170 | if (typeof result.stdout === "string") { 171 | let list = JSON.parse(result.stdout); 172 | for (let stack of list) { 173 | if (stack.Name?.startsWith(stackPrefix + "-")) { 174 | console.log(`Clearing ${stack.Name}`); 175 | let result = await childProcessAsync.spawn("docker", [ 176 | "compose", 177 | "--file", "compose-demo.yaml", 178 | "-p", stack.Name, 179 | "down", 180 | "--volumes", 181 | "--remove-orphans", 182 | ], { 183 | encoding: "utf-8", 184 | }); 185 | 186 | console.log(result.stdout, result.stderr); 187 | } 188 | } 189 | } 190 | } 191 | } 192 | 193 | -------------------------------------------------------------------------------- /public/demo-kuma/main.css: -------------------------------------------------------------------------------- 1 | html, body { 2 | margin: 0; 3 | overflow: hidden; 4 | } 5 | 6 | body { 7 | font-family: BlinkMacSystemFont, segoe ui, Roboto, helvetica neue, Arial, noto sans, sans-serif, apple color emoji, segoe ui emoji, segoe ui symbol, noto color emoji; 8 | font-size: 15px; 9 | color: #F0F6FC; 10 | text-align: center; 11 | background-color: #090C10; 12 | } 13 | 14 | .btn-primary { 15 | background: linear-gradient(135deg, #74c2ff 0%, #74c2ff 75%, #86e6a9); 16 | border-color: #54DC8C; 17 | border-radius: 50rem !important; 18 | color: black; 19 | } 20 | .btn-primary:hover { 21 | border-color: #54DC8C; 22 | color: black; 23 | } 24 | 25 | iframe { 26 | width: 100%; 27 | height: 100vh; 28 | border: 0; 29 | margin: 0; 30 | padding: 0; 31 | } 32 | 33 | .main { 34 | height: 100vh; 35 | display: flex; 36 | justify-content: center; 37 | align-items: center; 38 | flex-direction: column; 39 | } 40 | 41 | .fixed { 42 | background-color: rgba(0, 0, 0, 0.5); 43 | position: fixed; 44 | bottom: 15px; 45 | right: 15px; 46 | } 47 | 48 | /*# sourceMappingURL=main.css.map */ 49 | -------------------------------------------------------------------------------- /public/demo-kuma/main.scss: -------------------------------------------------------------------------------- 1 | html, body { 2 | margin: 0; 3 | overflow: hidden; 4 | } 5 | 6 | body { 7 | font-family: BlinkMacSystemFont, segoe ui, Roboto, helvetica neue, Arial, noto sans, sans-serif, apple color emoji, segoe ui emoji, segoe ui symbol, noto color emoji; 8 | font-size: 15px; 9 | color: #F0F6FC; 10 | text-align: center; 11 | background-color: #090C10; 12 | } 13 | 14 | .btn-primary { 15 | background: linear-gradient(135deg, #74c2ff 0%, #74c2ff 75%, #86e6a9); 16 | border-color: #54DC8C; 17 | border-radius: 50rem !important; 18 | color: black; 19 | 20 | &:hover { 21 | border-color: #54DC8C; 22 | color: black; 23 | } 24 | } 25 | 26 | iframe { 27 | width: 100%; 28 | height: 100vh; 29 | border: 0; 30 | margin: 0; 31 | padding: 0; 32 | } 33 | 34 | .main { 35 | height: 100vh; 36 | display: flex; 37 | // center 38 | justify-content: center; 39 | align-items: center; 40 | flex-direction: column; 41 | } 42 | 43 | .fixed { 44 | background-color: rgba(0, 0, 0, 0.5); 45 | position: fixed; 46 | bottom: 15px; 47 | right: 15px; 48 | } 49 | -------------------------------------------------------------------------------- /server.ts: -------------------------------------------------------------------------------- 1 | import "dotenv/config"; 2 | import { Pool } from "./pool"; 3 | import * as http from "http"; 4 | import httpProxy from "http-proxy"; 5 | import ejs from "ejs"; 6 | import fs from "fs/promises"; 7 | import path from "path"; 8 | import { contentType, sleep } from "./util"; 9 | import gracefulShutdown from "http-graceful-shutdown"; 10 | import { entryPath, installURL, serverPort, sessionTime, showEntry, websiteName } from "./config"; 11 | 12 | // Catch unexpected errors here 13 | let unexpectedErrorHandler = (error : unknown) => { 14 | console.trace(error); 15 | }; 16 | process.addListener("unhandledRejection", unexpectedErrorHandler); 17 | process.addListener("uncaughtException", unexpectedErrorHandler); 18 | 19 | const pool = new Pool(); 20 | const proxy = httpProxy.createProxyServer(); 21 | 22 | await pool.clearInstance(); 23 | 24 | const server = http.createServer(async (req, res) => { 25 | try { 26 | await requestHandler(req, res); 27 | } catch (e) { 28 | console.error(e); 29 | res.writeHead(500); 30 | res.end("Internal server error"); 31 | } 32 | }); 33 | 34 | console.log(`Listening on port ${serverPort}`); 35 | server.listen(serverPort); 36 | 37 | gracefulShutdown(server, { 38 | signals: "SIGINT SIGTERM", 39 | timeout: 30000, // timeout: 30 secs 40 | development: false, // not in dev mode 41 | forceExit: true, // triggers process.exit() at the end of shutdown process 42 | onShutdown: shutdownFunction, // shutdown function (async) - e.g. for cleanup DB, ... 43 | finally: finalFunction, // finally function (sync) - e.g. for logging 44 | }); 45 | 46 | /** 47 | * Get session ID from cookie 48 | * @param req 49 | */ 50 | function getSessionID(req : http.IncomingMessage) { 51 | let cookieList = req.headers.cookie?.split(";") || []; 52 | let sessionID = ""; 53 | 54 | for (let cookie of cookieList) { 55 | let [ key, value ] = cookie.split("="); 56 | if (key.trim() === "session-id") { 57 | sessionID = value.trim(); 58 | } 59 | } 60 | 61 | return sessionID; 62 | } 63 | 64 | async function requestHandler(req : http.IncomingMessage, res : http.ServerResponse) { 65 | if (!req.url) { 66 | res.end("No url"); 67 | return; 68 | } 69 | 70 | // Handle request 71 | if (req.url === "/") { 72 | let sessionID = getSessionID(req); 73 | let target = pool.getServiceURL(sessionID); 74 | 75 | // If a session is found, proxy it 76 | if (sessionID && target) { 77 | await proxyWeb(req, res); 78 | } else { 79 | // Redirect to "/start" 80 | res.writeHead(302, { 81 | "Location": "/start-demo", 82 | }); 83 | res.end(); 84 | } 85 | 86 | } else if (req.url === "/start-demo" || req.url === "/start-demo") { 87 | res.writeHead(200, { "Content-Type": "text/html" }); 88 | let indexTemplate = ejs.render(await fs.readFile("./views/index.ejs", "utf-8"), { 89 | websiteName, 90 | installURL, 91 | autoStart: !showEntry, 92 | entryPath, 93 | }); 94 | res.end(indexTemplate); 95 | 96 | } else if (req.url.startsWith("/demo-kuma/")) { 97 | if (req.url === "/demo" || req.url === "/demo-kuma/") { 98 | res.writeHead(200, { "Content-Type": "text/html" }); 99 | let indexTemplate = ejs.render(await fs.readFile("./views/index.ejs", "utf-8"), { 100 | websiteName, 101 | installURL, 102 | autoStart: !showEntry, 103 | entryPath, 104 | }); 105 | res.end(indexTemplate); 106 | 107 | } else if (req.url === "/demo-kuma/start-instance") { 108 | try { 109 | let { endSessionTime, sessionID } = await pool.startInstance(); 110 | res.writeHead(200, { 111 | "Content-Type": "application/json", 112 | "Set-Cookie": `session-id=${sessionID}; Max-Age=${sessionTime}; Path=/;` 113 | }); 114 | res.end(JSON.stringify({ 115 | ok: true, 116 | sessionID, 117 | endSessionTime, 118 | })); 119 | } catch (e) { 120 | console.error(e); 121 | res.writeHead(500, { "Content-Type": "application/json" }); 122 | res.end(JSON.stringify({ 123 | ok: false, 124 | })); 125 | } 126 | 127 | } else if (req.url === "/demo-kuma/validate-session") { 128 | let sessionID = getSessionID(req); 129 | 130 | res.writeHead(200, { 131 | "Content-Type": "application/json", 132 | }); 133 | 134 | res.end(JSON.stringify({ 135 | ok: pool.sessionList[sessionID] !== undefined, 136 | })); 137 | 138 | } else { 139 | try { 140 | let data = await fs.readFile(path.join("./public", req.url)); 141 | res.writeHead(200, { "Content-Type": contentType(req.url) }); 142 | res.end(data); 143 | } catch (e) { 144 | res.writeHead(404); 145 | res.end("Not found"); 146 | } 147 | } 148 | 149 | } else { 150 | await proxyWeb(req, res); 151 | } 152 | } 153 | 154 | async function proxyWeb(req : http.IncomingMessage, res : http.ServerResponse, retryCount = 0) { 155 | // Get the sessionID from cookie 156 | let sessionID = getSessionID(req); 157 | let target = pool.getServiceURL(sessionID); 158 | 159 | if (sessionID && target) { 160 | proxy.web(req, res, { 161 | target, 162 | }, async (err) => { 163 | if (retryCount <= 10) { 164 | await sleep(2000); 165 | await proxyWeb(req, res, retryCount + 1); 166 | } else { 167 | res.writeHead(500); 168 | res.end("Unable to connect to the instance"); 169 | } 170 | }); 171 | } else { 172 | res.writeHead(404); 173 | res.end("Session not found"); 174 | } 175 | } 176 | 177 | async function shutdownFunction(signal : string | undefined) { 178 | console.info("Shutdown requested"); 179 | console.info("server", "Called signal: " + signal); 180 | await pool.clearInstance(); 181 | } 182 | 183 | /** 184 | * Final function called before application exits 185 | */ 186 | function finalFunction() { 187 | console.info("Graceful shutdown successful!"); 188 | } 189 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "module": "ESNext", 4 | "target": "ESNext", 5 | "strict": true, 6 | "moduleResolution": "bundler", 7 | "skipLibCheck": true 8 | }, 9 | "include": [ 10 | "backend/**/*" 11 | ] 12 | } 13 | -------------------------------------------------------------------------------- /util.ts: -------------------------------------------------------------------------------- 1 | import fs from "fs"; 2 | 3 | function getRandomArbitrary(min, max) { 4 | return Math.random() * (max - min) + min; 5 | } 6 | 7 | export function getRandomInt(min, max) { 8 | min = Math.ceil(min); 9 | max = Math.floor(max); 10 | return Math.floor(Math.random() * (max - min + 1)) + min; 11 | } 12 | 13 | export function sleep(ms) { 14 | return new Promise(resolve => setTimeout(resolve, ms)); 15 | } 16 | 17 | /** 18 | * Get the content type of a URL 19 | * @param url 20 | */ 21 | export function contentType(url : string) { 22 | if (url.endsWith(".css")) { 23 | return "text/css"; 24 | } else if (url.endsWith(".js")) { 25 | return "text/javascript"; 26 | } else { 27 | return "text/html"; 28 | } 29 | } 30 | 31 | export function fileExists(file : string) { 32 | return fs.promises.access(file, fs.constants.F_OK) 33 | .then(() => true) 34 | .catch(() => false); 35 | } 36 | 37 | -------------------------------------------------------------------------------- /views/index.ejs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | <%= websiteName %> 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 |
13 | 14 |
15 |
16 |

<%= websiteName %>

17 |

18 | This is a demo of <%= websiteName %>. You can use this demo to try out the software before installing it on your own server. 19 |

20 | 21 |
22 | 23 |
24 |

Demo has Ended

25 |

26 | Your demo has ended, you can start a new demo by refreshing the page. 27 |

28 | Install now 29 |
30 | 31 |
32 | Preparing your demo, please wait... 33 |
34 | 35 |
36 | 37 |
38 | {{ countdown }} Install now 39 |
40 | 41 | 42 |
43 | 44 | 45 | 46 | 158 | 159 | 160 | 161 | --------------------------------------------------------------------------------