├── .gitattributes ├── .github └── workflows │ └── build.yml ├── .gitignore ├── .gitmodules ├── .prettierignore ├── .prettierrc.yml ├── embed.js ├── index.spec.ts ├── index.ts ├── license ├── package-lock.json ├── package.json ├── readme.md ├── terser.json ├── tsconfig.json └── tslint.json /.gitattributes: -------------------------------------------------------------------------------- 1 | * text=auto eol=lf -------------------------------------------------------------------------------- /.github/workflows/build.yml: -------------------------------------------------------------------------------- 1 | name: build 2 | on: 3 | push: 4 | branches: 5 | - master 6 | pull_request: 7 | jobs: 8 | build: 9 | runs-on: ubuntu-latest 10 | steps: 11 | - uses: actions/checkout@v3 12 | with: 13 | submodules: recursive 14 | - uses: actions/setup-node@v3 15 | with: 16 | node-version: 16 17 | - uses: supercharge/redis-github-action@1.4.0 18 | with: 19 | redis-version: 6 20 | - run: npm ci 21 | - run: npm run test:verify 22 | - uses: codecov/codecov-action@v3 23 | with: 24 | directory: ./coverage -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Modules 2 | node_modules/ 3 | 4 | # Built files 5 | coverage/ 6 | .*/ 7 | *.map 8 | *.tgz 9 | *.log 10 | *.rdb 11 | *.js 12 | *.d.ts 13 | 14 | # Config files 15 | !.github/ 16 | !embed.js -------------------------------------------------------------------------------- /.gitmodules: -------------------------------------------------------------------------------- 1 | [submodule "script"] 2 | path = script 3 | url = https://github.com/plsmphnx/redis-bucket-script 4 | -------------------------------------------------------------------------------- /.prettierignore: -------------------------------------------------------------------------------- 1 | # Ignore everything by default 2 | * 3 | 4 | # Enable for top-level files and our source code 5 | !/* 6 | 7 | # Ignore built files 8 | /index*.js* 9 | /index*.d.ts* 10 | 11 | # Ignore files that don't currently have formatters 12 | .* 13 | license 14 | *.lua 15 | *.tgz 16 | *.rdb 17 | 18 | # Ignore the npm files because it will format 19 | # them regardless of what we choose to do 20 | package.json 21 | package-lock.json -------------------------------------------------------------------------------- /.prettierrc.yml: -------------------------------------------------------------------------------- 1 | arrowParens: avoid 2 | proseWrap: always 3 | singleQuote: true 4 | tabWidth: 4 5 | -------------------------------------------------------------------------------- /embed.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Copyright (c) Microsoft Corporation. 3 | * Licensed under the MIT License. 4 | */ 5 | 6 | import { readFileSync, writeFileSync } from 'fs'; 7 | 8 | const script = 9 | process.argv[2] === 'minify' 10 | ? 'script/bucket.min.lua' 11 | : 'script/bucket.lua'; 12 | 13 | // Read built sources 14 | const file = readFileSync('index.js', 'utf8'); 15 | const code = readFileSync(script, 'utf8'); 16 | const hash = readFileSync(script + '.sha1', 'utf8'); 17 | 18 | // Replace placeholder strings in source 19 | const body = file 20 | .replace(`'{{LUA_CODE}}'`, JSON.stringify(code)) 21 | .replace(`'{{LUA_HASH}}'`, JSON.stringify(hash)); 22 | 23 | // Output updated source 24 | writeFileSync('index.js', body); 25 | -------------------------------------------------------------------------------- /index.spec.ts: -------------------------------------------------------------------------------- 1 | /*! 2 | * Copyright (c) Microsoft Corporation. 3 | * Licensed under the MIT License. 4 | */ 5 | 6 | import test, { ExecutionContext } from 'ava'; 7 | import * as redis from 'redis'; 8 | 9 | import * as limiter from './index.js'; 10 | 11 | // The number of actions allowed in a burst (assuming one attempt per second, 12 | // to make the simplifying assumption that actions and time are equivalent) is 13 | // calculated from the burst capacity and the flow returned over that duration; 14 | // it is the solution to the equation: time = burst + (time * flow) 15 | function calcTime(rate: limiter.Rate, bound = false): number { 16 | // If a zero bound applies to this calculation, 17 | // the flow rate is not applied to the first check 18 | return +bound + (rate.burst - +bound) / (1 - rate.flow); 19 | } 20 | 21 | // When in constant flow, the flow value determines the number of total attempts 22 | // per allowed action 23 | function calcLoop(rate: limiter.Rate): number { 24 | return 1 / rate.flow; 25 | } 26 | 27 | // When calculating remaining capacity, consider the number of used actions, 28 | // and the amount of this metric that would have been returned over that time 29 | function calcLeft(rate: limiter.Rate, used: number, bound = false): number { 30 | // If a zero bound applies to this calculation, 31 | // the flow rate is not applied to the first check 32 | return rate.burst - (+bound + (used - +bound) * (1 - rate.flow)); 33 | } 34 | 35 | // Shorthand for a 0..[n-1] array 36 | function repeat(times: number) { 37 | return Array.from(Array(times).keys()); 38 | } 39 | 40 | // Run a test against a live Redis instance 41 | function it( 42 | desc: string, 43 | cb: ( 44 | t: ExecutionContext, 45 | config: limiter.Config, 46 | key: string, 47 | now: () => number, 48 | sleep: (s: number) => Promise 49 | ) => Promise 50 | ) { 51 | test(desc, async t => { 52 | // Generate unique keys based on the test description 53 | const id = desc.replace(/ /g, '-'); 54 | const key = `redis-bucket-test:key:${id}`; 55 | const time = `redis-bucket-test:time:${id}`; 56 | 57 | // Connect to Redis using the default values 58 | const raw = redis.createClient(); 59 | await raw.connect(); 60 | 61 | // Translate the limiter.Config format to the Redis client format 62 | const config: limiter.Config = { 63 | async eval(script, keys, argv) { 64 | // Patch the script, using a list to perform a controlled mock 65 | // of the time function 66 | return raw.eval( 67 | script.replace(`'time'`, `'lrange','${time}',0,1`), 68 | { keys, arguments: argv.map(String) } 69 | ); 70 | }, 71 | async evalsha(hash, keys, argv) { 72 | // This will always fail since the sha1 hash will not match 73 | // the patched script, but it validates the fallback path 74 | return raw.evalSha(hash, { keys, arguments: argv.map(String) }); 75 | }, 76 | }; 77 | 78 | // Methods to utilize the controlled time mock 79 | let seconds = 1; 80 | await raw.lPush(time, ['0', '1']); 81 | const now = () => seconds; 82 | const sleep = async (s: number) => { 83 | seconds += s; 84 | await raw.lPush(time, [ 85 | String(Math.floor((seconds % 1) * 1e6)), // Microseconds 86 | String(Math.floor(seconds)), // Full seconds 87 | ]); 88 | }; 89 | 90 | // Execute the test 91 | await cb(t, config, key, now, sleep); 92 | 93 | // Clean up any keys the test may have generated 94 | await raw.del([key, time]); 95 | await raw.quit(); 96 | }); 97 | } 98 | 99 | it('performs basic validation', async (t, config) => { 100 | t.throws(() => limiter.create({ ...config }), { instanceOf: RangeError }); 101 | t.throws( 102 | () => 103 | limiter.create({ 104 | ...config, 105 | capacity: { window: -60, min: 10, max: 20 }, 106 | }), 107 | { instanceOf: RangeError } 108 | ); 109 | t.throws( 110 | () => 111 | limiter.create({ 112 | ...config, 113 | capacity: { window: 60, min: 10, max: 10 }, 114 | }), 115 | { instanceOf: RangeError } 116 | ); 117 | t.throws(() => limiter.create({ ...config, rate: { flow: 1, burst: 0 } }), { 118 | instanceOf: RangeError, 119 | }); 120 | }); 121 | 122 | it('handles basic capacity metrics', async (t, config, key, now, sleep) => { 123 | const capacity: limiter.Capacity = { window: 60, min: 10, max: 20 }; 124 | const limit = limiter.create({ ...config, capacity }); 125 | 126 | // Perform test twice, for burst and steady-state near capacity 127 | for (const {} of repeat(2)) { 128 | const base = now(); 129 | let allowed = 0; 130 | 131 | // Expend capacity for the duration of the window 132 | while (now() < base + capacity.window) { 133 | allowed += +(await limit(key)).allow; 134 | await sleep(1); 135 | } 136 | 137 | // Capacity should be within the bounds 138 | t.assert(allowed >= capacity.min); 139 | t.assert(allowed <= capacity.max); 140 | } 141 | }); 142 | 143 | it('handles basic rate metrics', async (t, config, key, now, sleep) => { 144 | const rate: limiter.Rate = { burst: 9, flow: 1 / 2 }; 145 | const limit = limiter.create({ ...config, rate }); 146 | 147 | // Perform test twice to ensure full drain 148 | for (const {} of repeat(2)) { 149 | const base = now(); 150 | let free = rate.burst - 1; 151 | 152 | // Expect initial burst to be allowed 153 | const time = calcTime(rate, true); 154 | while (now() < base + time) { 155 | t.deepEqual(await limit(key), { allow: true, free, wait: 0 }); 156 | await sleep(1); 157 | free += rate.flow - 1; 158 | } 159 | 160 | // Expect steady-state of flow rate near capacity 161 | const loop = calcLoop(rate); 162 | while (now() < base + time + loop * 4) { 163 | let allowed = 0; 164 | for (const {} of repeat(loop)) { 165 | allowed += +(await limit(key)).allow; 166 | await sleep(1); 167 | } 168 | t.is(allowed, 1); 169 | } 170 | 171 | // Once the flow would return the full burst capacity, 172 | // behavior should be reset to baseline 173 | await sleep(rate.burst / rate.flow); 174 | } 175 | }); 176 | 177 | it('handles multiple rates', async (t, config, key, now, sleep) => { 178 | const slow: limiter.Rate = { burst: 18, flow: 1 / 4 }; 179 | const fast: limiter.Rate = { burst: 9, flow: 1 / 2 }; 180 | const limit = limiter.create({ 181 | ...config, 182 | rate: [slow, fast], 183 | backoff: x => 2 ** x, 184 | }); 185 | const base = now(); 186 | let free = fast.burst - 1; 187 | 188 | // Expect initial burst to be allowed 189 | const timeFast = calcTime(fast, true); 190 | while (now() < base + timeFast) { 191 | t.deepEqual(await limit(key), { allow: true, free, wait: 0 }); 192 | await sleep(1); 193 | free += fast.flow - 1; 194 | } 195 | 196 | // Expect fast flow rate until slow burst is consumed 197 | const loopFast = calcLoop(fast); 198 | const timeSlow = 199 | calcTime({ 200 | burst: calcLeft(slow, timeFast, true), 201 | flow: slow.flow / fast.flow, 202 | }) * loopFast; 203 | while (now() < base + timeFast + timeSlow) { 204 | let allowed = 0; 205 | for (const {} of repeat(loopFast)) { 206 | allowed += +(await limit(key)).allow; 207 | await sleep(1); 208 | } 209 | t.is(allowed, 1); 210 | } 211 | 212 | // Expect slow flow rate afterwards 213 | const loopSlow = calcLoop(slow); 214 | let wait: number | undefined; 215 | while (now() < base + timeFast + timeSlow + loopSlow * 4) { 216 | let allowed = 0; 217 | for (const {} of repeat(loopSlow)) { 218 | const result = await limit(key); 219 | if (result.allow) { 220 | allowed += 1; 221 | } else if (wait) { 222 | t.is(result.wait, 2 * wait); 223 | } 224 | wait = result.wait; 225 | await sleep(1); 226 | } 227 | t.is(allowed, 1); 228 | } 229 | }); 230 | 231 | it('handles subsecond deltas', async (t, config, key, now, sleep) => { 232 | const capacity: limiter.Capacity = { window: 1, min: 4, max: 5 }; 233 | const limit = limiter.create({ ...config, capacity }); 234 | 235 | const base = now(); 236 | let allowed = 0; 237 | 238 | // Expend capacity for the duration of the window 239 | while (now() < base + capacity.window) { 240 | allowed += +(await limit(key)).allow; 241 | await sleep(0.125); 242 | } 243 | 244 | // Capacity should be within the bounds 245 | t.assert(allowed >= capacity.min); 246 | t.assert(allowed <= capacity.max); 247 | }); 248 | 249 | it('discards superfluous rates', async (t, _, key) => { 250 | const config: limiter.Config = { 251 | async eval(script, keys, argv) { 252 | t.deepEqual(keys, [key]); 253 | t.deepEqual(argv, [1, 0.1, 4, 0.2, 2, 0.4, 1]); 254 | return [1, 1, 1]; 255 | }, 256 | }; 257 | 258 | const rate: limiter.Rate[] = [ 259 | { burst: 4, flow: 0.1 }, // 1 - Valid 260 | { burst: 3, flow: 0.2 }, // 2 - Strictly larger than 3 261 | { burst: 2, flow: 0.2 }, // 3 - Valid 262 | { burst: 2, flow: 0.3 }, // 4 - Strictly larger than 3 263 | { burst: 1, flow: 0.4 }, // 5 - Valid 264 | ]; 265 | await limiter.create({ ...config, rate })(key); 266 | }); 267 | 268 | it('passes errors through', async (t, _, key) => { 269 | const error = Error(); 270 | const config: limiter.Config = { 271 | async eval() {}, 272 | async evalsha() { 273 | throw error; 274 | }, 275 | }; 276 | 277 | try { 278 | const rate: limiter.Rate = { burst: 4, flow: 0.1 }; 279 | await limiter.create({ ...config, rate })(key); 280 | t.fail('should throw underlying error'); 281 | } catch (err) { 282 | t.is(err, error); 283 | } 284 | }); 285 | -------------------------------------------------------------------------------- /index.ts: -------------------------------------------------------------------------------- 1 | /*! 2 | * Copyright (c) Microsoft Corporation. 3 | * Licensed under the MIT License. 4 | */ 5 | 6 | // Common validation shorthand 7 | const validate = (message: string, condition: unknown) => { 8 | if (!condition) { 9 | throw RangeError(message); 10 | } 11 | }; 12 | 13 | // Validate capacity metrics and translate them into flow/burst pairs 14 | const validateCapacity = ({ min, max, window }: Capacity): [number, number] => { 15 | validate( 16 | 'All capacity parameters must be greater than zero', 17 | min > 0 && window > 0 18 | ); 19 | validate( 20 | 'Maximum capacity must be greater than minimum capacity', 21 | max > min 22 | ); 23 | 24 | return [min / window, max - min]; 25 | }; 26 | 27 | // Validate rate metrics and translate them into flow/burst pairs 28 | const validateRate = ({ flow, burst }: Rate): [number, number] => { 29 | validate( 30 | 'All rate parameters must be greater than zero', 31 | flow > 0 && burst > 0 32 | ); 33 | 34 | return [flow, burst]; 35 | }; 36 | 37 | // Validate and sort flow/burst pairs and translate them to script parameters 38 | const validateLimits = (...input: number[][]): number[] => { 39 | validate( 40 | 'At least one rate or capacity metric must be specified', 41 | input.length 42 | ); 43 | 44 | // Sort rates by the slowest to fastest flow for consistency, or by burst 45 | // if flow is the same (to make them easier to filter out later) 46 | const limits = input.sort( 47 | ([flow1, burst1], [flow2, burst2]) => flow1 - flow2 || burst1 - burst2 48 | ); 49 | 50 | // Any limit that is strictly larger than another (in both flow and burst) 51 | // is superfluous, as the smaller limit will always be more restrictive 52 | return limits.reduce((params, [flow, burst]) => 53 | burst < params[params.length - 1] ? [...params, flow, burst] : params 54 | ); 55 | }; 56 | 57 | /** 58 | * Create a new rate-limiter test function 59 | * @param config Configuration options 60 | */ 61 | export function create({ 62 | eval: code, 63 | evalsha: hash, 64 | prefix = '', 65 | backoff = x => x, 66 | capacity = [], 67 | rate = [], 68 | }: Config): Test { 69 | // Precalculate parameters not dependent on test arguments 70 | const params = validateLimits( 71 | ...([] as Capacity[]).concat(capacity).map(validateCapacity), 72 | ...([] as Rate[]).concat(rate).map(validateRate) 73 | ); 74 | 75 | // Execute the Lua script on the Redis client to check available capacity 76 | return async (key, cost = 1) => { 77 | // Translate function arguments to Redis arguments 78 | const keys = [prefix + key]; 79 | const argv = [cost, ...params]; 80 | 81 | // Evaluate the script in the Redis cache 82 | const [allow, value, index] = 83 | (await hash?.('{{LUA_HASH}}', keys, argv).catch(err => { 84 | if (!/NOSCRIPT/.test(err)) { 85 | throw err; 86 | } 87 | })) || (await code('{{LUA_CODE}}', keys, argv)); 88 | 89 | // Translate the Redis response into a result object 90 | return { 91 | allow: allow == 1, 92 | free: allow * value, 93 | wait: 94 | 1 - allow && 95 | (cost / params[2 * index - 2]) * backoff(value / cost), 96 | }; 97 | }; 98 | } 99 | 100 | /** Rate-limiter instance configuration options */ 101 | export interface Config { 102 | /** EVAL call to Redis */ 103 | eval(script: string, keys: string[], argv: unknown[]): Promise; 104 | 105 | /** EVALSHA call to Redis */ 106 | evalsha?(hash: string, keys: string[], argv: unknown[]): Promise; 107 | 108 | /** Prefix all keys with this value (default empty) */ 109 | prefix?: string; 110 | 111 | /** Backoff scaling function (default linear) */ 112 | backoff?(denied: number): number; 113 | 114 | /** Capacity metric(s) to limit by */ 115 | capacity?: Capacity | Capacity[]; 116 | 117 | /** Rate metric(s) to limit by */ 118 | rate?: Rate | Rate[]; 119 | } 120 | 121 | /** A rate metric to limit by */ 122 | export interface Rate { 123 | /** The rate at which capacity is restored (per second) */ 124 | flow: number; 125 | 126 | /** The allowed capacity before requests start to be denied */ 127 | burst: number; 128 | } 129 | 130 | /** A capacity metric to limit by */ 131 | export interface Capacity { 132 | /** Time window over which to apply this capacity (in seconds) */ 133 | window: number; 134 | 135 | /** Minimum guaranteed capacity */ 136 | min: number; 137 | 138 | /** Maximum tolerable capacity */ 139 | max: number; 140 | } 141 | 142 | /** Result type for a rate-limited action */ 143 | export interface Result { 144 | /** Whether to allow this action */ 145 | allow: boolean; 146 | 147 | /** Remaining free capacity */ 148 | free: number; 149 | 150 | /** Wait this long before trying again (in seconds) */ 151 | wait: number; 152 | } 153 | 154 | /** 155 | * Rate-limiter test execution function 156 | * @param key The key against which this rate should be tested 157 | * @param cost (default 1) The abstract cost of this operation 158 | */ 159 | export type Test = (key: string, cost?: number) => Promise; 160 | -------------------------------------------------------------------------------- /license: -------------------------------------------------------------------------------- 1 | Copyright (c) Microsoft Corporation. 2 | 3 | MIT License 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. -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "redis-bucket", 3 | "version": "2.0.0", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "redis-bucket", 9 | "version": "2.0.0", 10 | "license": "MIT", 11 | "devDependencies": { 12 | "@princjef/tslint-config": "^2.0.2", 13 | "@types/node": "^18.11.11", 14 | "ava": "^5.1.0", 15 | "c8": "^7.12.0", 16 | "prettier": "^2.8.0", 17 | "redis": "^4.5.1", 18 | "terser": "^5.16.1", 19 | "tslint": "^5.20.1", 20 | "typescript": "^4.9.3" 21 | } 22 | }, 23 | "node_modules/@babel/code-frame": { 24 | "version": "7.18.6", 25 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.18.6.tgz", 26 | "integrity": "sha512-TDCmlK5eOvH+eH7cdAFlNXeVJqWIQ7gW9tY1GJIpUtFb6CmjVyq2VM3u71bOyR8CRihcCgMUYoDNyLXao3+70Q==", 27 | "dev": true, 28 | "dependencies": { 29 | "@babel/highlight": "^7.18.6" 30 | }, 31 | "engines": { 32 | "node": ">=6.9.0" 33 | } 34 | }, 35 | "node_modules/@babel/helper-validator-identifier": { 36 | "version": "7.19.1", 37 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.19.1.tgz", 38 | "integrity": "sha512-awrNfaMtnHUr653GgGEs++LlAvW6w+DcPrOliSMXWCKo597CwL5Acf/wWdNkf/tfEQE3mjkeD1YOVZOUV/od1w==", 39 | "dev": true, 40 | "engines": { 41 | "node": ">=6.9.0" 42 | } 43 | }, 44 | "node_modules/@babel/highlight": { 45 | "version": "7.18.6", 46 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.18.6.tgz", 47 | "integrity": "sha512-u7stbOuYjaPezCuLj29hNW1v64M2Md2qupEKP1fHc7WdOA3DgLh37suiSrZYY7haUB7iBeQZ9P1uiRF359do3g==", 48 | "dev": true, 49 | "dependencies": { 50 | "@babel/helper-validator-identifier": "^7.18.6", 51 | "chalk": "^2.0.0", 52 | "js-tokens": "^4.0.0" 53 | }, 54 | "engines": { 55 | "node": ">=6.9.0" 56 | } 57 | }, 58 | "node_modules/@babel/highlight/node_modules/ansi-styles": { 59 | "version": "3.2.1", 60 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 61 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 62 | "dev": true, 63 | "dependencies": { 64 | "color-convert": "^1.9.0" 65 | }, 66 | "engines": { 67 | "node": ">=4" 68 | } 69 | }, 70 | "node_modules/@babel/highlight/node_modules/chalk": { 71 | "version": "2.4.2", 72 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 73 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 74 | "dev": true, 75 | "dependencies": { 76 | "ansi-styles": "^3.2.1", 77 | "escape-string-regexp": "^1.0.5", 78 | "supports-color": "^5.3.0" 79 | }, 80 | "engines": { 81 | "node": ">=4" 82 | } 83 | }, 84 | "node_modules/@babel/highlight/node_modules/escape-string-regexp": { 85 | "version": "1.0.5", 86 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 87 | "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", 88 | "dev": true, 89 | "engines": { 90 | "node": ">=0.8.0" 91 | } 92 | }, 93 | "node_modules/@babel/highlight/node_modules/has-flag": { 94 | "version": "3.0.0", 95 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 96 | "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", 97 | "dev": true, 98 | "engines": { 99 | "node": ">=4" 100 | } 101 | }, 102 | "node_modules/@babel/highlight/node_modules/supports-color": { 103 | "version": "5.5.0", 104 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 105 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 106 | "dev": true, 107 | "dependencies": { 108 | "has-flag": "^3.0.0" 109 | }, 110 | "engines": { 111 | "node": ">=4" 112 | } 113 | }, 114 | "node_modules/@bcoe/v8-coverage": { 115 | "version": "0.2.3", 116 | "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz", 117 | "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==", 118 | "dev": true 119 | }, 120 | "node_modules/@fimbul/bifrost": { 121 | "version": "0.21.0", 122 | "resolved": "https://registry.npmjs.org/@fimbul/bifrost/-/bifrost-0.21.0.tgz", 123 | "integrity": "sha512-ou8VU+nTmOW1jeg+FT+sn+an/M0Xb9G16RucrfhjXGWv1Q97kCoM5CG9Qj7GYOSdu7km72k7nY83Eyr53Bkakg==", 124 | "dev": true, 125 | "dependencies": { 126 | "@fimbul/ymir": "^0.21.0", 127 | "get-caller-file": "^2.0.0", 128 | "tslib": "^1.8.1", 129 | "tsutils": "^3.5.0" 130 | }, 131 | "peerDependencies": { 132 | "tslint": "^5.0.0", 133 | "typescript": ">= 3.3.0 || >= 3.6.0-dev || >= 3.7.0-dev" 134 | } 135 | }, 136 | "node_modules/@fimbul/bifrost/node_modules/tsutils": { 137 | "version": "3.21.0", 138 | "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.21.0.tgz", 139 | "integrity": "sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==", 140 | "dev": true, 141 | "dependencies": { 142 | "tslib": "^1.8.1" 143 | }, 144 | "engines": { 145 | "node": ">= 6" 146 | }, 147 | "peerDependencies": { 148 | "typescript": ">=2.8.0 || >= 3.2.0-dev || >= 3.3.0-dev || >= 3.4.0-dev || >= 3.5.0-dev || >= 3.6.0-dev || >= 3.6.0-beta || >= 3.7.0-dev || >= 3.7.0-beta" 149 | } 150 | }, 151 | "node_modules/@fimbul/ymir": { 152 | "version": "0.21.0", 153 | "resolved": "https://registry.npmjs.org/@fimbul/ymir/-/ymir-0.21.0.tgz", 154 | "integrity": "sha512-T/y7WqPsm4n3zhT08EpB5sfdm2Kvw3gurAxr2Lr5dQeLi8ZsMlNT/Jby+ZmuuAAd1PnXYzKp+2SXgIkQIIMCUg==", 155 | "dev": true, 156 | "dependencies": { 157 | "inversify": "^5.0.0", 158 | "reflect-metadata": "^0.1.12", 159 | "tslib": "^1.8.1" 160 | }, 161 | "peerDependencies": { 162 | "tsutils": ">=2.29.0", 163 | "typescript": ">= 3.3.0 || >= 3.6.0-dev || >= 3.7.0-dev" 164 | } 165 | }, 166 | "node_modules/@istanbuljs/schema": { 167 | "version": "0.1.3", 168 | "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", 169 | "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", 170 | "dev": true, 171 | "engines": { 172 | "node": ">=8" 173 | } 174 | }, 175 | "node_modules/@jridgewell/gen-mapping": { 176 | "version": "0.3.2", 177 | "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.2.tgz", 178 | "integrity": "sha512-mh65xKQAzI6iBcFzwv28KVWSmCkdRBWoOh+bYQGW3+6OZvbbN3TqMGo5hqYxQniRcH9F2VZIoJCm4pa3BPDK/A==", 179 | "dev": true, 180 | "dependencies": { 181 | "@jridgewell/set-array": "^1.0.1", 182 | "@jridgewell/sourcemap-codec": "^1.4.10", 183 | "@jridgewell/trace-mapping": "^0.3.9" 184 | }, 185 | "engines": { 186 | "node": ">=6.0.0" 187 | } 188 | }, 189 | "node_modules/@jridgewell/resolve-uri": { 190 | "version": "3.1.0", 191 | "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.0.tgz", 192 | "integrity": "sha512-F2msla3tad+Mfht5cJq7LSXcdudKTWCVYUgw6pLFOOHSTtZlj6SWNYAp+AhuqLmWdBO2X5hPrLcu8cVP8fy28w==", 193 | "dev": true, 194 | "engines": { 195 | "node": ">=6.0.0" 196 | } 197 | }, 198 | "node_modules/@jridgewell/set-array": { 199 | "version": "1.1.2", 200 | "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.1.2.tgz", 201 | "integrity": "sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw==", 202 | "dev": true, 203 | "engines": { 204 | "node": ">=6.0.0" 205 | } 206 | }, 207 | "node_modules/@jridgewell/source-map": { 208 | "version": "0.3.2", 209 | "resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.2.tgz", 210 | "integrity": "sha512-m7O9o2uR8k2ObDysZYzdfhb08VuEml5oWGiosa1VdaPZ/A6QyPkAJuwN0Q1lhULOf6B7MtQmHENS743hWtCrgw==", 211 | "dev": true, 212 | "dependencies": { 213 | "@jridgewell/gen-mapping": "^0.3.0", 214 | "@jridgewell/trace-mapping": "^0.3.9" 215 | } 216 | }, 217 | "node_modules/@jridgewell/sourcemap-codec": { 218 | "version": "1.4.14", 219 | "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.14.tgz", 220 | "integrity": "sha512-XPSJHWmi394fuUuzDnGz1wiKqWfo1yXecHQMRf2l6hztTO+nPru658AyDngaBe7isIxEkRsPR3FZh+s7iVa4Uw==", 221 | "dev": true 222 | }, 223 | "node_modules/@jridgewell/trace-mapping": { 224 | "version": "0.3.17", 225 | "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.17.tgz", 226 | "integrity": "sha512-MCNzAp77qzKca9+W/+I0+sEpaUnZoeasnghNeVc41VZCEKaCH73Vq3BZZ/SzWIgrqE4H4ceI+p+b6C0mHf9T4g==", 227 | "dev": true, 228 | "dependencies": { 229 | "@jridgewell/resolve-uri": "3.1.0", 230 | "@jridgewell/sourcemap-codec": "1.4.14" 231 | } 232 | }, 233 | "node_modules/@nodelib/fs.scandir": { 234 | "version": "2.1.5", 235 | "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", 236 | "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", 237 | "dev": true, 238 | "dependencies": { 239 | "@nodelib/fs.stat": "2.0.5", 240 | "run-parallel": "^1.1.9" 241 | }, 242 | "engines": { 243 | "node": ">= 8" 244 | } 245 | }, 246 | "node_modules/@nodelib/fs.stat": { 247 | "version": "2.0.5", 248 | "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", 249 | "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", 250 | "dev": true, 251 | "engines": { 252 | "node": ">= 8" 253 | } 254 | }, 255 | "node_modules/@nodelib/fs.walk": { 256 | "version": "1.2.8", 257 | "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", 258 | "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", 259 | "dev": true, 260 | "dependencies": { 261 | "@nodelib/fs.scandir": "2.1.5", 262 | "fastq": "^1.6.0" 263 | }, 264 | "engines": { 265 | "node": ">= 8" 266 | } 267 | }, 268 | "node_modules/@princjef/tslint-config": { 269 | "version": "2.0.2", 270 | "resolved": "https://registry.npmjs.org/@princjef/tslint-config/-/tslint-config-2.0.2.tgz", 271 | "integrity": "sha512-ALOnGQODFJstrq2Qgd84SdGG6s6QkZkd/1J3thOcSGsFlYXgfdp7C6xrnCsLgoLmXkVECBkhXTvkwcoKX3UQhg==", 272 | "dev": true, 273 | "dependencies": { 274 | "tslint-config-airbnb": "^5.11.2", 275 | "tslint-config-prettier": "^1.18.0", 276 | "tslint-consistent-codestyle": "^1.16.0", 277 | "tslint-microsoft-contrib": "^6.2.0" 278 | }, 279 | "engines": { 280 | "node": ">8.0.0" 281 | }, 282 | "peerDependencies": { 283 | "tslint": ">=5.0.0" 284 | } 285 | }, 286 | "node_modules/@princjef/tslint-config/node_modules/tslint-microsoft-contrib": { 287 | "version": "6.2.0", 288 | "resolved": "https://registry.npmjs.org/tslint-microsoft-contrib/-/tslint-microsoft-contrib-6.2.0.tgz", 289 | "integrity": "sha512-6tfi/2tHqV/3CL77pULBcK+foty11Rr0idRDxKnteTaKm6gWF9qmaCNU17HVssOuwlYNyOmd9Jsmjd+1t3a3qw==", 290 | "dev": true, 291 | "dependencies": { 292 | "tsutils": "^2.27.2 <2.29.0" 293 | }, 294 | "peerDependencies": { 295 | "tslint": "^5.1.0", 296 | "typescript": "^2.1.0 || ^3.0.0" 297 | } 298 | }, 299 | "node_modules/@princjef/tslint-config/node_modules/tsutils": { 300 | "version": "2.28.0", 301 | "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.28.0.tgz", 302 | "integrity": "sha512-bh5nAtW0tuhvOJnx1GLRn5ScraRLICGyJV5wJhtRWOLsxW70Kk5tZtpK3O/hW6LDnqKS9mlUMPZj9fEMJ0gxqA==", 303 | "dev": true, 304 | "dependencies": { 305 | "tslib": "^1.8.1" 306 | }, 307 | "peerDependencies": { 308 | "typescript": ">=2.1.0 || >=2.1.0-dev || >=2.2.0-dev || >=2.3.0-dev || >=2.4.0-dev || >=2.5.0-dev || >=2.6.0-dev || >=2.7.0-dev || >=2.8.0-dev || >=2.9.0-dev || >= 3.0.0-dev || >= 3.1.0-dev" 309 | } 310 | }, 311 | "node_modules/@princjef/tslint-config/node_modules/typescript": { 312 | "version": "3.9.10", 313 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.10.tgz", 314 | "integrity": "sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q==", 315 | "dev": true, 316 | "peer": true, 317 | "bin": { 318 | "tsc": "bin/tsc", 319 | "tsserver": "bin/tsserver" 320 | }, 321 | "engines": { 322 | "node": ">=4.2.0" 323 | } 324 | }, 325 | "node_modules/@redis/bloom": { 326 | "version": "1.1.0", 327 | "resolved": "https://registry.npmjs.org/@redis/bloom/-/bloom-1.1.0.tgz", 328 | "integrity": "sha512-9QovlxmpRtvxVbN0UBcv8WfdSMudNZZTFqCsnBszcQXqaZb/TVe30ScgGEO7u1EAIacTPAo7/oCYjYAxiHLanQ==", 329 | "dev": true, 330 | "peerDependencies": { 331 | "@redis/client": "^1.0.0" 332 | } 333 | }, 334 | "node_modules/@redis/client": { 335 | "version": "1.4.2", 336 | "resolved": "https://registry.npmjs.org/@redis/client/-/client-1.4.2.tgz", 337 | "integrity": "sha512-oUdEjE0I7JS5AyaAjkD3aOXn9NhO7XKyPyXEyrgFDu++VrVBHUPnV6dgEya9TcMuj5nIJRuCzCm8ZP+c9zCHPw==", 338 | "dev": true, 339 | "dependencies": { 340 | "cluster-key-slot": "1.1.1", 341 | "generic-pool": "3.9.0", 342 | "yallist": "4.0.0" 343 | }, 344 | "engines": { 345 | "node": ">=14" 346 | } 347 | }, 348 | "node_modules/@redis/graph": { 349 | "version": "1.1.0", 350 | "resolved": "https://registry.npmjs.org/@redis/graph/-/graph-1.1.0.tgz", 351 | "integrity": "sha512-16yZWngxyXPd+MJxeSr0dqh2AIOi8j9yXKcKCwVaKDbH3HTuETpDVPcLujhFYVPtYrngSco31BUcSa9TH31Gqg==", 352 | "dev": true, 353 | "peerDependencies": { 354 | "@redis/client": "^1.0.0" 355 | } 356 | }, 357 | "node_modules/@redis/json": { 358 | "version": "1.0.4", 359 | "resolved": "https://registry.npmjs.org/@redis/json/-/json-1.0.4.tgz", 360 | "integrity": "sha512-LUZE2Gdrhg0Rx7AN+cZkb1e6HjoSKaeeW8rYnt89Tly13GBI5eP4CwDVr+MY8BAYfCg4/N15OUrtLoona9uSgw==", 361 | "dev": true, 362 | "peerDependencies": { 363 | "@redis/client": "^1.0.0" 364 | } 365 | }, 366 | "node_modules/@redis/search": { 367 | "version": "1.1.0", 368 | "resolved": "https://registry.npmjs.org/@redis/search/-/search-1.1.0.tgz", 369 | "integrity": "sha512-NyFZEVnxIJEybpy+YskjgOJRNsfTYqaPbK/Buv6W2kmFNaRk85JiqjJZA5QkRmWvGbyQYwoO5QfDi2wHskKrQQ==", 370 | "dev": true, 371 | "peerDependencies": { 372 | "@redis/client": "^1.0.0" 373 | } 374 | }, 375 | "node_modules/@redis/time-series": { 376 | "version": "1.0.4", 377 | "resolved": "https://registry.npmjs.org/@redis/time-series/-/time-series-1.0.4.tgz", 378 | "integrity": "sha512-ThUIgo2U/g7cCuZavucQTQzA9g9JbDDY2f64u3AbAoz/8vE2lt2U37LamDUVChhaDA3IRT9R6VvJwqnUfTJzng==", 379 | "dev": true, 380 | "peerDependencies": { 381 | "@redis/client": "^1.0.0" 382 | } 383 | }, 384 | "node_modules/@types/istanbul-lib-coverage": { 385 | "version": "2.0.4", 386 | "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.4.tgz", 387 | "integrity": "sha512-z/QT1XN4K4KYuslS23k62yDIDLwLFkzxOuMplDtObz0+y7VqJCaO2o+SPwHCvLFZh7xazvvoor2tA/hPz9ee7g==", 388 | "dev": true 389 | }, 390 | "node_modules/@types/node": { 391 | "version": "18.11.11", 392 | "resolved": "https://registry.npmjs.org/@types/node/-/node-18.11.11.tgz", 393 | "integrity": "sha512-KJ021B1nlQUBLopzZmPBVuGU9un7WJd/W4ya7Ih02B4Uwky5Nja0yGYav2EfYIk0RR2Q9oVhf60S2XR1BCWJ2g==", 394 | "dev": true 395 | }, 396 | "node_modules/acorn": { 397 | "version": "8.8.1", 398 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.8.1.tgz", 399 | "integrity": "sha512-7zFpHzhnqYKrkYdUjF1HI1bzd0VygEGX8lFk4k5zVMqHEoES+P+7TKI+EvLO9WVMJ8eekdO0aDEK044xTXwPPA==", 400 | "dev": true, 401 | "bin": { 402 | "acorn": "bin/acorn" 403 | }, 404 | "engines": { 405 | "node": ">=0.4.0" 406 | } 407 | }, 408 | "node_modules/acorn-walk": { 409 | "version": "8.2.0", 410 | "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.2.0.tgz", 411 | "integrity": "sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==", 412 | "dev": true, 413 | "engines": { 414 | "node": ">=0.4.0" 415 | } 416 | }, 417 | "node_modules/aggregate-error": { 418 | "version": "4.0.1", 419 | "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-4.0.1.tgz", 420 | "integrity": "sha512-0poP0T7el6Vq3rstR8Mn4V/IQrpBLO6POkUSrN7RhyY+GF/InCFShQzsQ39T25gkHhLgSLByyAz+Kjb+c2L98w==", 421 | "dev": true, 422 | "dependencies": { 423 | "clean-stack": "^4.0.0", 424 | "indent-string": "^5.0.0" 425 | }, 426 | "engines": { 427 | "node": ">=12" 428 | }, 429 | "funding": { 430 | "url": "https://github.com/sponsors/sindresorhus" 431 | } 432 | }, 433 | "node_modules/ansi-regex": { 434 | "version": "6.0.1", 435 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.0.1.tgz", 436 | "integrity": "sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==", 437 | "dev": true, 438 | "engines": { 439 | "node": ">=12" 440 | }, 441 | "funding": { 442 | "url": "https://github.com/chalk/ansi-regex?sponsor=1" 443 | } 444 | }, 445 | "node_modules/ansi-styles": { 446 | "version": "6.2.1", 447 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.1.tgz", 448 | "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==", 449 | "dev": true, 450 | "engines": { 451 | "node": ">=12" 452 | }, 453 | "funding": { 454 | "url": "https://github.com/chalk/ansi-styles?sponsor=1" 455 | } 456 | }, 457 | "node_modules/anymatch": { 458 | "version": "3.1.3", 459 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", 460 | "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", 461 | "dev": true, 462 | "dependencies": { 463 | "normalize-path": "^3.0.0", 464 | "picomatch": "^2.0.4" 465 | }, 466 | "engines": { 467 | "node": ">= 8" 468 | } 469 | }, 470 | "node_modules/argparse": { 471 | "version": "1.0.10", 472 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 473 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 474 | "dev": true, 475 | "dependencies": { 476 | "sprintf-js": "~1.0.2" 477 | } 478 | }, 479 | "node_modules/array-find-index": { 480 | "version": "1.0.2", 481 | "resolved": "https://registry.npmjs.org/array-find-index/-/array-find-index-1.0.2.tgz", 482 | "integrity": "sha512-M1HQyIXcBGtVywBt8WVdim+lrNaK7VHp99Qt5pSNziXznKHViIBbXWtfRTpEFpF/c4FdfxNAsCCwPp5phBYJtw==", 483 | "dev": true, 484 | "engines": { 485 | "node": ">=0.10.0" 486 | } 487 | }, 488 | "node_modules/arrgv": { 489 | "version": "1.0.2", 490 | "resolved": "https://registry.npmjs.org/arrgv/-/arrgv-1.0.2.tgz", 491 | "integrity": "sha512-a4eg4yhp7mmruZDQFqVMlxNRFGi/i1r87pt8SDHy0/I8PqSXoUTlWZRdAZo0VXgvEARcujbtTk8kiZRi1uDGRw==", 492 | "dev": true, 493 | "engines": { 494 | "node": ">=8.0.0" 495 | } 496 | }, 497 | "node_modules/arrify": { 498 | "version": "3.0.0", 499 | "resolved": "https://registry.npmjs.org/arrify/-/arrify-3.0.0.tgz", 500 | "integrity": "sha512-tLkvA81vQG/XqE2mjDkGQHoOINtMHtysSnemrmoGe6PydDPMRbVugqyk4A6V/WDWEfm3l+0d8anA9r8cv/5Jaw==", 501 | "dev": true, 502 | "engines": { 503 | "node": ">=12" 504 | }, 505 | "funding": { 506 | "url": "https://github.com/sponsors/sindresorhus" 507 | } 508 | }, 509 | "node_modules/ava": { 510 | "version": "5.1.0", 511 | "resolved": "https://registry.npmjs.org/ava/-/ava-5.1.0.tgz", 512 | "integrity": "sha512-e5VFrSQ0WBPyZJWRXVrO7RFOizFeNM0t2PORwrPvWtApgkORI6cvGnY3GX1G+lzpd0HjqNx5Jus22AhxVnUMNA==", 513 | "dev": true, 514 | "dependencies": { 515 | "acorn": "^8.8.1", 516 | "acorn-walk": "^8.2.0", 517 | "ansi-styles": "^6.2.1", 518 | "arrgv": "^1.0.2", 519 | "arrify": "^3.0.0", 520 | "callsites": "^4.0.0", 521 | "cbor": "^8.1.0", 522 | "chalk": "^5.1.2", 523 | "chokidar": "^3.5.3", 524 | "chunkd": "^2.0.1", 525 | "ci-info": "^3.6.1", 526 | "ci-parallel-vars": "^1.0.1", 527 | "clean-yaml-object": "^0.1.0", 528 | "cli-truncate": "^3.1.0", 529 | "code-excerpt": "^4.0.0", 530 | "common-path-prefix": "^3.0.0", 531 | "concordance": "^5.0.4", 532 | "currently-unhandled": "^0.4.1", 533 | "debug": "^4.3.4", 534 | "del": "^7.0.0", 535 | "emittery": "^1.0.1", 536 | "figures": "^5.0.0", 537 | "globby": "^13.1.2", 538 | "ignore-by-default": "^2.1.0", 539 | "indent-string": "^5.0.0", 540 | "is-error": "^2.2.2", 541 | "is-plain-object": "^5.0.0", 542 | "is-promise": "^4.0.0", 543 | "matcher": "^5.0.0", 544 | "mem": "^9.0.2", 545 | "ms": "^2.1.3", 546 | "p-event": "^5.0.1", 547 | "p-map": "^5.5.0", 548 | "picomatch": "^2.3.1", 549 | "pkg-conf": "^4.0.0", 550 | "plur": "^5.1.0", 551 | "pretty-ms": "^8.0.0", 552 | "resolve-cwd": "^3.0.0", 553 | "slash": "^3.0.0", 554 | "stack-utils": "^2.0.6", 555 | "strip-ansi": "^7.0.1", 556 | "supertap": "^3.0.1", 557 | "temp-dir": "^3.0.0", 558 | "write-file-atomic": "^5.0.0", 559 | "yargs": "^17.6.2" 560 | }, 561 | "bin": { 562 | "ava": "entrypoints/cli.mjs" 563 | }, 564 | "engines": { 565 | "node": ">=14.19 <15 || >=16.15 <17 || >=18" 566 | }, 567 | "peerDependencies": { 568 | "@ava/typescript": "*" 569 | }, 570 | "peerDependenciesMeta": { 571 | "@ava/typescript": { 572 | "optional": true 573 | } 574 | } 575 | }, 576 | "node_modules/balanced-match": { 577 | "version": "1.0.2", 578 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 579 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 580 | "dev": true 581 | }, 582 | "node_modules/binary-extensions": { 583 | "version": "2.2.0", 584 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 585 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 586 | "dev": true, 587 | "engines": { 588 | "node": ">=8" 589 | } 590 | }, 591 | "node_modules/blueimp-md5": { 592 | "version": "2.19.0", 593 | "resolved": "https://registry.npmjs.org/blueimp-md5/-/blueimp-md5-2.19.0.tgz", 594 | "integrity": "sha512-DRQrD6gJyy8FbiE4s+bDoXS9hiW3Vbx5uCdwvcCf3zLHL+Iv7LtGHLpr+GZV8rHG8tK766FGYBwRbu8pELTt+w==", 595 | "dev": true 596 | }, 597 | "node_modules/brace-expansion": { 598 | "version": "1.1.11", 599 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 600 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 601 | "dev": true, 602 | "dependencies": { 603 | "balanced-match": "^1.0.0", 604 | "concat-map": "0.0.1" 605 | } 606 | }, 607 | "node_modules/braces": { 608 | "version": "3.0.2", 609 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 610 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 611 | "dev": true, 612 | "dependencies": { 613 | "fill-range": "^7.0.1" 614 | }, 615 | "engines": { 616 | "node": ">=8" 617 | } 618 | }, 619 | "node_modules/buffer-from": { 620 | "version": "1.1.2", 621 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", 622 | "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", 623 | "dev": true 624 | }, 625 | "node_modules/builtin-modules": { 626 | "version": "1.1.1", 627 | "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", 628 | "integrity": "sha512-wxXCdllwGhI2kCC0MnvTGYTMvnVZTvqgypkiTI8Pa5tcz2i6VqsqwYGgqwXji+4RgCzms6EajE4IxiUH6HH8nQ==", 629 | "dev": true, 630 | "engines": { 631 | "node": ">=0.10.0" 632 | } 633 | }, 634 | "node_modules/c8": { 635 | "version": "7.12.0", 636 | "resolved": "https://registry.npmjs.org/c8/-/c8-7.12.0.tgz", 637 | "integrity": "sha512-CtgQrHOkyxr5koX1wEUmN/5cfDa2ckbHRA4Gy5LAL0zaCFtVWJS5++n+w4/sr2GWGerBxgTjpKeDclk/Qk6W/A==", 638 | "dev": true, 639 | "dependencies": { 640 | "@bcoe/v8-coverage": "^0.2.3", 641 | "@istanbuljs/schema": "^0.1.3", 642 | "find-up": "^5.0.0", 643 | "foreground-child": "^2.0.0", 644 | "istanbul-lib-coverage": "^3.2.0", 645 | "istanbul-lib-report": "^3.0.0", 646 | "istanbul-reports": "^3.1.4", 647 | "rimraf": "^3.0.2", 648 | "test-exclude": "^6.0.0", 649 | "v8-to-istanbul": "^9.0.0", 650 | "yargs": "^16.2.0", 651 | "yargs-parser": "^20.2.9" 652 | }, 653 | "bin": { 654 | "c8": "bin/c8.js" 655 | }, 656 | "engines": { 657 | "node": ">=10.12.0" 658 | } 659 | }, 660 | "node_modules/c8/node_modules/ansi-regex": { 661 | "version": "5.0.1", 662 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 663 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 664 | "dev": true, 665 | "engines": { 666 | "node": ">=8" 667 | } 668 | }, 669 | "node_modules/c8/node_modules/cliui": { 670 | "version": "7.0.4", 671 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", 672 | "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", 673 | "dev": true, 674 | "dependencies": { 675 | "string-width": "^4.2.0", 676 | "strip-ansi": "^6.0.0", 677 | "wrap-ansi": "^7.0.0" 678 | } 679 | }, 680 | "node_modules/c8/node_modules/emoji-regex": { 681 | "version": "8.0.0", 682 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 683 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 684 | "dev": true 685 | }, 686 | "node_modules/c8/node_modules/is-fullwidth-code-point": { 687 | "version": "3.0.0", 688 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 689 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 690 | "dev": true, 691 | "engines": { 692 | "node": ">=8" 693 | } 694 | }, 695 | "node_modules/c8/node_modules/string-width": { 696 | "version": "4.2.3", 697 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 698 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 699 | "dev": true, 700 | "dependencies": { 701 | "emoji-regex": "^8.0.0", 702 | "is-fullwidth-code-point": "^3.0.0", 703 | "strip-ansi": "^6.0.1" 704 | }, 705 | "engines": { 706 | "node": ">=8" 707 | } 708 | }, 709 | "node_modules/c8/node_modules/strip-ansi": { 710 | "version": "6.0.1", 711 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 712 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 713 | "dev": true, 714 | "dependencies": { 715 | "ansi-regex": "^5.0.1" 716 | }, 717 | "engines": { 718 | "node": ">=8" 719 | } 720 | }, 721 | "node_modules/c8/node_modules/yargs": { 722 | "version": "16.2.0", 723 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", 724 | "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", 725 | "dev": true, 726 | "dependencies": { 727 | "cliui": "^7.0.2", 728 | "escalade": "^3.1.1", 729 | "get-caller-file": "^2.0.5", 730 | "require-directory": "^2.1.1", 731 | "string-width": "^4.2.0", 732 | "y18n": "^5.0.5", 733 | "yargs-parser": "^20.2.2" 734 | }, 735 | "engines": { 736 | "node": ">=10" 737 | } 738 | }, 739 | "node_modules/callsites": { 740 | "version": "4.0.0", 741 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-4.0.0.tgz", 742 | "integrity": "sha512-y3jRROutgpKdz5vzEhWM34TidDU8vkJppF8dszITeb1PQmSqV3DTxyV8G/lyO/DNvtE1YTedehmw9MPZsCBHxQ==", 743 | "dev": true, 744 | "engines": { 745 | "node": ">=12.20" 746 | }, 747 | "funding": { 748 | "url": "https://github.com/sponsors/sindresorhus" 749 | } 750 | }, 751 | "node_modules/cbor": { 752 | "version": "8.1.0", 753 | "resolved": "https://registry.npmjs.org/cbor/-/cbor-8.1.0.tgz", 754 | "integrity": "sha512-DwGjNW9omn6EwP70aXsn7FQJx5kO12tX0bZkaTjzdVFM6/7nhA4t0EENocKGx6D2Bch9PE2KzCUf5SceBdeijg==", 755 | "dev": true, 756 | "dependencies": { 757 | "nofilter": "^3.1.0" 758 | }, 759 | "engines": { 760 | "node": ">=12.19" 761 | } 762 | }, 763 | "node_modules/chalk": { 764 | "version": "5.1.2", 765 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.1.2.tgz", 766 | "integrity": "sha512-E5CkT4jWURs1Vy5qGJye+XwCkNj7Od3Af7CP6SujMetSMkLs8Do2RWJK5yx1wamHV/op8Rz+9rltjaTQWDnEFQ==", 767 | "dev": true, 768 | "engines": { 769 | "node": "^12.17.0 || ^14.13 || >=16.0.0" 770 | }, 771 | "funding": { 772 | "url": "https://github.com/chalk/chalk?sponsor=1" 773 | } 774 | }, 775 | "node_modules/chokidar": { 776 | "version": "3.5.3", 777 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", 778 | "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", 779 | "dev": true, 780 | "funding": [ 781 | { 782 | "type": "individual", 783 | "url": "https://paulmillr.com/funding/" 784 | } 785 | ], 786 | "dependencies": { 787 | "anymatch": "~3.1.2", 788 | "braces": "~3.0.2", 789 | "glob-parent": "~5.1.2", 790 | "is-binary-path": "~2.1.0", 791 | "is-glob": "~4.0.1", 792 | "normalize-path": "~3.0.0", 793 | "readdirp": "~3.6.0" 794 | }, 795 | "engines": { 796 | "node": ">= 8.10.0" 797 | }, 798 | "optionalDependencies": { 799 | "fsevents": "~2.3.2" 800 | } 801 | }, 802 | "node_modules/chunkd": { 803 | "version": "2.0.1", 804 | "resolved": "https://registry.npmjs.org/chunkd/-/chunkd-2.0.1.tgz", 805 | "integrity": "sha512-7d58XsFmOq0j6el67Ug9mHf9ELUXsQXYJBkyxhH/k+6Ke0qXRnv0kbemx+Twc6fRJ07C49lcbdgm9FL1Ei/6SQ==", 806 | "dev": true 807 | }, 808 | "node_modules/ci-info": { 809 | "version": "3.7.0", 810 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.7.0.tgz", 811 | "integrity": "sha512-2CpRNYmImPx+RXKLq6jko/L07phmS9I02TyqkcNU20GCF/GgaWvc58hPtjxDX8lPpkdwc9sNh72V9k00S7ezog==", 812 | "dev": true, 813 | "engines": { 814 | "node": ">=8" 815 | } 816 | }, 817 | "node_modules/ci-parallel-vars": { 818 | "version": "1.0.1", 819 | "resolved": "https://registry.npmjs.org/ci-parallel-vars/-/ci-parallel-vars-1.0.1.tgz", 820 | "integrity": "sha512-uvzpYrpmidaoxvIQHM+rKSrigjOe9feHYbw4uOI2gdfe1C3xIlxO+kVXq83WQWNniTf8bAxVpy+cQeFQsMERKg==", 821 | "dev": true 822 | }, 823 | "node_modules/clean-stack": { 824 | "version": "4.2.0", 825 | "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-4.2.0.tgz", 826 | "integrity": "sha512-LYv6XPxoyODi36Dp976riBtSY27VmFo+MKqEU9QCCWyTrdEPDog+RWA7xQWHi6Vbp61j5c4cdzzX1NidnwtUWg==", 827 | "dev": true, 828 | "dependencies": { 829 | "escape-string-regexp": "5.0.0" 830 | }, 831 | "engines": { 832 | "node": ">=12" 833 | }, 834 | "funding": { 835 | "url": "https://github.com/sponsors/sindresorhus" 836 | } 837 | }, 838 | "node_modules/clean-yaml-object": { 839 | "version": "0.1.0", 840 | "resolved": "https://registry.npmjs.org/clean-yaml-object/-/clean-yaml-object-0.1.0.tgz", 841 | "integrity": "sha512-3yONmlN9CSAkzNwnRCiJQ7Q2xK5mWuEfL3PuTZcAUzhObbXsfsnMptJzXwz93nc5zn9V9TwCVMmV7w4xsm43dw==", 842 | "dev": true, 843 | "engines": { 844 | "node": ">=0.10.0" 845 | } 846 | }, 847 | "node_modules/cli-truncate": { 848 | "version": "3.1.0", 849 | "resolved": "https://registry.npmjs.org/cli-truncate/-/cli-truncate-3.1.0.tgz", 850 | "integrity": "sha512-wfOBkjXteqSnI59oPcJkcPl/ZmwvMMOj340qUIY1SKZCv0B9Cf4D4fAucRkIKQmsIuYK3x1rrgU7MeGRruiuiA==", 851 | "dev": true, 852 | "dependencies": { 853 | "slice-ansi": "^5.0.0", 854 | "string-width": "^5.0.0" 855 | }, 856 | "engines": { 857 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 858 | }, 859 | "funding": { 860 | "url": "https://github.com/sponsors/sindresorhus" 861 | } 862 | }, 863 | "node_modules/cliui": { 864 | "version": "8.0.1", 865 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", 866 | "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", 867 | "dev": true, 868 | "dependencies": { 869 | "string-width": "^4.2.0", 870 | "strip-ansi": "^6.0.1", 871 | "wrap-ansi": "^7.0.0" 872 | }, 873 | "engines": { 874 | "node": ">=12" 875 | } 876 | }, 877 | "node_modules/cliui/node_modules/ansi-regex": { 878 | "version": "5.0.1", 879 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 880 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 881 | "dev": true, 882 | "engines": { 883 | "node": ">=8" 884 | } 885 | }, 886 | "node_modules/cliui/node_modules/emoji-regex": { 887 | "version": "8.0.0", 888 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 889 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 890 | "dev": true 891 | }, 892 | "node_modules/cliui/node_modules/is-fullwidth-code-point": { 893 | "version": "3.0.0", 894 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 895 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 896 | "dev": true, 897 | "engines": { 898 | "node": ">=8" 899 | } 900 | }, 901 | "node_modules/cliui/node_modules/string-width": { 902 | "version": "4.2.3", 903 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 904 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 905 | "dev": true, 906 | "dependencies": { 907 | "emoji-regex": "^8.0.0", 908 | "is-fullwidth-code-point": "^3.0.0", 909 | "strip-ansi": "^6.0.1" 910 | }, 911 | "engines": { 912 | "node": ">=8" 913 | } 914 | }, 915 | "node_modules/cliui/node_modules/strip-ansi": { 916 | "version": "6.0.1", 917 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 918 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 919 | "dev": true, 920 | "dependencies": { 921 | "ansi-regex": "^5.0.1" 922 | }, 923 | "engines": { 924 | "node": ">=8" 925 | } 926 | }, 927 | "node_modules/cluster-key-slot": { 928 | "version": "1.1.1", 929 | "resolved": "https://registry.npmjs.org/cluster-key-slot/-/cluster-key-slot-1.1.1.tgz", 930 | "integrity": "sha512-rwHwUfXL40Chm1r08yrhU3qpUvdVlgkKNeyeGPOxnW8/SyVDvgRaed/Uz54AqWNaTCAThlj6QAs3TZcKI0xDEw==", 931 | "dev": true, 932 | "engines": { 933 | "node": ">=0.10.0" 934 | } 935 | }, 936 | "node_modules/code-excerpt": { 937 | "version": "4.0.0", 938 | "resolved": "https://registry.npmjs.org/code-excerpt/-/code-excerpt-4.0.0.tgz", 939 | "integrity": "sha512-xxodCmBen3iy2i0WtAK8FlFNrRzjUqjRsMfho58xT/wvZU1YTM3fCnRjcy1gJPMepaRlgm/0e6w8SpWHpn3/cA==", 940 | "dev": true, 941 | "dependencies": { 942 | "convert-to-spaces": "^2.0.1" 943 | }, 944 | "engines": { 945 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 946 | } 947 | }, 948 | "node_modules/color-convert": { 949 | "version": "1.9.3", 950 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 951 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 952 | "dev": true, 953 | "dependencies": { 954 | "color-name": "1.1.3" 955 | } 956 | }, 957 | "node_modules/color-name": { 958 | "version": "1.1.3", 959 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 960 | "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", 961 | "dev": true 962 | }, 963 | "node_modules/commander": { 964 | "version": "2.20.3", 965 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", 966 | "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", 967 | "dev": true 968 | }, 969 | "node_modules/common-path-prefix": { 970 | "version": "3.0.0", 971 | "resolved": "https://registry.npmjs.org/common-path-prefix/-/common-path-prefix-3.0.0.tgz", 972 | "integrity": "sha512-QE33hToZseCH3jS0qN96O/bSh3kaw/h+Tq7ngyY9eWDUnTlTNUyqfqvCXioLe5Na5jFsL78ra/wuBU4iuEgd4w==", 973 | "dev": true 974 | }, 975 | "node_modules/concat-map": { 976 | "version": "0.0.1", 977 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 978 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", 979 | "dev": true 980 | }, 981 | "node_modules/concordance": { 982 | "version": "5.0.4", 983 | "resolved": "https://registry.npmjs.org/concordance/-/concordance-5.0.4.tgz", 984 | "integrity": "sha512-OAcsnTEYu1ARJqWVGwf4zh4JDfHZEaSNlNccFmt8YjB2l/n19/PF2viLINHc57vO4FKIAFl2FWASIGZZWZ2Kxw==", 985 | "dev": true, 986 | "dependencies": { 987 | "date-time": "^3.1.0", 988 | "esutils": "^2.0.3", 989 | "fast-diff": "^1.2.0", 990 | "js-string-escape": "^1.0.1", 991 | "lodash": "^4.17.15", 992 | "md5-hex": "^3.0.1", 993 | "semver": "^7.3.2", 994 | "well-known-symbols": "^2.0.0" 995 | }, 996 | "engines": { 997 | "node": ">=10.18.0 <11 || >=12.14.0 <13 || >=14" 998 | } 999 | }, 1000 | "node_modules/convert-source-map": { 1001 | "version": "1.9.0", 1002 | "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.9.0.tgz", 1003 | "integrity": "sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A==", 1004 | "dev": true 1005 | }, 1006 | "node_modules/convert-to-spaces": { 1007 | "version": "2.0.1", 1008 | "resolved": "https://registry.npmjs.org/convert-to-spaces/-/convert-to-spaces-2.0.1.tgz", 1009 | "integrity": "sha512-rcQ1bsQO9799wq24uE5AM2tAILy4gXGIK/njFWcVQkGNZ96edlpY+A7bjwvzjYvLDyzmG1MmMLZhpcsb+klNMQ==", 1010 | "dev": true, 1011 | "engines": { 1012 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 1013 | } 1014 | }, 1015 | "node_modules/cross-spawn": { 1016 | "version": "7.0.3", 1017 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", 1018 | "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", 1019 | "dev": true, 1020 | "dependencies": { 1021 | "path-key": "^3.1.0", 1022 | "shebang-command": "^2.0.0", 1023 | "which": "^2.0.1" 1024 | }, 1025 | "engines": { 1026 | "node": ">= 8" 1027 | } 1028 | }, 1029 | "node_modules/currently-unhandled": { 1030 | "version": "0.4.1", 1031 | "resolved": "https://registry.npmjs.org/currently-unhandled/-/currently-unhandled-0.4.1.tgz", 1032 | "integrity": "sha512-/fITjgjGU50vjQ4FH6eUoYu+iUoUKIXws2hL15JJpIR+BbTxaXQsMuuyjtNh2WqsSBS5nsaZHFsFecyw5CCAng==", 1033 | "dev": true, 1034 | "dependencies": { 1035 | "array-find-index": "^1.0.1" 1036 | }, 1037 | "engines": { 1038 | "node": ">=0.10.0" 1039 | } 1040 | }, 1041 | "node_modules/date-time": { 1042 | "version": "3.1.0", 1043 | "resolved": "https://registry.npmjs.org/date-time/-/date-time-3.1.0.tgz", 1044 | "integrity": "sha512-uqCUKXE5q1PNBXjPqvwhwJf9SwMoAHBgWJ6DcrnS5o+W2JOiIILl0JEdVD8SGujrNS02GGxgwAg2PN2zONgtjg==", 1045 | "dev": true, 1046 | "dependencies": { 1047 | "time-zone": "^1.0.0" 1048 | }, 1049 | "engines": { 1050 | "node": ">=6" 1051 | } 1052 | }, 1053 | "node_modules/debug": { 1054 | "version": "4.3.4", 1055 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 1056 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 1057 | "dev": true, 1058 | "dependencies": { 1059 | "ms": "2.1.2" 1060 | }, 1061 | "engines": { 1062 | "node": ">=6.0" 1063 | }, 1064 | "peerDependenciesMeta": { 1065 | "supports-color": { 1066 | "optional": true 1067 | } 1068 | } 1069 | }, 1070 | "node_modules/debug/node_modules/ms": { 1071 | "version": "2.1.2", 1072 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1073 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 1074 | "dev": true 1075 | }, 1076 | "node_modules/del": { 1077 | "version": "7.0.0", 1078 | "resolved": "https://registry.npmjs.org/del/-/del-7.0.0.tgz", 1079 | "integrity": "sha512-tQbV/4u5WVB8HMJr08pgw0b6nG4RGt/tj+7Numvq+zqcvUFeMaIWWOUFltiU+6go8BSO2/ogsB4EasDaj0y68Q==", 1080 | "dev": true, 1081 | "dependencies": { 1082 | "globby": "^13.1.2", 1083 | "graceful-fs": "^4.2.10", 1084 | "is-glob": "^4.0.3", 1085 | "is-path-cwd": "^3.0.0", 1086 | "is-path-inside": "^4.0.0", 1087 | "p-map": "^5.5.0", 1088 | "rimraf": "^3.0.2", 1089 | "slash": "^4.0.0" 1090 | }, 1091 | "engines": { 1092 | "node": ">=14.16" 1093 | }, 1094 | "funding": { 1095 | "url": "https://github.com/sponsors/sindresorhus" 1096 | } 1097 | }, 1098 | "node_modules/del/node_modules/slash": { 1099 | "version": "4.0.0", 1100 | "resolved": "https://registry.npmjs.org/slash/-/slash-4.0.0.tgz", 1101 | "integrity": "sha512-3dOsAHXXUkQTpOYcoAxLIorMTp4gIQr5IW3iVb7A7lFIp0VHhnynm9izx6TssdrIcVIESAlVjtnO2K8bg+Coew==", 1102 | "dev": true, 1103 | "engines": { 1104 | "node": ">=12" 1105 | }, 1106 | "funding": { 1107 | "url": "https://github.com/sponsors/sindresorhus" 1108 | } 1109 | }, 1110 | "node_modules/diff": { 1111 | "version": "4.0.2", 1112 | "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", 1113 | "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", 1114 | "dev": true, 1115 | "engines": { 1116 | "node": ">=0.3.1" 1117 | } 1118 | }, 1119 | "node_modules/dir-glob": { 1120 | "version": "3.0.1", 1121 | "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", 1122 | "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", 1123 | "dev": true, 1124 | "dependencies": { 1125 | "path-type": "^4.0.0" 1126 | }, 1127 | "engines": { 1128 | "node": ">=8" 1129 | } 1130 | }, 1131 | "node_modules/doctrine": { 1132 | "version": "0.7.2", 1133 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-0.7.2.tgz", 1134 | "integrity": "sha512-qiB/Rir6Un6Ad/TIgTRzsremsTGWzs8j7woXvp14jgq00676uBiBT5eUOi+FgRywZFVy5Us/c04ISRpZhRbS6w==", 1135 | "dev": true, 1136 | "dependencies": { 1137 | "esutils": "^1.1.6", 1138 | "isarray": "0.0.1" 1139 | }, 1140 | "engines": { 1141 | "node": ">=0.10.0" 1142 | } 1143 | }, 1144 | "node_modules/doctrine/node_modules/esutils": { 1145 | "version": "1.1.6", 1146 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-1.1.6.tgz", 1147 | "integrity": "sha512-RG1ZkUT7iFJG9LSHr7KDuuMSlujfeTtMNIcInURxKAxhMtwQhI3NrQhz26gZQYlsYZQKzsnwtpKrFKj9K9Qu1A==", 1148 | "dev": true, 1149 | "engines": { 1150 | "node": ">=0.10.0" 1151 | } 1152 | }, 1153 | "node_modules/eastasianwidth": { 1154 | "version": "0.2.0", 1155 | "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", 1156 | "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", 1157 | "dev": true 1158 | }, 1159 | "node_modules/emittery": { 1160 | "version": "1.0.1", 1161 | "resolved": "https://registry.npmjs.org/emittery/-/emittery-1.0.1.tgz", 1162 | "integrity": "sha512-2ID6FdrMD9KDLldGesP6317G78K7km/kMcwItRtVFva7I/cSEOIaLpewaUb+YLXVwdAp3Ctfxh/V5zIl1sj7dQ==", 1163 | "dev": true, 1164 | "engines": { 1165 | "node": ">=14.16" 1166 | }, 1167 | "funding": { 1168 | "url": "https://github.com/sindresorhus/emittery?sponsor=1" 1169 | } 1170 | }, 1171 | "node_modules/emoji-regex": { 1172 | "version": "9.2.2", 1173 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", 1174 | "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", 1175 | "dev": true 1176 | }, 1177 | "node_modules/escalade": { 1178 | "version": "3.1.1", 1179 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", 1180 | "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", 1181 | "dev": true, 1182 | "engines": { 1183 | "node": ">=6" 1184 | } 1185 | }, 1186 | "node_modules/escape-string-regexp": { 1187 | "version": "5.0.0", 1188 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-5.0.0.tgz", 1189 | "integrity": "sha512-/veY75JbMK4j1yjvuUxuVsiS/hr/4iHs9FTT6cgTexxdE0Ly/glccBAkloH/DofkjRbZU3bnoj38mOmhkZ0lHw==", 1190 | "dev": true, 1191 | "engines": { 1192 | "node": ">=12" 1193 | }, 1194 | "funding": { 1195 | "url": "https://github.com/sponsors/sindresorhus" 1196 | } 1197 | }, 1198 | "node_modules/esprima": { 1199 | "version": "4.0.1", 1200 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 1201 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", 1202 | "dev": true, 1203 | "bin": { 1204 | "esparse": "bin/esparse.js", 1205 | "esvalidate": "bin/esvalidate.js" 1206 | }, 1207 | "engines": { 1208 | "node": ">=4" 1209 | } 1210 | }, 1211 | "node_modules/esutils": { 1212 | "version": "2.0.3", 1213 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", 1214 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", 1215 | "dev": true, 1216 | "engines": { 1217 | "node": ">=0.10.0" 1218 | } 1219 | }, 1220 | "node_modules/fast-diff": { 1221 | "version": "1.2.0", 1222 | "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.2.0.tgz", 1223 | "integrity": "sha512-xJuoT5+L99XlZ8twedaRf6Ax2TgQVxvgZOYoPKqZufmJib0tL2tegPBOZb1pVNgIhlqDlA0eO0c3wBvQcmzx4w==", 1224 | "dev": true 1225 | }, 1226 | "node_modules/fast-glob": { 1227 | "version": "3.2.12", 1228 | "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.12.tgz", 1229 | "integrity": "sha512-DVj4CQIYYow0BlaelwK1pHl5n5cRSJfM60UA0zK891sVInoPri2Ekj7+e1CT3/3qxXenpI+nBBmQAcJPJgaj4w==", 1230 | "dev": true, 1231 | "dependencies": { 1232 | "@nodelib/fs.stat": "^2.0.2", 1233 | "@nodelib/fs.walk": "^1.2.3", 1234 | "glob-parent": "^5.1.2", 1235 | "merge2": "^1.3.0", 1236 | "micromatch": "^4.0.4" 1237 | }, 1238 | "engines": { 1239 | "node": ">=8.6.0" 1240 | } 1241 | }, 1242 | "node_modules/fastq": { 1243 | "version": "1.14.0", 1244 | "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.14.0.tgz", 1245 | "integrity": "sha512-eR2D+V9/ExcbF9ls441yIuN6TI2ED1Y2ZcA5BmMtJsOkWOFRJQ0Jt0g1UwqXJJVAb+V+umH5Dfr8oh4EVP7VVg==", 1246 | "dev": true, 1247 | "dependencies": { 1248 | "reusify": "^1.0.4" 1249 | } 1250 | }, 1251 | "node_modules/figures": { 1252 | "version": "5.0.0", 1253 | "resolved": "https://registry.npmjs.org/figures/-/figures-5.0.0.tgz", 1254 | "integrity": "sha512-ej8ksPF4x6e5wvK9yevct0UCXh8TTFlWGVLlgjZuoBH1HwjIfKE/IdL5mq89sFA7zELi1VhKpmtDnrs7zWyeyg==", 1255 | "dev": true, 1256 | "dependencies": { 1257 | "escape-string-regexp": "^5.0.0", 1258 | "is-unicode-supported": "^1.2.0" 1259 | }, 1260 | "engines": { 1261 | "node": ">=14" 1262 | }, 1263 | "funding": { 1264 | "url": "https://github.com/sponsors/sindresorhus" 1265 | } 1266 | }, 1267 | "node_modules/fill-range": { 1268 | "version": "7.0.1", 1269 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 1270 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 1271 | "dev": true, 1272 | "dependencies": { 1273 | "to-regex-range": "^5.0.1" 1274 | }, 1275 | "engines": { 1276 | "node": ">=8" 1277 | } 1278 | }, 1279 | "node_modules/find-up": { 1280 | "version": "5.0.0", 1281 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", 1282 | "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", 1283 | "dev": true, 1284 | "dependencies": { 1285 | "locate-path": "^6.0.0", 1286 | "path-exists": "^4.0.0" 1287 | }, 1288 | "engines": { 1289 | "node": ">=10" 1290 | }, 1291 | "funding": { 1292 | "url": "https://github.com/sponsors/sindresorhus" 1293 | } 1294 | }, 1295 | "node_modules/foreground-child": { 1296 | "version": "2.0.0", 1297 | "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-2.0.0.tgz", 1298 | "integrity": "sha512-dCIq9FpEcyQyXKCkyzmlPTFNgrCzPudOe+mhvJU5zAtlBnGVy2yKxtfsxK2tQBThwq225jcvBjpw1Gr40uzZCA==", 1299 | "dev": true, 1300 | "dependencies": { 1301 | "cross-spawn": "^7.0.0", 1302 | "signal-exit": "^3.0.2" 1303 | }, 1304 | "engines": { 1305 | "node": ">=8.0.0" 1306 | } 1307 | }, 1308 | "node_modules/fs.realpath": { 1309 | "version": "1.0.0", 1310 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 1311 | "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", 1312 | "dev": true 1313 | }, 1314 | "node_modules/fsevents": { 1315 | "version": "2.3.2", 1316 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 1317 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 1318 | "dev": true, 1319 | "hasInstallScript": true, 1320 | "optional": true, 1321 | "os": [ 1322 | "darwin" 1323 | ], 1324 | "engines": { 1325 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 1326 | } 1327 | }, 1328 | "node_modules/function-bind": { 1329 | "version": "1.1.1", 1330 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 1331 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", 1332 | "dev": true 1333 | }, 1334 | "node_modules/generic-pool": { 1335 | "version": "3.9.0", 1336 | "resolved": "https://registry.npmjs.org/generic-pool/-/generic-pool-3.9.0.tgz", 1337 | "integrity": "sha512-hymDOu5B53XvN4QT9dBmZxPX4CWhBPPLguTZ9MMFeFa/Kg0xWVfylOVNlJji/E7yTZWFd/q9GO5TxDLq156D7g==", 1338 | "dev": true, 1339 | "engines": { 1340 | "node": ">= 4" 1341 | } 1342 | }, 1343 | "node_modules/get-caller-file": { 1344 | "version": "2.0.5", 1345 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 1346 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 1347 | "dev": true, 1348 | "engines": { 1349 | "node": "6.* || 8.* || >= 10.*" 1350 | } 1351 | }, 1352 | "node_modules/glob": { 1353 | "version": "7.2.3", 1354 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", 1355 | "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", 1356 | "dev": true, 1357 | "dependencies": { 1358 | "fs.realpath": "^1.0.0", 1359 | "inflight": "^1.0.4", 1360 | "inherits": "2", 1361 | "minimatch": "^3.1.1", 1362 | "once": "^1.3.0", 1363 | "path-is-absolute": "^1.0.0" 1364 | }, 1365 | "engines": { 1366 | "node": "*" 1367 | }, 1368 | "funding": { 1369 | "url": "https://github.com/sponsors/isaacs" 1370 | } 1371 | }, 1372 | "node_modules/glob-parent": { 1373 | "version": "5.1.2", 1374 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 1375 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 1376 | "dev": true, 1377 | "dependencies": { 1378 | "is-glob": "^4.0.1" 1379 | }, 1380 | "engines": { 1381 | "node": ">= 6" 1382 | } 1383 | }, 1384 | "node_modules/globby": { 1385 | "version": "13.1.2", 1386 | "resolved": "https://registry.npmjs.org/globby/-/globby-13.1.2.tgz", 1387 | "integrity": "sha512-LKSDZXToac40u8Q1PQtZihbNdTYSNMuWe+K5l+oa6KgDzSvVrHXlJy40hUP522RjAIoNLJYBJi7ow+rbFpIhHQ==", 1388 | "dev": true, 1389 | "dependencies": { 1390 | "dir-glob": "^3.0.1", 1391 | "fast-glob": "^3.2.11", 1392 | "ignore": "^5.2.0", 1393 | "merge2": "^1.4.1", 1394 | "slash": "^4.0.0" 1395 | }, 1396 | "engines": { 1397 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 1398 | }, 1399 | "funding": { 1400 | "url": "https://github.com/sponsors/sindresorhus" 1401 | } 1402 | }, 1403 | "node_modules/globby/node_modules/slash": { 1404 | "version": "4.0.0", 1405 | "resolved": "https://registry.npmjs.org/slash/-/slash-4.0.0.tgz", 1406 | "integrity": "sha512-3dOsAHXXUkQTpOYcoAxLIorMTp4gIQr5IW3iVb7A7lFIp0VHhnynm9izx6TssdrIcVIESAlVjtnO2K8bg+Coew==", 1407 | "dev": true, 1408 | "engines": { 1409 | "node": ">=12" 1410 | }, 1411 | "funding": { 1412 | "url": "https://github.com/sponsors/sindresorhus" 1413 | } 1414 | }, 1415 | "node_modules/graceful-fs": { 1416 | "version": "4.2.10", 1417 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", 1418 | "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==", 1419 | "dev": true 1420 | }, 1421 | "node_modules/has": { 1422 | "version": "1.0.3", 1423 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 1424 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 1425 | "dev": true, 1426 | "dependencies": { 1427 | "function-bind": "^1.1.1" 1428 | }, 1429 | "engines": { 1430 | "node": ">= 0.4.0" 1431 | } 1432 | }, 1433 | "node_modules/has-flag": { 1434 | "version": "4.0.0", 1435 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 1436 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 1437 | "dev": true, 1438 | "engines": { 1439 | "node": ">=8" 1440 | } 1441 | }, 1442 | "node_modules/html-escaper": { 1443 | "version": "2.0.2", 1444 | "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", 1445 | "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", 1446 | "dev": true 1447 | }, 1448 | "node_modules/ignore": { 1449 | "version": "5.2.1", 1450 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.1.tgz", 1451 | "integrity": "sha512-d2qQLzTJ9WxQftPAuEQpSPmKqzxePjzVbpAVv62AQ64NTL+wR4JkrVqR/LqFsFEUsHDAiId52mJteHDFuDkElA==", 1452 | "dev": true, 1453 | "engines": { 1454 | "node": ">= 4" 1455 | } 1456 | }, 1457 | "node_modules/ignore-by-default": { 1458 | "version": "2.1.0", 1459 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-2.1.0.tgz", 1460 | "integrity": "sha512-yiWd4GVmJp0Q6ghmM2B/V3oZGRmjrKLXvHR3TE1nfoXsmoggllfZUQe74EN0fJdPFZu2NIvNdrMMLm3OsV7Ohw==", 1461 | "dev": true, 1462 | "engines": { 1463 | "node": ">=10 <11 || >=12 <13 || >=14" 1464 | } 1465 | }, 1466 | "node_modules/imurmurhash": { 1467 | "version": "0.1.4", 1468 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 1469 | "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", 1470 | "dev": true, 1471 | "engines": { 1472 | "node": ">=0.8.19" 1473 | } 1474 | }, 1475 | "node_modules/indent-string": { 1476 | "version": "5.0.0", 1477 | "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-5.0.0.tgz", 1478 | "integrity": "sha512-m6FAo/spmsW2Ab2fU35JTYwtOKa2yAwXSwgjSv1TJzh4Mh7mC3lzAOVLBprb72XsTrgkEIsl7YrFNAiDiRhIGg==", 1479 | "dev": true, 1480 | "engines": { 1481 | "node": ">=12" 1482 | }, 1483 | "funding": { 1484 | "url": "https://github.com/sponsors/sindresorhus" 1485 | } 1486 | }, 1487 | "node_modules/inflight": { 1488 | "version": "1.0.6", 1489 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 1490 | "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", 1491 | "dev": true, 1492 | "dependencies": { 1493 | "once": "^1.3.0", 1494 | "wrappy": "1" 1495 | } 1496 | }, 1497 | "node_modules/inherits": { 1498 | "version": "2.0.4", 1499 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 1500 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 1501 | "dev": true 1502 | }, 1503 | "node_modules/inversify": { 1504 | "version": "5.1.1", 1505 | "resolved": "https://registry.npmjs.org/inversify/-/inversify-5.1.1.tgz", 1506 | "integrity": "sha512-j8grHGDzv1v+8T1sAQ+3boTCntFPfvxLCkNcxB1J8qA0lUN+fAlSyYd+RXKvaPRL4AGyPxViutBEJHNXOyUdFQ==", 1507 | "dev": true 1508 | }, 1509 | "node_modules/irregular-plurals": { 1510 | "version": "3.3.0", 1511 | "resolved": "https://registry.npmjs.org/irregular-plurals/-/irregular-plurals-3.3.0.tgz", 1512 | "integrity": "sha512-MVBLKUTangM3EfRPFROhmWQQKRDsrgI83J8GS3jXy+OwYqiR2/aoWndYQ5416jLE3uaGgLH7ncme3X9y09gZ3g==", 1513 | "dev": true, 1514 | "engines": { 1515 | "node": ">=8" 1516 | } 1517 | }, 1518 | "node_modules/is-binary-path": { 1519 | "version": "2.1.0", 1520 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 1521 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 1522 | "dev": true, 1523 | "dependencies": { 1524 | "binary-extensions": "^2.0.0" 1525 | }, 1526 | "engines": { 1527 | "node": ">=8" 1528 | } 1529 | }, 1530 | "node_modules/is-core-module": { 1531 | "version": "2.11.0", 1532 | "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.11.0.tgz", 1533 | "integrity": "sha512-RRjxlvLDkD1YJwDbroBHMb+cukurkDWNyHx7D3oNB5x9rb5ogcksMC5wHCadcXoo67gVr/+3GFySh3134zi6rw==", 1534 | "dev": true, 1535 | "dependencies": { 1536 | "has": "^1.0.3" 1537 | }, 1538 | "funding": { 1539 | "url": "https://github.com/sponsors/ljharb" 1540 | } 1541 | }, 1542 | "node_modules/is-error": { 1543 | "version": "2.2.2", 1544 | "resolved": "https://registry.npmjs.org/is-error/-/is-error-2.2.2.tgz", 1545 | "integrity": "sha512-IOQqts/aHWbiisY5DuPJQ0gcbvaLFCa7fBa9xoLfxBZvQ+ZI/Zh9xoI7Gk+G64N0FdK4AbibytHht2tWgpJWLg==", 1546 | "dev": true 1547 | }, 1548 | "node_modules/is-extglob": { 1549 | "version": "2.1.1", 1550 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 1551 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", 1552 | "dev": true, 1553 | "engines": { 1554 | "node": ">=0.10.0" 1555 | } 1556 | }, 1557 | "node_modules/is-fullwidth-code-point": { 1558 | "version": "4.0.0", 1559 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-4.0.0.tgz", 1560 | "integrity": "sha512-O4L094N2/dZ7xqVdrXhh9r1KODPJpFms8B5sGdJLPy664AgvXsreZUyCQQNItZRDlYug4xStLjNp/sz3HvBowQ==", 1561 | "dev": true, 1562 | "engines": { 1563 | "node": ">=12" 1564 | }, 1565 | "funding": { 1566 | "url": "https://github.com/sponsors/sindresorhus" 1567 | } 1568 | }, 1569 | "node_modules/is-glob": { 1570 | "version": "4.0.3", 1571 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 1572 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 1573 | "dev": true, 1574 | "dependencies": { 1575 | "is-extglob": "^2.1.1" 1576 | }, 1577 | "engines": { 1578 | "node": ">=0.10.0" 1579 | } 1580 | }, 1581 | "node_modules/is-number": { 1582 | "version": "7.0.0", 1583 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 1584 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 1585 | "dev": true, 1586 | "engines": { 1587 | "node": ">=0.12.0" 1588 | } 1589 | }, 1590 | "node_modules/is-path-cwd": { 1591 | "version": "3.0.0", 1592 | "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-3.0.0.tgz", 1593 | "integrity": "sha512-kyiNFFLU0Ampr6SDZitD/DwUo4Zs1nSdnygUBqsu3LooL00Qvb5j+UnvApUn/TTj1J3OuE6BTdQ5rudKmU2ZaA==", 1594 | "dev": true, 1595 | "engines": { 1596 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 1597 | }, 1598 | "funding": { 1599 | "url": "https://github.com/sponsors/sindresorhus" 1600 | } 1601 | }, 1602 | "node_modules/is-path-inside": { 1603 | "version": "4.0.0", 1604 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-4.0.0.tgz", 1605 | "integrity": "sha512-lJJV/5dYS+RcL8uQdBDW9c9uWFLLBNRyFhnAKXw5tVqLlKZ4RMGZKv+YQ/IA3OhD+RpbJa1LLFM1FQPGyIXvOA==", 1606 | "dev": true, 1607 | "engines": { 1608 | "node": ">=12" 1609 | }, 1610 | "funding": { 1611 | "url": "https://github.com/sponsors/sindresorhus" 1612 | } 1613 | }, 1614 | "node_modules/is-plain-object": { 1615 | "version": "5.0.0", 1616 | "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-5.0.0.tgz", 1617 | "integrity": "sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q==", 1618 | "dev": true, 1619 | "engines": { 1620 | "node": ">=0.10.0" 1621 | } 1622 | }, 1623 | "node_modules/is-promise": { 1624 | "version": "4.0.0", 1625 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz", 1626 | "integrity": "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ==", 1627 | "dev": true 1628 | }, 1629 | "node_modules/is-unicode-supported": { 1630 | "version": "1.3.0", 1631 | "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-1.3.0.tgz", 1632 | "integrity": "sha512-43r2mRvz+8JRIKnWJ+3j8JtjRKZ6GmjzfaE/qiBJnikNnYv/6bagRJ1kUhNk8R5EX/GkobD+r+sfxCPJsiKBLQ==", 1633 | "dev": true, 1634 | "engines": { 1635 | "node": ">=12" 1636 | }, 1637 | "funding": { 1638 | "url": "https://github.com/sponsors/sindresorhus" 1639 | } 1640 | }, 1641 | "node_modules/isarray": { 1642 | "version": "0.0.1", 1643 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", 1644 | "integrity": "sha512-D2S+3GLxWH+uhrNEcoh/fnmYeP8E8/zHl644d/jdA0g2uyXvy3sb0qxotE+ne0LtccHknQzWwZEzhak7oJ0COQ==", 1645 | "dev": true 1646 | }, 1647 | "node_modules/isexe": { 1648 | "version": "2.0.0", 1649 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 1650 | "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", 1651 | "dev": true 1652 | }, 1653 | "node_modules/istanbul-lib-coverage": { 1654 | "version": "3.2.0", 1655 | "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.0.tgz", 1656 | "integrity": "sha512-eOeJ5BHCmHYvQK7xt9GkdHuzuCGS1Y6g9Gvnx3Ym33fz/HpLRYxiS0wHNr+m/MBC8B647Xt608vCDEvhl9c6Mw==", 1657 | "dev": true, 1658 | "engines": { 1659 | "node": ">=8" 1660 | } 1661 | }, 1662 | "node_modules/istanbul-lib-report": { 1663 | "version": "3.0.0", 1664 | "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", 1665 | "integrity": "sha512-wcdi+uAKzfiGT2abPpKZ0hSU1rGQjUQnLvtY5MpQ7QCTahD3VODhcu4wcfY1YtkGaDD5yuydOLINXsfbus9ROw==", 1666 | "dev": true, 1667 | "dependencies": { 1668 | "istanbul-lib-coverage": "^3.0.0", 1669 | "make-dir": "^3.0.0", 1670 | "supports-color": "^7.1.0" 1671 | }, 1672 | "engines": { 1673 | "node": ">=8" 1674 | } 1675 | }, 1676 | "node_modules/istanbul-reports": { 1677 | "version": "3.1.5", 1678 | "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.5.tgz", 1679 | "integrity": "sha512-nUsEMa9pBt/NOHqbcbeJEgqIlY/K7rVWUX6Lql2orY5e9roQOthbR3vtY4zzf2orPELg80fnxxk9zUyPlgwD1w==", 1680 | "dev": true, 1681 | "dependencies": { 1682 | "html-escaper": "^2.0.0", 1683 | "istanbul-lib-report": "^3.0.0" 1684 | }, 1685 | "engines": { 1686 | "node": ">=8" 1687 | } 1688 | }, 1689 | "node_modules/js-string-escape": { 1690 | "version": "1.0.1", 1691 | "resolved": "https://registry.npmjs.org/js-string-escape/-/js-string-escape-1.0.1.tgz", 1692 | "integrity": "sha512-Smw4xcfIQ5LVjAOuJCvN/zIodzA/BBSsluuoSykP+lUvScIi4U6RJLfwHet5cxFnCswUjISV8oAXaqaJDY3chg==", 1693 | "dev": true, 1694 | "engines": { 1695 | "node": ">= 0.8" 1696 | } 1697 | }, 1698 | "node_modules/js-tokens": { 1699 | "version": "4.0.0", 1700 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 1701 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 1702 | "dev": true 1703 | }, 1704 | "node_modules/js-yaml": { 1705 | "version": "3.14.1", 1706 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", 1707 | "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", 1708 | "dev": true, 1709 | "dependencies": { 1710 | "argparse": "^1.0.7", 1711 | "esprima": "^4.0.0" 1712 | }, 1713 | "bin": { 1714 | "js-yaml": "bin/js-yaml.js" 1715 | } 1716 | }, 1717 | "node_modules/load-json-file": { 1718 | "version": "7.0.1", 1719 | "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-7.0.1.tgz", 1720 | "integrity": "sha512-Gnxj3ev3mB5TkVBGad0JM6dmLiQL+o0t23JPBZ9sd+yvSLk05mFoqKBw5N8gbbkU4TNXyqCgIrl/VM17OgUIgQ==", 1721 | "dev": true, 1722 | "engines": { 1723 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 1724 | }, 1725 | "funding": { 1726 | "url": "https://github.com/sponsors/sindresorhus" 1727 | } 1728 | }, 1729 | "node_modules/locate-path": { 1730 | "version": "6.0.0", 1731 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", 1732 | "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", 1733 | "dev": true, 1734 | "dependencies": { 1735 | "p-locate": "^5.0.0" 1736 | }, 1737 | "engines": { 1738 | "node": ">=10" 1739 | }, 1740 | "funding": { 1741 | "url": "https://github.com/sponsors/sindresorhus" 1742 | } 1743 | }, 1744 | "node_modules/lodash": { 1745 | "version": "4.17.21", 1746 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", 1747 | "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", 1748 | "dev": true 1749 | }, 1750 | "node_modules/lru-cache": { 1751 | "version": "6.0.0", 1752 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", 1753 | "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", 1754 | "dev": true, 1755 | "dependencies": { 1756 | "yallist": "^4.0.0" 1757 | }, 1758 | "engines": { 1759 | "node": ">=10" 1760 | } 1761 | }, 1762 | "node_modules/make-dir": { 1763 | "version": "3.1.0", 1764 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 1765 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 1766 | "dev": true, 1767 | "dependencies": { 1768 | "semver": "^6.0.0" 1769 | }, 1770 | "engines": { 1771 | "node": ">=8" 1772 | }, 1773 | "funding": { 1774 | "url": "https://github.com/sponsors/sindresorhus" 1775 | } 1776 | }, 1777 | "node_modules/make-dir/node_modules/semver": { 1778 | "version": "6.3.0", 1779 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1780 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1781 | "dev": true, 1782 | "bin": { 1783 | "semver": "bin/semver.js" 1784 | } 1785 | }, 1786 | "node_modules/map-age-cleaner": { 1787 | "version": "0.1.3", 1788 | "resolved": "https://registry.npmjs.org/map-age-cleaner/-/map-age-cleaner-0.1.3.tgz", 1789 | "integrity": "sha512-bJzx6nMoP6PDLPBFmg7+xRKeFZvFboMrGlxmNj9ClvX53KrmvM5bXFXEWjbz4cz1AFn+jWJ9z/DJSz7hrs0w3w==", 1790 | "dev": true, 1791 | "dependencies": { 1792 | "p-defer": "^1.0.0" 1793 | }, 1794 | "engines": { 1795 | "node": ">=6" 1796 | } 1797 | }, 1798 | "node_modules/matcher": { 1799 | "version": "5.0.0", 1800 | "resolved": "https://registry.npmjs.org/matcher/-/matcher-5.0.0.tgz", 1801 | "integrity": "sha512-s2EMBOWtXFc8dgqvoAzKJXxNHibcdJMV0gwqKUaw9E2JBJuGUK7DrNKrA6g/i+v72TT16+6sVm5mS3thaMLQUw==", 1802 | "dev": true, 1803 | "dependencies": { 1804 | "escape-string-regexp": "^5.0.0" 1805 | }, 1806 | "engines": { 1807 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 1808 | }, 1809 | "funding": { 1810 | "url": "https://github.com/sponsors/sindresorhus" 1811 | } 1812 | }, 1813 | "node_modules/md5-hex": { 1814 | "version": "3.0.1", 1815 | "resolved": "https://registry.npmjs.org/md5-hex/-/md5-hex-3.0.1.tgz", 1816 | "integrity": "sha512-BUiRtTtV39LIJwinWBjqVsU9xhdnz7/i889V859IBFpuqGAj6LuOvHv5XLbgZ2R7ptJoJaEcxkv88/h25T7Ciw==", 1817 | "dev": true, 1818 | "dependencies": { 1819 | "blueimp-md5": "^2.10.0" 1820 | }, 1821 | "engines": { 1822 | "node": ">=8" 1823 | } 1824 | }, 1825 | "node_modules/mem": { 1826 | "version": "9.0.2", 1827 | "resolved": "https://registry.npmjs.org/mem/-/mem-9.0.2.tgz", 1828 | "integrity": "sha512-F2t4YIv9XQUBHt6AOJ0y7lSmP1+cY7Fm1DRh9GClTGzKST7UWLMx6ly9WZdLH/G/ppM5RL4MlQfRT71ri9t19A==", 1829 | "dev": true, 1830 | "dependencies": { 1831 | "map-age-cleaner": "^0.1.3", 1832 | "mimic-fn": "^4.0.0" 1833 | }, 1834 | "engines": { 1835 | "node": ">=12.20" 1836 | }, 1837 | "funding": { 1838 | "url": "https://github.com/sindresorhus/mem?sponsor=1" 1839 | } 1840 | }, 1841 | "node_modules/merge2": { 1842 | "version": "1.4.1", 1843 | "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", 1844 | "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", 1845 | "dev": true, 1846 | "engines": { 1847 | "node": ">= 8" 1848 | } 1849 | }, 1850 | "node_modules/micromatch": { 1851 | "version": "4.0.5", 1852 | "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", 1853 | "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", 1854 | "dev": true, 1855 | "dependencies": { 1856 | "braces": "^3.0.2", 1857 | "picomatch": "^2.3.1" 1858 | }, 1859 | "engines": { 1860 | "node": ">=8.6" 1861 | } 1862 | }, 1863 | "node_modules/mimic-fn": { 1864 | "version": "4.0.0", 1865 | "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-4.0.0.tgz", 1866 | "integrity": "sha512-vqiC06CuhBTUdZH+RYl8sFrL096vA45Ok5ISO6sE/Mr1jRbGH4Csnhi8f3wKVl7x8mO4Au7Ir9D3Oyv1VYMFJw==", 1867 | "dev": true, 1868 | "engines": { 1869 | "node": ">=12" 1870 | }, 1871 | "funding": { 1872 | "url": "https://github.com/sponsors/sindresorhus" 1873 | } 1874 | }, 1875 | "node_modules/minimatch": { 1876 | "version": "3.1.2", 1877 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 1878 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 1879 | "dev": true, 1880 | "dependencies": { 1881 | "brace-expansion": "^1.1.7" 1882 | }, 1883 | "engines": { 1884 | "node": "*" 1885 | } 1886 | }, 1887 | "node_modules/minimist": { 1888 | "version": "1.2.7", 1889 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.7.tgz", 1890 | "integrity": "sha512-bzfL1YUZsP41gmu/qjrEk0Q6i2ix/cVeAhbCbqH9u3zYutS1cLg00qhrD0M2MVdCcx4Sc0UpP2eBWo9rotpq6g==", 1891 | "dev": true, 1892 | "funding": { 1893 | "url": "https://github.com/sponsors/ljharb" 1894 | } 1895 | }, 1896 | "node_modules/mkdirp": { 1897 | "version": "0.5.6", 1898 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz", 1899 | "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==", 1900 | "dev": true, 1901 | "dependencies": { 1902 | "minimist": "^1.2.6" 1903 | }, 1904 | "bin": { 1905 | "mkdirp": "bin/cmd.js" 1906 | } 1907 | }, 1908 | "node_modules/ms": { 1909 | "version": "2.1.3", 1910 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1911 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 1912 | "dev": true 1913 | }, 1914 | "node_modules/nofilter": { 1915 | "version": "3.1.0", 1916 | "resolved": "https://registry.npmjs.org/nofilter/-/nofilter-3.1.0.tgz", 1917 | "integrity": "sha512-l2NNj07e9afPnhAhvgVrCD/oy2Ai1yfLpuo3EpiO1jFTsB4sFz6oIfAfSZyQzVpkZQ9xS8ZS5g1jCBgq4Hwo0g==", 1918 | "dev": true, 1919 | "engines": { 1920 | "node": ">=12.19" 1921 | } 1922 | }, 1923 | "node_modules/normalize-path": { 1924 | "version": "3.0.0", 1925 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1926 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 1927 | "dev": true, 1928 | "engines": { 1929 | "node": ">=0.10.0" 1930 | } 1931 | }, 1932 | "node_modules/once": { 1933 | "version": "1.4.0", 1934 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1935 | "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", 1936 | "dev": true, 1937 | "dependencies": { 1938 | "wrappy": "1" 1939 | } 1940 | }, 1941 | "node_modules/p-defer": { 1942 | "version": "1.0.0", 1943 | "resolved": "https://registry.npmjs.org/p-defer/-/p-defer-1.0.0.tgz", 1944 | "integrity": "sha512-wB3wfAxZpk2AzOfUMJNL+d36xothRSyj8EXOa4f6GMqYDN9BJaaSISbsk+wS9abmnebVw95C2Kb5t85UmpCxuw==", 1945 | "dev": true, 1946 | "engines": { 1947 | "node": ">=4" 1948 | } 1949 | }, 1950 | "node_modules/p-event": { 1951 | "version": "5.0.1", 1952 | "resolved": "https://registry.npmjs.org/p-event/-/p-event-5.0.1.tgz", 1953 | "integrity": "sha512-dd589iCQ7m1L0bmC5NLlVYfy3TbBEsMUfWx9PyAgPeIcFZ/E2yaTZ4Rz4MiBmmJShviiftHVXOqfnfzJ6kyMrQ==", 1954 | "dev": true, 1955 | "dependencies": { 1956 | "p-timeout": "^5.0.2" 1957 | }, 1958 | "engines": { 1959 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 1960 | }, 1961 | "funding": { 1962 | "url": "https://github.com/sponsors/sindresorhus" 1963 | } 1964 | }, 1965 | "node_modules/p-limit": { 1966 | "version": "3.1.0", 1967 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", 1968 | "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", 1969 | "dev": true, 1970 | "dependencies": { 1971 | "yocto-queue": "^0.1.0" 1972 | }, 1973 | "engines": { 1974 | "node": ">=10" 1975 | }, 1976 | "funding": { 1977 | "url": "https://github.com/sponsors/sindresorhus" 1978 | } 1979 | }, 1980 | "node_modules/p-locate": { 1981 | "version": "5.0.0", 1982 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", 1983 | "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", 1984 | "dev": true, 1985 | "dependencies": { 1986 | "p-limit": "^3.0.2" 1987 | }, 1988 | "engines": { 1989 | "node": ">=10" 1990 | }, 1991 | "funding": { 1992 | "url": "https://github.com/sponsors/sindresorhus" 1993 | } 1994 | }, 1995 | "node_modules/p-map": { 1996 | "version": "5.5.0", 1997 | "resolved": "https://registry.npmjs.org/p-map/-/p-map-5.5.0.tgz", 1998 | "integrity": "sha512-VFqfGDHlx87K66yZrNdI4YGtD70IRyd+zSvgks6mzHPRNkoKy+9EKP4SFC77/vTTQYmRmti7dvqC+m5jBrBAcg==", 1999 | "dev": true, 2000 | "dependencies": { 2001 | "aggregate-error": "^4.0.0" 2002 | }, 2003 | "engines": { 2004 | "node": ">=12" 2005 | }, 2006 | "funding": { 2007 | "url": "https://github.com/sponsors/sindresorhus" 2008 | } 2009 | }, 2010 | "node_modules/p-timeout": { 2011 | "version": "5.1.0", 2012 | "resolved": "https://registry.npmjs.org/p-timeout/-/p-timeout-5.1.0.tgz", 2013 | "integrity": "sha512-auFDyzzzGZZZdHz3BtET9VEz0SE/uMEAx7uWfGPucfzEwwe/xH0iVeZibQmANYE/hp9T2+UUZT5m+BKyrDp3Ew==", 2014 | "dev": true, 2015 | "engines": { 2016 | "node": ">=12" 2017 | }, 2018 | "funding": { 2019 | "url": "https://github.com/sponsors/sindresorhus" 2020 | } 2021 | }, 2022 | "node_modules/parse-ms": { 2023 | "version": "3.0.0", 2024 | "resolved": "https://registry.npmjs.org/parse-ms/-/parse-ms-3.0.0.tgz", 2025 | "integrity": "sha512-Tpb8Z7r7XbbtBTrM9UhpkzzaMrqA2VXMT3YChzYltwV3P3pM6t8wl7TvpMnSTosz1aQAdVib7kdoys7vYOPerw==", 2026 | "dev": true, 2027 | "engines": { 2028 | "node": ">=12" 2029 | }, 2030 | "funding": { 2031 | "url": "https://github.com/sponsors/sindresorhus" 2032 | } 2033 | }, 2034 | "node_modules/path-exists": { 2035 | "version": "4.0.0", 2036 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 2037 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 2038 | "dev": true, 2039 | "engines": { 2040 | "node": ">=8" 2041 | } 2042 | }, 2043 | "node_modules/path-is-absolute": { 2044 | "version": "1.0.1", 2045 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 2046 | "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", 2047 | "dev": true, 2048 | "engines": { 2049 | "node": ">=0.10.0" 2050 | } 2051 | }, 2052 | "node_modules/path-key": { 2053 | "version": "3.1.1", 2054 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", 2055 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", 2056 | "dev": true, 2057 | "engines": { 2058 | "node": ">=8" 2059 | } 2060 | }, 2061 | "node_modules/path-parse": { 2062 | "version": "1.0.7", 2063 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", 2064 | "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", 2065 | "dev": true 2066 | }, 2067 | "node_modules/path-type": { 2068 | "version": "4.0.0", 2069 | "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", 2070 | "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", 2071 | "dev": true, 2072 | "engines": { 2073 | "node": ">=8" 2074 | } 2075 | }, 2076 | "node_modules/picomatch": { 2077 | "version": "2.3.1", 2078 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 2079 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 2080 | "dev": true, 2081 | "engines": { 2082 | "node": ">=8.6" 2083 | }, 2084 | "funding": { 2085 | "url": "https://github.com/sponsors/jonschlinkert" 2086 | } 2087 | }, 2088 | "node_modules/pkg-conf": { 2089 | "version": "4.0.0", 2090 | "resolved": "https://registry.npmjs.org/pkg-conf/-/pkg-conf-4.0.0.tgz", 2091 | "integrity": "sha512-7dmgi4UY4qk+4mj5Cd8v/GExPo0K+SlY+hulOSdfZ/T6jVH6//y7NtzZo5WrfhDBxuQ0jCa7fLZmNaNh7EWL/w==", 2092 | "dev": true, 2093 | "dependencies": { 2094 | "find-up": "^6.0.0", 2095 | "load-json-file": "^7.0.0" 2096 | }, 2097 | "engines": { 2098 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 2099 | }, 2100 | "funding": { 2101 | "url": "https://github.com/sponsors/sindresorhus" 2102 | } 2103 | }, 2104 | "node_modules/pkg-conf/node_modules/find-up": { 2105 | "version": "6.3.0", 2106 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-6.3.0.tgz", 2107 | "integrity": "sha512-v2ZsoEuVHYy8ZIlYqwPe/39Cy+cFDzp4dXPaxNvkEuouymu+2Jbz0PxpKarJHYJTmv2HWT3O382qY8l4jMWthw==", 2108 | "dev": true, 2109 | "dependencies": { 2110 | "locate-path": "^7.1.0", 2111 | "path-exists": "^5.0.0" 2112 | }, 2113 | "engines": { 2114 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 2115 | }, 2116 | "funding": { 2117 | "url": "https://github.com/sponsors/sindresorhus" 2118 | } 2119 | }, 2120 | "node_modules/pkg-conf/node_modules/locate-path": { 2121 | "version": "7.1.1", 2122 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-7.1.1.tgz", 2123 | "integrity": "sha512-vJXaRMJgRVD3+cUZs3Mncj2mxpt5mP0EmNOsxRSZRMlbqjvxzDEOIUWXGmavo0ZC9+tNZCBLQ66reA11nbpHZg==", 2124 | "dev": true, 2125 | "dependencies": { 2126 | "p-locate": "^6.0.0" 2127 | }, 2128 | "engines": { 2129 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 2130 | }, 2131 | "funding": { 2132 | "url": "https://github.com/sponsors/sindresorhus" 2133 | } 2134 | }, 2135 | "node_modules/pkg-conf/node_modules/p-limit": { 2136 | "version": "4.0.0", 2137 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-4.0.0.tgz", 2138 | "integrity": "sha512-5b0R4txpzjPWVw/cXXUResoD4hb6U/x9BH08L7nw+GN1sezDzPdxeRvpc9c433fZhBan/wusjbCsqwqm4EIBIQ==", 2139 | "dev": true, 2140 | "dependencies": { 2141 | "yocto-queue": "^1.0.0" 2142 | }, 2143 | "engines": { 2144 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 2145 | }, 2146 | "funding": { 2147 | "url": "https://github.com/sponsors/sindresorhus" 2148 | } 2149 | }, 2150 | "node_modules/pkg-conf/node_modules/p-locate": { 2151 | "version": "6.0.0", 2152 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-6.0.0.tgz", 2153 | "integrity": "sha512-wPrq66Llhl7/4AGC6I+cqxT07LhXvWL08LNXz1fENOw0Ap4sRZZ/gZpTTJ5jpurzzzfS2W/Ge9BY3LgLjCShcw==", 2154 | "dev": true, 2155 | "dependencies": { 2156 | "p-limit": "^4.0.0" 2157 | }, 2158 | "engines": { 2159 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 2160 | }, 2161 | "funding": { 2162 | "url": "https://github.com/sponsors/sindresorhus" 2163 | } 2164 | }, 2165 | "node_modules/pkg-conf/node_modules/path-exists": { 2166 | "version": "5.0.0", 2167 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-5.0.0.tgz", 2168 | "integrity": "sha512-RjhtfwJOxzcFmNOi6ltcbcu4Iu+FL3zEj83dk4kAS+fVpTxXLO1b38RvJgT/0QwvV/L3aY9TAnyv0EOqW4GoMQ==", 2169 | "dev": true, 2170 | "engines": { 2171 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 2172 | } 2173 | }, 2174 | "node_modules/pkg-conf/node_modules/yocto-queue": { 2175 | "version": "1.0.0", 2176 | "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-1.0.0.tgz", 2177 | "integrity": "sha512-9bnSc/HEW2uRy67wc+T8UwauLuPJVn28jb+GtJY16iiKWyvmYJRXVT4UamsAEGQfPohgr2q4Tq0sQbQlxTfi1g==", 2178 | "dev": true, 2179 | "engines": { 2180 | "node": ">=12.20" 2181 | }, 2182 | "funding": { 2183 | "url": "https://github.com/sponsors/sindresorhus" 2184 | } 2185 | }, 2186 | "node_modules/plur": { 2187 | "version": "5.1.0", 2188 | "resolved": "https://registry.npmjs.org/plur/-/plur-5.1.0.tgz", 2189 | "integrity": "sha512-VP/72JeXqak2KiOzjgKtQen5y3IZHn+9GOuLDafPv0eXa47xq0At93XahYBs26MsifCQ4enGKwbjBTKgb9QJXg==", 2190 | "dev": true, 2191 | "dependencies": { 2192 | "irregular-plurals": "^3.3.0" 2193 | }, 2194 | "engines": { 2195 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 2196 | }, 2197 | "funding": { 2198 | "url": "https://github.com/sponsors/sindresorhus" 2199 | } 2200 | }, 2201 | "node_modules/prettier": { 2202 | "version": "2.8.0", 2203 | "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.8.0.tgz", 2204 | "integrity": "sha512-9Lmg8hTFZKG0Asr/kW9Bp8tJjRVluO8EJQVfY2T7FMw9T5jy4I/Uvx0Rca/XWf50QQ1/SS48+6IJWnrb+2yemA==", 2205 | "dev": true, 2206 | "bin": { 2207 | "prettier": "bin-prettier.js" 2208 | }, 2209 | "engines": { 2210 | "node": ">=10.13.0" 2211 | }, 2212 | "funding": { 2213 | "url": "https://github.com/prettier/prettier?sponsor=1" 2214 | } 2215 | }, 2216 | "node_modules/pretty-ms": { 2217 | "version": "8.0.0", 2218 | "resolved": "https://registry.npmjs.org/pretty-ms/-/pretty-ms-8.0.0.tgz", 2219 | "integrity": "sha512-ASJqOugUF1bbzI35STMBUpZqdfYKlJugy6JBziGi2EE+AL5JPJGSzvpeVXojxrr0ViUYoToUjb5kjSEGf7Y83Q==", 2220 | "dev": true, 2221 | "dependencies": { 2222 | "parse-ms": "^3.0.0" 2223 | }, 2224 | "engines": { 2225 | "node": ">=14.16" 2226 | }, 2227 | "funding": { 2228 | "url": "https://github.com/sponsors/sindresorhus" 2229 | } 2230 | }, 2231 | "node_modules/queue-microtask": { 2232 | "version": "1.2.3", 2233 | "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", 2234 | "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", 2235 | "dev": true, 2236 | "funding": [ 2237 | { 2238 | "type": "github", 2239 | "url": "https://github.com/sponsors/feross" 2240 | }, 2241 | { 2242 | "type": "patreon", 2243 | "url": "https://www.patreon.com/feross" 2244 | }, 2245 | { 2246 | "type": "consulting", 2247 | "url": "https://feross.org/support" 2248 | } 2249 | ] 2250 | }, 2251 | "node_modules/readdirp": { 2252 | "version": "3.6.0", 2253 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 2254 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 2255 | "dev": true, 2256 | "dependencies": { 2257 | "picomatch": "^2.2.1" 2258 | }, 2259 | "engines": { 2260 | "node": ">=8.10.0" 2261 | } 2262 | }, 2263 | "node_modules/redis": { 2264 | "version": "4.5.1", 2265 | "resolved": "https://registry.npmjs.org/redis/-/redis-4.5.1.tgz", 2266 | "integrity": "sha512-oxXSoIqMJCQVBTfxP6BNTCtDMyh9G6Vi5wjdPdV/sRKkufyZslDqCScSGcOr6XGR/reAWZefz7E4leM31RgdBA==", 2267 | "dev": true, 2268 | "dependencies": { 2269 | "@redis/bloom": "1.1.0", 2270 | "@redis/client": "1.4.2", 2271 | "@redis/graph": "1.1.0", 2272 | "@redis/json": "1.0.4", 2273 | "@redis/search": "1.1.0", 2274 | "@redis/time-series": "1.0.4" 2275 | } 2276 | }, 2277 | "node_modules/reflect-metadata": { 2278 | "version": "0.1.13", 2279 | "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.13.tgz", 2280 | "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==", 2281 | "dev": true 2282 | }, 2283 | "node_modules/require-directory": { 2284 | "version": "2.1.1", 2285 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 2286 | "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", 2287 | "dev": true, 2288 | "engines": { 2289 | "node": ">=0.10.0" 2290 | } 2291 | }, 2292 | "node_modules/resolve": { 2293 | "version": "1.22.1", 2294 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.1.tgz", 2295 | "integrity": "sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw==", 2296 | "dev": true, 2297 | "dependencies": { 2298 | "is-core-module": "^2.9.0", 2299 | "path-parse": "^1.0.7", 2300 | "supports-preserve-symlinks-flag": "^1.0.0" 2301 | }, 2302 | "bin": { 2303 | "resolve": "bin/resolve" 2304 | }, 2305 | "funding": { 2306 | "url": "https://github.com/sponsors/ljharb" 2307 | } 2308 | }, 2309 | "node_modules/resolve-cwd": { 2310 | "version": "3.0.0", 2311 | "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", 2312 | "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", 2313 | "dev": true, 2314 | "dependencies": { 2315 | "resolve-from": "^5.0.0" 2316 | }, 2317 | "engines": { 2318 | "node": ">=8" 2319 | } 2320 | }, 2321 | "node_modules/resolve-from": { 2322 | "version": "5.0.0", 2323 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", 2324 | "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", 2325 | "dev": true, 2326 | "engines": { 2327 | "node": ">=8" 2328 | } 2329 | }, 2330 | "node_modules/reusify": { 2331 | "version": "1.0.4", 2332 | "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", 2333 | "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", 2334 | "dev": true, 2335 | "engines": { 2336 | "iojs": ">=1.0.0", 2337 | "node": ">=0.10.0" 2338 | } 2339 | }, 2340 | "node_modules/rimraf": { 2341 | "version": "3.0.2", 2342 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", 2343 | "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", 2344 | "dev": true, 2345 | "dependencies": { 2346 | "glob": "^7.1.3" 2347 | }, 2348 | "bin": { 2349 | "rimraf": "bin.js" 2350 | }, 2351 | "funding": { 2352 | "url": "https://github.com/sponsors/isaacs" 2353 | } 2354 | }, 2355 | "node_modules/run-parallel": { 2356 | "version": "1.2.0", 2357 | "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", 2358 | "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", 2359 | "dev": true, 2360 | "funding": [ 2361 | { 2362 | "type": "github", 2363 | "url": "https://github.com/sponsors/feross" 2364 | }, 2365 | { 2366 | "type": "patreon", 2367 | "url": "https://www.patreon.com/feross" 2368 | }, 2369 | { 2370 | "type": "consulting", 2371 | "url": "https://feross.org/support" 2372 | } 2373 | ], 2374 | "dependencies": { 2375 | "queue-microtask": "^1.2.2" 2376 | } 2377 | }, 2378 | "node_modules/semver": { 2379 | "version": "7.3.8", 2380 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.8.tgz", 2381 | "integrity": "sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==", 2382 | "dev": true, 2383 | "dependencies": { 2384 | "lru-cache": "^6.0.0" 2385 | }, 2386 | "bin": { 2387 | "semver": "bin/semver.js" 2388 | }, 2389 | "engines": { 2390 | "node": ">=10" 2391 | } 2392 | }, 2393 | "node_modules/serialize-error": { 2394 | "version": "7.0.1", 2395 | "resolved": "https://registry.npmjs.org/serialize-error/-/serialize-error-7.0.1.tgz", 2396 | "integrity": "sha512-8I8TjW5KMOKsZQTvoxjuSIa7foAwPWGOts+6o7sgjz41/qMD9VQHEDxi6PBvK2l0MXUmqZyNpUK+T2tQaaElvw==", 2397 | "dev": true, 2398 | "dependencies": { 2399 | "type-fest": "^0.13.1" 2400 | }, 2401 | "engines": { 2402 | "node": ">=10" 2403 | }, 2404 | "funding": { 2405 | "url": "https://github.com/sponsors/sindresorhus" 2406 | } 2407 | }, 2408 | "node_modules/shebang-command": { 2409 | "version": "2.0.0", 2410 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", 2411 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", 2412 | "dev": true, 2413 | "dependencies": { 2414 | "shebang-regex": "^3.0.0" 2415 | }, 2416 | "engines": { 2417 | "node": ">=8" 2418 | } 2419 | }, 2420 | "node_modules/shebang-regex": { 2421 | "version": "3.0.0", 2422 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", 2423 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", 2424 | "dev": true, 2425 | "engines": { 2426 | "node": ">=8" 2427 | } 2428 | }, 2429 | "node_modules/signal-exit": { 2430 | "version": "3.0.7", 2431 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", 2432 | "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", 2433 | "dev": true 2434 | }, 2435 | "node_modules/slash": { 2436 | "version": "3.0.0", 2437 | "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", 2438 | "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", 2439 | "dev": true, 2440 | "engines": { 2441 | "node": ">=8" 2442 | } 2443 | }, 2444 | "node_modules/slice-ansi": { 2445 | "version": "5.0.0", 2446 | "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-5.0.0.tgz", 2447 | "integrity": "sha512-FC+lgizVPfie0kkhqUScwRu1O/lF6NOgJmlCgK+/LYxDCTk8sGelYaHDhFcDN+Sn3Cv+3VSa4Byeo+IMCzpMgQ==", 2448 | "dev": true, 2449 | "dependencies": { 2450 | "ansi-styles": "^6.0.0", 2451 | "is-fullwidth-code-point": "^4.0.0" 2452 | }, 2453 | "engines": { 2454 | "node": ">=12" 2455 | }, 2456 | "funding": { 2457 | "url": "https://github.com/chalk/slice-ansi?sponsor=1" 2458 | } 2459 | }, 2460 | "node_modules/source-map": { 2461 | "version": "0.6.1", 2462 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 2463 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", 2464 | "dev": true, 2465 | "engines": { 2466 | "node": ">=0.10.0" 2467 | } 2468 | }, 2469 | "node_modules/source-map-support": { 2470 | "version": "0.5.21", 2471 | "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", 2472 | "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", 2473 | "dev": true, 2474 | "dependencies": { 2475 | "buffer-from": "^1.0.0", 2476 | "source-map": "^0.6.0" 2477 | } 2478 | }, 2479 | "node_modules/sprintf-js": { 2480 | "version": "1.0.3", 2481 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 2482 | "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", 2483 | "dev": true 2484 | }, 2485 | "node_modules/stack-utils": { 2486 | "version": "2.0.6", 2487 | "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-2.0.6.tgz", 2488 | "integrity": "sha512-XlkWvfIm6RmsWtNJx+uqtKLS8eqFbxUg0ZzLXqY0caEy9l7hruX8IpiDnjsLavoBgqCCR71TqWO8MaXYheJ3RQ==", 2489 | "dev": true, 2490 | "dependencies": { 2491 | "escape-string-regexp": "^2.0.0" 2492 | }, 2493 | "engines": { 2494 | "node": ">=10" 2495 | } 2496 | }, 2497 | "node_modules/stack-utils/node_modules/escape-string-regexp": { 2498 | "version": "2.0.0", 2499 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", 2500 | "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", 2501 | "dev": true, 2502 | "engines": { 2503 | "node": ">=8" 2504 | } 2505 | }, 2506 | "node_modules/string-width": { 2507 | "version": "5.1.2", 2508 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", 2509 | "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", 2510 | "dev": true, 2511 | "dependencies": { 2512 | "eastasianwidth": "^0.2.0", 2513 | "emoji-regex": "^9.2.2", 2514 | "strip-ansi": "^7.0.1" 2515 | }, 2516 | "engines": { 2517 | "node": ">=12" 2518 | }, 2519 | "funding": { 2520 | "url": "https://github.com/sponsors/sindresorhus" 2521 | } 2522 | }, 2523 | "node_modules/strip-ansi": { 2524 | "version": "7.0.1", 2525 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.0.1.tgz", 2526 | "integrity": "sha512-cXNxvT8dFNRVfhVME3JAe98mkXDYN2O1l7jmcwMnOslDeESg1rF/OZMtK0nRAhiari1unG5cD4jG3rapUAkLbw==", 2527 | "dev": true, 2528 | "dependencies": { 2529 | "ansi-regex": "^6.0.1" 2530 | }, 2531 | "engines": { 2532 | "node": ">=12" 2533 | }, 2534 | "funding": { 2535 | "url": "https://github.com/chalk/strip-ansi?sponsor=1" 2536 | } 2537 | }, 2538 | "node_modules/supertap": { 2539 | "version": "3.0.1", 2540 | "resolved": "https://registry.npmjs.org/supertap/-/supertap-3.0.1.tgz", 2541 | "integrity": "sha512-u1ZpIBCawJnO+0QePsEiOknOfCRq0yERxiAchT0i4li0WHNUJbf0evXXSXOcCAR4M8iMDoajXYmstm/qO81Isw==", 2542 | "dev": true, 2543 | "dependencies": { 2544 | "indent-string": "^5.0.0", 2545 | "js-yaml": "^3.14.1", 2546 | "serialize-error": "^7.0.1", 2547 | "strip-ansi": "^7.0.1" 2548 | }, 2549 | "engines": { 2550 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0" 2551 | } 2552 | }, 2553 | "node_modules/supports-color": { 2554 | "version": "7.2.0", 2555 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 2556 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 2557 | "dev": true, 2558 | "dependencies": { 2559 | "has-flag": "^4.0.0" 2560 | }, 2561 | "engines": { 2562 | "node": ">=8" 2563 | } 2564 | }, 2565 | "node_modules/supports-preserve-symlinks-flag": { 2566 | "version": "1.0.0", 2567 | "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", 2568 | "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", 2569 | "dev": true, 2570 | "engines": { 2571 | "node": ">= 0.4" 2572 | }, 2573 | "funding": { 2574 | "url": "https://github.com/sponsors/ljharb" 2575 | } 2576 | }, 2577 | "node_modules/temp-dir": { 2578 | "version": "3.0.0", 2579 | "resolved": "https://registry.npmjs.org/temp-dir/-/temp-dir-3.0.0.tgz", 2580 | "integrity": "sha512-nHc6S/bwIilKHNRgK/3jlhDoIHcp45YgyiwcAk46Tr0LfEqGBVpmiAyuiuxeVE44m3mXnEeVhaipLOEWmH+Njw==", 2581 | "dev": true, 2582 | "engines": { 2583 | "node": ">=14.16" 2584 | } 2585 | }, 2586 | "node_modules/terser": { 2587 | "version": "5.16.1", 2588 | "resolved": "https://registry.npmjs.org/terser/-/terser-5.16.1.tgz", 2589 | "integrity": "sha512-xvQfyfA1ayT0qdK47zskQgRZeWLoOQ8JQ6mIgRGVNwZKdQMU+5FkCBjmv4QjcrTzyZquRw2FVtlJSRUmMKQslw==", 2590 | "dev": true, 2591 | "dependencies": { 2592 | "@jridgewell/source-map": "^0.3.2", 2593 | "acorn": "^8.5.0", 2594 | "commander": "^2.20.0", 2595 | "source-map-support": "~0.5.20" 2596 | }, 2597 | "bin": { 2598 | "terser": "bin/terser" 2599 | }, 2600 | "engines": { 2601 | "node": ">=10" 2602 | } 2603 | }, 2604 | "node_modules/test-exclude": { 2605 | "version": "6.0.0", 2606 | "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", 2607 | "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", 2608 | "dev": true, 2609 | "dependencies": { 2610 | "@istanbuljs/schema": "^0.1.2", 2611 | "glob": "^7.1.4", 2612 | "minimatch": "^3.0.4" 2613 | }, 2614 | "engines": { 2615 | "node": ">=8" 2616 | } 2617 | }, 2618 | "node_modules/time-zone": { 2619 | "version": "1.0.0", 2620 | "resolved": "https://registry.npmjs.org/time-zone/-/time-zone-1.0.0.tgz", 2621 | "integrity": "sha512-TIsDdtKo6+XrPtiTm1ssmMngN1sAhyKnTO2kunQWqNPWIVvCm15Wmw4SWInwTVgJ5u/Tr04+8Ei9TNcw4x4ONA==", 2622 | "dev": true, 2623 | "engines": { 2624 | "node": ">=4" 2625 | } 2626 | }, 2627 | "node_modules/to-regex-range": { 2628 | "version": "5.0.1", 2629 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 2630 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 2631 | "dev": true, 2632 | "dependencies": { 2633 | "is-number": "^7.0.0" 2634 | }, 2635 | "engines": { 2636 | "node": ">=8.0" 2637 | } 2638 | }, 2639 | "node_modules/tslib": { 2640 | "version": "1.14.1", 2641 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", 2642 | "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", 2643 | "dev": true 2644 | }, 2645 | "node_modules/tslint": { 2646 | "version": "5.20.1", 2647 | "resolved": "https://registry.npmjs.org/tslint/-/tslint-5.20.1.tgz", 2648 | "integrity": "sha512-EcMxhzCFt8k+/UP5r8waCf/lzmeSyVlqxqMEDQE7rWYiQky8KpIBz1JAoYXfROHrPZ1XXd43q8yQnULOLiBRQg==", 2649 | "dev": true, 2650 | "dependencies": { 2651 | "@babel/code-frame": "^7.0.0", 2652 | "builtin-modules": "^1.1.1", 2653 | "chalk": "^2.3.0", 2654 | "commander": "^2.12.1", 2655 | "diff": "^4.0.1", 2656 | "glob": "^7.1.1", 2657 | "js-yaml": "^3.13.1", 2658 | "minimatch": "^3.0.4", 2659 | "mkdirp": "^0.5.1", 2660 | "resolve": "^1.3.2", 2661 | "semver": "^5.3.0", 2662 | "tslib": "^1.8.0", 2663 | "tsutils": "^2.29.0" 2664 | }, 2665 | "bin": { 2666 | "tslint": "bin/tslint" 2667 | }, 2668 | "engines": { 2669 | "node": ">=4.8.0" 2670 | }, 2671 | "peerDependencies": { 2672 | "typescript": ">=2.3.0-dev || >=2.4.0-dev || >=2.5.0-dev || >=2.6.0-dev || >=2.7.0-dev || >=2.8.0-dev || >=2.9.0-dev || >=3.0.0-dev || >= 3.1.0-dev || >= 3.2.0-dev" 2673 | } 2674 | }, 2675 | "node_modules/tslint-config-airbnb": { 2676 | "version": "5.11.2", 2677 | "resolved": "https://registry.npmjs.org/tslint-config-airbnb/-/tslint-config-airbnb-5.11.2.tgz", 2678 | "integrity": "sha512-mUpHPTeeCFx8XARGG/kzYP4dPSOgoCqNiYbGHh09qTH8q+Y1ghsOgaeZKYYQT7IyxMos523z/QBaiv2zKNBcow==", 2679 | "dev": true, 2680 | "dependencies": { 2681 | "tslint-consistent-codestyle": "^1.14.1", 2682 | "tslint-eslint-rules": "^5.4.0", 2683 | "tslint-microsoft-contrib": "~5.2.1" 2684 | }, 2685 | "peerDependencies": { 2686 | "tslint": "^5.11.0" 2687 | } 2688 | }, 2689 | "node_modules/tslint-config-airbnb/node_modules/tslib": { 2690 | "version": "1.9.0", 2691 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.9.0.tgz", 2692 | "integrity": "sha512-f/qGG2tUkrISBlQZEjEqoZ3B2+npJjIf04H1wuAv9iA8i04Icp+61KRXxFdha22670NJopsZCIjhC3SnjPRKrQ==", 2693 | "dev": true 2694 | }, 2695 | "node_modules/tslint-config-airbnb/node_modules/tslint-eslint-rules": { 2696 | "version": "5.4.0", 2697 | "resolved": "https://registry.npmjs.org/tslint-eslint-rules/-/tslint-eslint-rules-5.4.0.tgz", 2698 | "integrity": "sha512-WlSXE+J2vY/VPgIcqQuijMQiel+UtmXS+4nvK4ZzlDiqBfXse8FAvkNnTcYhnQyOTW5KFM+uRRGXxYhFpuBc6w==", 2699 | "dev": true, 2700 | "dependencies": { 2701 | "doctrine": "0.7.2", 2702 | "tslib": "1.9.0", 2703 | "tsutils": "^3.0.0" 2704 | }, 2705 | "peerDependencies": { 2706 | "tslint": "^5.0.0", 2707 | "typescript": "^2.2.0 || ^3.0.0" 2708 | } 2709 | }, 2710 | "node_modules/tslint-config-airbnb/node_modules/tslint-microsoft-contrib": { 2711 | "version": "5.2.1", 2712 | "resolved": "https://registry.npmjs.org/tslint-microsoft-contrib/-/tslint-microsoft-contrib-5.2.1.tgz", 2713 | "integrity": "sha512-PDYjvpo0gN9IfMULwKk0KpVOPMhU6cNoT9VwCOLeDl/QS8v8W2yspRpFFuUS7/c5EIH/n8ApMi8TxJAz1tfFUA==", 2714 | "dev": true, 2715 | "dependencies": { 2716 | "tsutils": "^2.27.2 <2.29.0" 2717 | }, 2718 | "peerDependencies": { 2719 | "tslint": "^5.1.0", 2720 | "typescript": "^2.1.0 || ^3.0.0" 2721 | } 2722 | }, 2723 | "node_modules/tslint-config-airbnb/node_modules/tslint-microsoft-contrib/node_modules/tsutils": { 2724 | "version": "2.28.0", 2725 | "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.28.0.tgz", 2726 | "integrity": "sha512-bh5nAtW0tuhvOJnx1GLRn5ScraRLICGyJV5wJhtRWOLsxW70Kk5tZtpK3O/hW6LDnqKS9mlUMPZj9fEMJ0gxqA==", 2727 | "dev": true, 2728 | "dependencies": { 2729 | "tslib": "^1.8.1" 2730 | }, 2731 | "peerDependencies": { 2732 | "typescript": ">=2.1.0 || >=2.1.0-dev || >=2.2.0-dev || >=2.3.0-dev || >=2.4.0-dev || >=2.5.0-dev || >=2.6.0-dev || >=2.7.0-dev || >=2.8.0-dev || >=2.9.0-dev || >= 3.0.0-dev || >= 3.1.0-dev" 2733 | } 2734 | }, 2735 | "node_modules/tslint-config-airbnb/node_modules/tsutils": { 2736 | "version": "3.21.0", 2737 | "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.21.0.tgz", 2738 | "integrity": "sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==", 2739 | "dev": true, 2740 | "dependencies": { 2741 | "tslib": "^1.8.1" 2742 | }, 2743 | "engines": { 2744 | "node": ">= 6" 2745 | }, 2746 | "peerDependencies": { 2747 | "typescript": ">=2.8.0 || >= 3.2.0-dev || >= 3.3.0-dev || >= 3.4.0-dev || >= 3.5.0-dev || >= 3.6.0-dev || >= 3.6.0-beta || >= 3.7.0-dev || >= 3.7.0-beta" 2748 | } 2749 | }, 2750 | "node_modules/tslint-config-airbnb/node_modules/typescript": { 2751 | "version": "3.9.10", 2752 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.10.tgz", 2753 | "integrity": "sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q==", 2754 | "dev": true, 2755 | "peer": true, 2756 | "bin": { 2757 | "tsc": "bin/tsc", 2758 | "tsserver": "bin/tsserver" 2759 | }, 2760 | "engines": { 2761 | "node": ">=4.2.0" 2762 | } 2763 | }, 2764 | "node_modules/tslint-config-prettier": { 2765 | "version": "1.18.0", 2766 | "resolved": "https://registry.npmjs.org/tslint-config-prettier/-/tslint-config-prettier-1.18.0.tgz", 2767 | "integrity": "sha512-xPw9PgNPLG3iKRxmK7DWr+Ea/SzrvfHtjFt5LBl61gk2UBG/DB9kCXRjv+xyIU1rUtnayLeMUVJBcMX8Z17nDg==", 2768 | "dev": true, 2769 | "bin": { 2770 | "tslint-config-prettier-check": "bin/check.js" 2771 | }, 2772 | "engines": { 2773 | "node": ">=4.0.0" 2774 | } 2775 | }, 2776 | "node_modules/tslint-consistent-codestyle": { 2777 | "version": "1.16.0", 2778 | "resolved": "https://registry.npmjs.org/tslint-consistent-codestyle/-/tslint-consistent-codestyle-1.16.0.tgz", 2779 | "integrity": "sha512-ebR/xHyMEuU36hGNOgCfjGBNYxBPixf0yU1Yoo6s3BrpBRFccjPOmIVaVvQsWAUAMdmfzHOCihVkcaMfimqvHw==", 2780 | "dev": true, 2781 | "dependencies": { 2782 | "@fimbul/bifrost": "^0.21.0", 2783 | "tslib": "^1.7.1", 2784 | "tsutils": "^2.29.0" 2785 | }, 2786 | "peerDependencies": { 2787 | "tslint": "^5.0.0", 2788 | "typescript": ">=2.1.4 || >=2.1.0-dev || >=2.2.0-dev || >=2.3.0-dev || >=2.4.0-dev || >=2.5.0-dev || >=2.6.0-dev || >=2.7.0-dev || >=2.8.0-dev || >=2.9.0-dev || >=3.0.0-dev || >=3.1.0-dev || >=3.2.0-dev || >=3.3.0-dev || >=3.4.0-dev" 2789 | } 2790 | }, 2791 | "node_modules/tslint/node_modules/ansi-styles": { 2792 | "version": "3.2.1", 2793 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 2794 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 2795 | "dev": true, 2796 | "dependencies": { 2797 | "color-convert": "^1.9.0" 2798 | }, 2799 | "engines": { 2800 | "node": ">=4" 2801 | } 2802 | }, 2803 | "node_modules/tslint/node_modules/chalk": { 2804 | "version": "2.4.2", 2805 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 2806 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 2807 | "dev": true, 2808 | "dependencies": { 2809 | "ansi-styles": "^3.2.1", 2810 | "escape-string-regexp": "^1.0.5", 2811 | "supports-color": "^5.3.0" 2812 | }, 2813 | "engines": { 2814 | "node": ">=4" 2815 | } 2816 | }, 2817 | "node_modules/tslint/node_modules/escape-string-regexp": { 2818 | "version": "1.0.5", 2819 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 2820 | "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", 2821 | "dev": true, 2822 | "engines": { 2823 | "node": ">=0.8.0" 2824 | } 2825 | }, 2826 | "node_modules/tslint/node_modules/has-flag": { 2827 | "version": "3.0.0", 2828 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 2829 | "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", 2830 | "dev": true, 2831 | "engines": { 2832 | "node": ">=4" 2833 | } 2834 | }, 2835 | "node_modules/tslint/node_modules/semver": { 2836 | "version": "5.7.1", 2837 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 2838 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 2839 | "dev": true, 2840 | "bin": { 2841 | "semver": "bin/semver" 2842 | } 2843 | }, 2844 | "node_modules/tslint/node_modules/supports-color": { 2845 | "version": "5.5.0", 2846 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 2847 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 2848 | "dev": true, 2849 | "dependencies": { 2850 | "has-flag": "^3.0.0" 2851 | }, 2852 | "engines": { 2853 | "node": ">=4" 2854 | } 2855 | }, 2856 | "node_modules/tsutils": { 2857 | "version": "2.29.0", 2858 | "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", 2859 | "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", 2860 | "dev": true, 2861 | "dependencies": { 2862 | "tslib": "^1.8.1" 2863 | }, 2864 | "peerDependencies": { 2865 | "typescript": ">=2.1.0 || >=2.1.0-dev || >=2.2.0-dev || >=2.3.0-dev || >=2.4.0-dev || >=2.5.0-dev || >=2.6.0-dev || >=2.7.0-dev || >=2.8.0-dev || >=2.9.0-dev || >= 3.0.0-dev || >= 3.1.0-dev" 2866 | } 2867 | }, 2868 | "node_modules/type-fest": { 2869 | "version": "0.13.1", 2870 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.13.1.tgz", 2871 | "integrity": "sha512-34R7HTnG0XIJcBSn5XhDd7nNFPRcXYRZrBB2O2jdKqYODldSzBAqzsWoZYYvduky73toYS/ESqxPvkDf/F0XMg==", 2872 | "dev": true, 2873 | "engines": { 2874 | "node": ">=10" 2875 | }, 2876 | "funding": { 2877 | "url": "https://github.com/sponsors/sindresorhus" 2878 | } 2879 | }, 2880 | "node_modules/typescript": { 2881 | "version": "4.9.3", 2882 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.9.3.tgz", 2883 | "integrity": "sha512-CIfGzTelbKNEnLpLdGFgdyKhG23CKdKgQPOBc+OUNrkJ2vr+KSzsSV5kq5iWhEQbok+quxgGzrAtGWCyU7tHnA==", 2884 | "dev": true, 2885 | "bin": { 2886 | "tsc": "bin/tsc", 2887 | "tsserver": "bin/tsserver" 2888 | }, 2889 | "engines": { 2890 | "node": ">=4.2.0" 2891 | } 2892 | }, 2893 | "node_modules/v8-to-istanbul": { 2894 | "version": "9.0.1", 2895 | "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-9.0.1.tgz", 2896 | "integrity": "sha512-74Y4LqY74kLE6IFyIjPtkSTWzUZmj8tdHT9Ii/26dvQ6K9Dl2NbEfj0XgU2sHCtKgt5VupqhlO/5aWuqS+IY1w==", 2897 | "dev": true, 2898 | "dependencies": { 2899 | "@jridgewell/trace-mapping": "^0.3.12", 2900 | "@types/istanbul-lib-coverage": "^2.0.1", 2901 | "convert-source-map": "^1.6.0" 2902 | }, 2903 | "engines": { 2904 | "node": ">=10.12.0" 2905 | } 2906 | }, 2907 | "node_modules/well-known-symbols": { 2908 | "version": "2.0.0", 2909 | "resolved": "https://registry.npmjs.org/well-known-symbols/-/well-known-symbols-2.0.0.tgz", 2910 | "integrity": "sha512-ZMjC3ho+KXo0BfJb7JgtQ5IBuvnShdlACNkKkdsqBmYw3bPAaJfPeYUo6tLUaT5tG/Gkh7xkpBhKRQ9e7pyg9Q==", 2911 | "dev": true, 2912 | "engines": { 2913 | "node": ">=6" 2914 | } 2915 | }, 2916 | "node_modules/which": { 2917 | "version": "2.0.2", 2918 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 2919 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 2920 | "dev": true, 2921 | "dependencies": { 2922 | "isexe": "^2.0.0" 2923 | }, 2924 | "bin": { 2925 | "node-which": "bin/node-which" 2926 | }, 2927 | "engines": { 2928 | "node": ">= 8" 2929 | } 2930 | }, 2931 | "node_modules/wrap-ansi": { 2932 | "version": "7.0.0", 2933 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", 2934 | "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", 2935 | "dev": true, 2936 | "dependencies": { 2937 | "ansi-styles": "^4.0.0", 2938 | "string-width": "^4.1.0", 2939 | "strip-ansi": "^6.0.0" 2940 | }, 2941 | "engines": { 2942 | "node": ">=10" 2943 | }, 2944 | "funding": { 2945 | "url": "https://github.com/chalk/wrap-ansi?sponsor=1" 2946 | } 2947 | }, 2948 | "node_modules/wrap-ansi/node_modules/ansi-regex": { 2949 | "version": "5.0.1", 2950 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 2951 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 2952 | "dev": true, 2953 | "engines": { 2954 | "node": ">=8" 2955 | } 2956 | }, 2957 | "node_modules/wrap-ansi/node_modules/ansi-styles": { 2958 | "version": "4.3.0", 2959 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 2960 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 2961 | "dev": true, 2962 | "dependencies": { 2963 | "color-convert": "^2.0.1" 2964 | }, 2965 | "engines": { 2966 | "node": ">=8" 2967 | }, 2968 | "funding": { 2969 | "url": "https://github.com/chalk/ansi-styles?sponsor=1" 2970 | } 2971 | }, 2972 | "node_modules/wrap-ansi/node_modules/color-convert": { 2973 | "version": "2.0.1", 2974 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 2975 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 2976 | "dev": true, 2977 | "dependencies": { 2978 | "color-name": "~1.1.4" 2979 | }, 2980 | "engines": { 2981 | "node": ">=7.0.0" 2982 | } 2983 | }, 2984 | "node_modules/wrap-ansi/node_modules/color-name": { 2985 | "version": "1.1.4", 2986 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 2987 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 2988 | "dev": true 2989 | }, 2990 | "node_modules/wrap-ansi/node_modules/emoji-regex": { 2991 | "version": "8.0.0", 2992 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 2993 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 2994 | "dev": true 2995 | }, 2996 | "node_modules/wrap-ansi/node_modules/is-fullwidth-code-point": { 2997 | "version": "3.0.0", 2998 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 2999 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 3000 | "dev": true, 3001 | "engines": { 3002 | "node": ">=8" 3003 | } 3004 | }, 3005 | "node_modules/wrap-ansi/node_modules/string-width": { 3006 | "version": "4.2.3", 3007 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 3008 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 3009 | "dev": true, 3010 | "dependencies": { 3011 | "emoji-regex": "^8.0.0", 3012 | "is-fullwidth-code-point": "^3.0.0", 3013 | "strip-ansi": "^6.0.1" 3014 | }, 3015 | "engines": { 3016 | "node": ">=8" 3017 | } 3018 | }, 3019 | "node_modules/wrap-ansi/node_modules/strip-ansi": { 3020 | "version": "6.0.1", 3021 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 3022 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 3023 | "dev": true, 3024 | "dependencies": { 3025 | "ansi-regex": "^5.0.1" 3026 | }, 3027 | "engines": { 3028 | "node": ">=8" 3029 | } 3030 | }, 3031 | "node_modules/wrappy": { 3032 | "version": "1.0.2", 3033 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 3034 | "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", 3035 | "dev": true 3036 | }, 3037 | "node_modules/write-file-atomic": { 3038 | "version": "5.0.0", 3039 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-5.0.0.tgz", 3040 | "integrity": "sha512-R7NYMnHSlV42K54lwY9lvW6MnSm1HSJqZL3xiSgi9E7//FYaI74r2G0rd+/X6VAMkHEdzxQaU5HUOXWUz5kA/w==", 3041 | "dev": true, 3042 | "dependencies": { 3043 | "imurmurhash": "^0.1.4", 3044 | "signal-exit": "^3.0.7" 3045 | }, 3046 | "engines": { 3047 | "node": "^14.17.0 || ^16.13.0 || >=18.0.0" 3048 | } 3049 | }, 3050 | "node_modules/y18n": { 3051 | "version": "5.0.8", 3052 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", 3053 | "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", 3054 | "dev": true, 3055 | "engines": { 3056 | "node": ">=10" 3057 | } 3058 | }, 3059 | "node_modules/yallist": { 3060 | "version": "4.0.0", 3061 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", 3062 | "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", 3063 | "dev": true 3064 | }, 3065 | "node_modules/yargs": { 3066 | "version": "17.6.2", 3067 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.6.2.tgz", 3068 | "integrity": "sha512-1/9UrdHjDZc0eOU0HxOHoS78C69UD3JRMvzlJ7S79S2nTaWRA/whGCTV8o9e/N/1Va9YIV7Q4sOxD8VV4pCWOw==", 3069 | "dev": true, 3070 | "dependencies": { 3071 | "cliui": "^8.0.1", 3072 | "escalade": "^3.1.1", 3073 | "get-caller-file": "^2.0.5", 3074 | "require-directory": "^2.1.1", 3075 | "string-width": "^4.2.3", 3076 | "y18n": "^5.0.5", 3077 | "yargs-parser": "^21.1.1" 3078 | }, 3079 | "engines": { 3080 | "node": ">=12" 3081 | } 3082 | }, 3083 | "node_modules/yargs-parser": { 3084 | "version": "20.2.9", 3085 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz", 3086 | "integrity": "sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==", 3087 | "dev": true, 3088 | "engines": { 3089 | "node": ">=10" 3090 | } 3091 | }, 3092 | "node_modules/yargs/node_modules/ansi-regex": { 3093 | "version": "5.0.1", 3094 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 3095 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 3096 | "dev": true, 3097 | "engines": { 3098 | "node": ">=8" 3099 | } 3100 | }, 3101 | "node_modules/yargs/node_modules/emoji-regex": { 3102 | "version": "8.0.0", 3103 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 3104 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 3105 | "dev": true 3106 | }, 3107 | "node_modules/yargs/node_modules/is-fullwidth-code-point": { 3108 | "version": "3.0.0", 3109 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 3110 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 3111 | "dev": true, 3112 | "engines": { 3113 | "node": ">=8" 3114 | } 3115 | }, 3116 | "node_modules/yargs/node_modules/string-width": { 3117 | "version": "4.2.3", 3118 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 3119 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 3120 | "dev": true, 3121 | "dependencies": { 3122 | "emoji-regex": "^8.0.0", 3123 | "is-fullwidth-code-point": "^3.0.0", 3124 | "strip-ansi": "^6.0.1" 3125 | }, 3126 | "engines": { 3127 | "node": ">=8" 3128 | } 3129 | }, 3130 | "node_modules/yargs/node_modules/strip-ansi": { 3131 | "version": "6.0.1", 3132 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 3133 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 3134 | "dev": true, 3135 | "dependencies": { 3136 | "ansi-regex": "^5.0.1" 3137 | }, 3138 | "engines": { 3139 | "node": ">=8" 3140 | } 3141 | }, 3142 | "node_modules/yargs/node_modules/yargs-parser": { 3143 | "version": "21.1.1", 3144 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", 3145 | "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", 3146 | "dev": true, 3147 | "engines": { 3148 | "node": ">=12" 3149 | } 3150 | }, 3151 | "node_modules/yocto-queue": { 3152 | "version": "0.1.0", 3153 | "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", 3154 | "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", 3155 | "dev": true, 3156 | "engines": { 3157 | "node": ">=10" 3158 | }, 3159 | "funding": { 3160 | "url": "https://github.com/sponsors/sindresorhus" 3161 | } 3162 | } 3163 | } 3164 | } 3165 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "redis-bucket", 3 | "version": "2.0.0", 4 | "description": "A Redis-backed leaky-bucket rate limiter", 5 | "keywords": [ 6 | "redis", 7 | "leaky-bucket", 8 | "rate", 9 | "limit", 10 | "capacity" 11 | ], 12 | "repository": { 13 | "type": "git", 14 | "url": "https://github.com/plsmphnx/redis-bucket.git" 15 | }, 16 | "author": "Microsoft", 17 | "license": "MIT", 18 | "type": "module", 19 | "main": "index.js", 20 | "typings": "index.d.ts", 21 | "files": [ 22 | "index.js", 23 | "index.d.ts" 24 | ], 25 | "devDependencies": { 26 | "@princjef/tslint-config": "^2.0.2", 27 | "@types/node": "^18.11.11", 28 | "ava": "^5.1.0", 29 | "c8": "^7.12.0", 30 | "prettier": "^2.8.0", 31 | "redis": "^4.5.1", 32 | "terser": "^5.16.1", 33 | "tslint": "^5.20.1", 34 | "typescript": "^4.9.3" 35 | }, 36 | "scripts": { 37 | "lint": "tslint --project tsconfig.json --fix", 38 | "lint:verify": "tslint --project tsconfig.json", 39 | "format": "prettier --write *", 40 | "format:verify": "prettier --check *", 41 | "prebuild": "npm run lint && npm run format", 42 | "prebuild:verify": "npm run lint:verify && npm run format:verify", 43 | "build": "tsc && node embed", 44 | "build:verify": "tsc && node embed minify && npm run minify", 45 | "pretest": "npm run build", 46 | "pretest:verify": "npm run build:verify", 47 | "test": "c8 ava", 48 | "test:verify": "c8 ava", 49 | "minify": "terser index.js -o index.js --config-file terser.json", 50 | "prepublishOnly": "npm run test:verify" 51 | }, 52 | "c8": { 53 | "100": true, 54 | "exclude": [ 55 | "*.spec.*" 56 | ] 57 | } 58 | } 59 | -------------------------------------------------------------------------------- /readme.md: -------------------------------------------------------------------------------- 1 | # redis-bucket 2 | 3 | [![build status](https://github.com/plsmphnx/redis-bucket/workflows/build/badge.svg)](https://github.com/plsmphnx/redis-bucket/actions?query=workflow%3Abuild+branch%3Amaster) 4 | [![codecov](https://codecov.io/gh/plsmphnx/redis-bucket/branch/master/graph/badge.svg)](https://codecov.io/gh/plsmphnx/redis-bucket) 5 | [![npm version](https://img.shields.io/npm/v/redis-bucket.svg)](https://npmjs.org/package/redis-bucket) 6 | 7 | A [Redis](https://redis.io/)-backed rate limiter, based on the 8 | [leaky-bucket algorithm](https://en.wikipedia.org/wiki/Leaky_bucket#As_a_meter). 9 | Implemented using a purely EVAL-based solution, which provides the following 10 | advantages: 11 | 12 | - It is optimized for cases where multiple instances of a service share common 13 | rate-limiting metrics, since the counts are shared via a single store. 14 | - It supports multiple simultaneous metrics, allowing for tiered rates. 15 | - It works with hosted Redis solutions that may not support custom modules. 16 | 17 | ## Requirements 18 | 19 | - _Development_ - [Node.js](https://nodejs.org/) and a running Redis instance 20 | for testing. 21 | - _Runtime_ - A Redis client supporting EVAL; EVALSHA is recommended but not 22 | required. 23 | 24 | ## Example 25 | 26 | ```ts 27 | import * as express from 'express'; 28 | import * as redis from 'redis'; 29 | import * as limiter from 'redis-bucket'; 30 | 31 | // Create a Redis client with appropriate configuration and error handling 32 | const client = redis.createClient({}); 33 | client.on('error', () => {}); 34 | 35 | // Create the limiter 36 | const limit = limiter.create({ 37 | capacity: { window: 60, min: 10, max: 20 }, // 10-20 calls per minute 38 | backoff: x => 2 ** x, // Exponential backoff 39 | async eval(script: string, keys: string[], argv: unknown[]) { 40 | return client.eval(script, { keys, arguments: argv.map(String) }); 41 | }, 42 | async evalsha(sha: string, keys: string[], argv: unknown[]) { 43 | return client.evalSha(sha, { keys, arguments: argv.map(String) }); 44 | }, 45 | }); 46 | 47 | // Simple server, expects a "user" query parameter to identify callers 48 | const app = express(); 49 | app.get('/', async (req, res) => { 50 | // Scope rate-limiting to a given user 51 | const result = await limit(req.query.user); 52 | if (result.allow) { 53 | // Accept this call 54 | res.sendStatus(200); 55 | } else { 56 | // Reject this call with "Too Many Requests" 57 | res.set('Retry-After', result.wait); 58 | res.sendStatus(429); 59 | } 60 | }); 61 | app.listen(8080); 62 | ``` 63 | 64 | ## API 65 | 66 | ### create(config) 67 | 68 | Creates a [test function](#testkey-cost) to perform rate limiting against a 69 | given set of metrics. Takes a configuration object containing the following 70 | options: 71 | 72 | - `eval` - A callback to execute an EVAL call on Redis. 73 | - `evalsha` _(default none)_ - A callback to execute an EVALSHA call on Redis. 74 | - `prefix` _(default none)_ - A string prefix to apply to all Redis keys used 75 | by this instance. 76 | - `backoff` _(default linear)_ - The backoff scaling function used for 77 | retries. 78 | - `capacity` - A capacity metric (or array thereof) to limit by (see 79 | [below](#capacity-limits)). 80 | - `rate` - A rate metric (or array thereof) to limit by (see 81 | [below](#rate-limits)). 82 | 83 | ### _test_(key, [cost]) 84 | 85 | Tests whether the given action should be allowed according to the rate limits. 86 | Returns a [`Result`](#result) object. Takes the following arguments: 87 | 88 | - `key` - A string specifying the instance to be tested. Limits will only be 89 | applied to a given key against itself. 90 | - `cost` _(default 1)_ - The capacity cost of this action (see 91 | [below](#specifying-limits)). 92 | 93 | ### Result 94 | 95 | An object representing the result of a test. Contains the following parameters: 96 | 97 | - `allow` - Whether or not this action should be allowed according to the rate 98 | limits. 99 | - `free` - The current remaining capacity before actions will be rejected; 0 100 | if allow is false. 101 | - `wait` - How long the caller should wait before trying again, in seconds; 0 102 | if allow is true. 103 | 104 | ## Specifying Limits 105 | 106 | The allowable limits for a given instance of the rate-limiter can be specified 107 | in two ways. Note that 'capacity' is an abstract value; typically it represents 108 | a number of actions, but it can also indicate an overall net 'cost' of actions. 109 | 110 | ### Capacity Limits 111 | 112 | Capacity limits are specified using three values: 113 | 114 | - `window` - The time window over which these limits are considered, in 115 | seconds. 116 | - `min` - The minimum capacity that is guaranteed over this time window, 117 | assuming a perfectly uniform call pattern (see note below). 118 | - `max` - The maximum capacity that the system can handle over this time 119 | window. This value is absolute; callers will be limited in such a way to 120 | enforce this. It must be sufficiently greater than the minimum capacity to 121 | cover the highest cost the test function will be called with. 122 | 123 | **Note**: If the maximum capacity is set as low as possible (in other words, 124 | just greater than the minimum capacity), the caller must request capacity in a 125 | perfectly uniform manner in order to receive the minimum capacity. This 126 | restriction becomes increasingly loose as the maximum capacity increases, 127 | disappearing when the maximum capacity is twice the minimum capacity (at which 128 | point the minimum capacity becomes guaranteed independent of the call pattern). 129 | 130 | ### Rate Limits 131 | 132 | Rate limits are specified using two values: 133 | 134 | - `flow` - The rate at which capacity becomes available, per second. In a 135 | fully-stressed system, calls will be limited to exactly this rate. 136 | - `burst` - The amount of leeway in capacity the system can support. This is 137 | the amount of capacity that can be utilized before rate-limiting is applied. 138 | It must be at least equal to the highest cost the test function will be 139 | called with. 140 | 141 | ## Contributing 142 | 143 | This project has adopted the 144 | [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/). 145 | For more information see the 146 | [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or 147 | contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any 148 | additional questions or comments. 149 | -------------------------------------------------------------------------------- /terser.json: -------------------------------------------------------------------------------- 1 | { 2 | "ecma": 2020, 3 | "module": true, 4 | "compress": { 5 | "unsafe": true 6 | }, 7 | "output": { 8 | "comments": false 9 | }, 10 | "sourceMap": { 11 | "root": "." 12 | } 13 | } 14 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "es2020", 4 | "module": "node16", 5 | "sourceMap": true, 6 | "declaration": true, 7 | "strict": true, 8 | "noImplicitReturns": true, 9 | "noUnusedLocals": true, 10 | "suppressImplicitAnyIndexErrors": true, 11 | "forceConsistentCasingInFileNames": true, 12 | "newLine": "lf", 13 | "baseUrl": "." 14 | }, 15 | "include": ["index.ts", "index.spec.ts"] 16 | } 17 | -------------------------------------------------------------------------------- /tslint.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": "@princjef/tslint-config", 3 | "rules": { 4 | "file-header": [ 5 | true, 6 | "^!\\s*\\* Copyright \\(c\\) Microsoft Corporation\\.\r?\n \\* Licensed under the MIT License\\.\\s*$", 7 | "Copyright (c) Microsoft Corporation.\nLicensed under the MIT License." 8 | ], 9 | "no-namespace": true, 10 | "no-with-statement": false, 11 | "triple-equals": false 12 | } 13 | } 14 | --------------------------------------------------------------------------------