├── .eslintignore ├── .eslintrc.yml ├── .github └── workflows │ └── node.js.yml ├── .gitignore ├── .npmignore ├── README.md ├── index.js ├── lib ├── acm.js ├── middleware.js └── utils.js ├── package-lock.json ├── package.json └── test ├── askTest.js ├── checkTest.js └── middlewareTest.js /.eslintignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | coverage -------------------------------------------------------------------------------- /.eslintrc.yml: -------------------------------------------------------------------------------- 1 | env: 2 | es6: true 3 | node: true 4 | mocha: true 5 | extends: 'eslint:recommended' 6 | rules: 7 | accessor-pairs: error 8 | array-bracket-newline: error 9 | array-bracket-spacing: 10 | - error 11 | - never 12 | array-callback-return: error 13 | array-element-newline: 'off' 14 | arrow-body-style: error 15 | arrow-parens: 'off' 16 | arrow-spacing: 17 | - error 18 | - after: true 19 | before: true 20 | block-scoped-var: error 21 | block-spacing: 22 | - error 23 | - always 24 | brace-style: 25 | - error 26 | - 1tbs 27 | - allowSingleLine: true 28 | callback-return: error 29 | camelcase: error 30 | capitalized-comments: error 31 | class-methods-use-this: error 32 | comma-dangle: 'off' 33 | comma-spacing: 34 | - error 35 | - after: true 36 | before: false 37 | comma-style: 38 | - error 39 | - last 40 | complexity: error 41 | computed-property-spacing: 42 | - error 43 | - never 44 | consistent-return: 'off' 45 | consistent-this: error 46 | curly: 'off' 47 | default-case: error 48 | dot-location: 49 | - error 50 | - property 51 | dot-notation: 52 | - error 53 | - allowKeywords: true 54 | eol-last: 'off' 55 | eqeqeq: 'off' 56 | for-direction: error 57 | func-call-spacing: error 58 | func-name-matching: error 59 | func-names: 60 | - error 61 | - never 62 | func-style: 63 | - error 64 | - expression 65 | function-paren-newline: 'off' 66 | generator-star-spacing: error 67 | getter-return: error 68 | global-require: error 69 | guard-for-in: error 70 | handle-callback-err: error 71 | id-blacklist: error 72 | id-length: 'off' 73 | id-match: error 74 | implicit-arrow-linebreak: 'off' 75 | indent: 'off' 76 | indent-legacy: 'off' 77 | init-declarations: error 78 | jsx-quotes: error 79 | key-spacing: error 80 | keyword-spacing: 81 | - error 82 | - after: true 83 | before: true 84 | line-comment-position: error 85 | linebreak-style: 86 | - error 87 | - unix 88 | lines-around-comment: error 89 | lines-around-directive: error 90 | lines-between-class-members: error 91 | max-depth: error 92 | max-len: 'off' 93 | max-lines: 'off' 94 | max-nested-callbacks: error 95 | max-params: 'off' 96 | max-statements: 'off' 97 | max-statements-per-line: error 98 | multiline-comment-style: error 99 | new-cap: error 100 | new-parens: error 101 | newline-after-var: 'off' 102 | newline-before-return: 'off' 103 | newline-per-chained-call: 'off' 104 | no-alert: error 105 | no-array-constructor: error 106 | no-await-in-loop: error 107 | no-bitwise: error 108 | no-buffer-constructor: error 109 | no-caller: error 110 | no-catch-shadow: error 111 | no-confusing-arrow: error 112 | no-continue: error 113 | no-div-regex: error 114 | no-duplicate-imports: error 115 | no-else-return: error 116 | no-empty-function: error 117 | no-eq-null: error 118 | no-eval: error 119 | no-extend-native: error 120 | no-extra-bind: error 121 | no-extra-label: error 122 | no-extra-parens: error 123 | no-floating-decimal: error 124 | no-implicit-coercion: error 125 | no-implicit-globals: error 126 | no-implied-eval: error 127 | no-inline-comments: error 128 | no-invalid-this: 'off' 129 | no-iterator: error 130 | no-label-var: error 131 | no-labels: error 132 | no-lone-blocks: error 133 | no-lonely-if: error 134 | no-loop-func: error 135 | no-magic-numbers: 'off' 136 | no-mixed-operators: error 137 | no-mixed-requires: error 138 | no-multi-assign: error 139 | no-multi-spaces: error 140 | no-multi-str: error 141 | no-multiple-empty-lines: error 142 | no-native-reassign: error 143 | no-negated-condition: error 144 | no-negated-in-lhs: error 145 | no-nested-ternary: error 146 | no-new: error 147 | no-new-func: error 148 | no-new-object: error 149 | no-new-require: error 150 | no-new-wrappers: error 151 | no-octal-escape: error 152 | no-param-reassign: error 153 | no-path-concat: error 154 | no-plusplus: error 155 | no-process-env: error 156 | no-process-exit: error 157 | no-proto: error 158 | no-prototype-builtins: error 159 | no-restricted-globals: error 160 | no-restricted-imports: error 161 | no-restricted-modules: error 162 | no-restricted-properties: error 163 | no-restricted-syntax: error 164 | no-return-assign: 'off' 165 | no-return-await: error 166 | no-script-url: error 167 | no-self-compare: error 168 | no-sequences: error 169 | no-shadow: 'off' 170 | no-shadow-restricted-names: error 171 | no-spaced-func: error 172 | no-sync: error 173 | no-tabs: error 174 | no-template-curly-in-string: error 175 | no-ternary: 'off' 176 | no-throw-literal: error 177 | no-trailing-spaces: error 178 | no-undef-init: error 179 | no-undefined: 'off' 180 | no-underscore-dangle: 'off' 181 | no-unmodified-loop-condition: error 182 | no-unneeded-ternary: error 183 | no-unused-expressions: error 184 | no-use-before-define: error 185 | no-useless-call: error 186 | no-useless-computed-key: error 187 | no-useless-concat: error 188 | no-useless-constructor: 'off' 189 | no-useless-rename: error 190 | no-useless-return: error 191 | no-var: error 192 | no-void: error 193 | no-warning-comments: error 194 | no-whitespace-before-property: error 195 | no-with: error 196 | nonblock-statement-body-position: error 197 | object-curly-newline: 'off' 198 | object-curly-spacing: 199 | - error 200 | - always 201 | object-shorthand: error 202 | one-var: 'off' 203 | one-var-declaration-per-line: error 204 | operator-assignment: error 205 | operator-linebreak: 206 | - error 207 | - before 208 | padded-blocks: 'off' 209 | padding-line-between-statements: error 210 | prefer-arrow-callback: error 211 | prefer-const: 'off' 212 | prefer-destructuring: 'off' 213 | prefer-numeric-literals: error 214 | prefer-promise-reject-errors: error 215 | prefer-reflect: 'off' 216 | prefer-rest-params: error 217 | prefer-spread: error 218 | prefer-template: error 219 | quote-props: 'off' 220 | quotes: 'off' 221 | radix: error 222 | require-await: error 223 | require-jsdoc: error 224 | rest-spread-spacing: 225 | - error 226 | - never 227 | semi: 'off' 228 | semi-spacing: 229 | - error 230 | - after: false 231 | before: false 232 | semi-style: 233 | - error 234 | - last 235 | sort-imports: error 236 | sort-keys: 'off' 237 | sort-vars: error 238 | space-before-blocks: error 239 | space-before-function-paren: 'off' 240 | space-in-parens: 241 | - error 242 | - never 243 | space-infix-ops: error 244 | space-unary-ops: error 245 | spaced-comment: error 246 | strict: 247 | - error 248 | - never 249 | switch-colon-spacing: error 250 | symbol-description: error 251 | template-curly-spacing: 252 | - error 253 | - never 254 | template-tag-spacing: error 255 | unicode-bom: 256 | - error 257 | - never 258 | valid-jsdoc: error 259 | vars-on-top: error 260 | wrap-iife: error 261 | wrap-regex: error 262 | yield-star-spacing: error 263 | yoda: 264 | - error 265 | - never 266 | -------------------------------------------------------------------------------- /.github/workflows/node.js.yml: -------------------------------------------------------------------------------- 1 | name: CI 2 | 3 | on: 4 | push: 5 | branches: [master] 6 | pull_request: 7 | branches: [master] 8 | 9 | jobs: 10 | build: 11 | runs-on: ubuntu-latest 12 | 13 | strategy: 14 | matrix: 15 | node-version: [16.x, 18.x, 20.x, 22.x] 16 | 17 | steps: 18 | - uses: actions/checkout@v2 19 | - name: Use Node.js ${{ matrix.node-version }} 20 | uses: actions/setup-node@v2 21 | with: 22 | node-version: ${{ matrix.node-version }} 23 | cache: "npm" 24 | - run: npm ci 25 | - run: npm run lint 26 | - run: npm test 27 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | .nyc_output 3 | coverage -------------------------------------------------------------------------------- /.npmignore: -------------------------------------------------------------------------------- 1 | test 2 | coverage 3 | .nyc_output 4 | .vscode 5 | .travis.yml -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # visa.js 2 | 3 | [![Build Status](https://github.com/ZenitechSoftware/visa-js/actions/workflows/node.js.yml/badge.svg)](https://github.com/ZenitechSoftware/visa-js/actions/workflows/node.js.yml) 4 | [![Known Vulnerabilities](https://snyk.io/test/github/ZenitechSoftware/visa-js/badge.svg)]([https://www.npmjs.com/package/visa-js](https://snyk.io/test/github/ZenitechSoftware/visa-js)) 5 | [![Maintainability](https://api.codeclimate.com/v1/badges/37c6655235e6edad1c58/maintainability)](https://codeclimate.com/github/ZenitechSoftware/visa-js/maintainability) 6 | [![Test coverage](https://api.codeclimate.com/v1/badges/37c6655235e6edad1c58/test_coverage)](https://codeclimate.com/github/ZenitechSoftware/visa-js/test_coverage) 7 | 8 | Simple and flexible Access Control Manager based on [Attribute Based Access Control (ABAC) paradigm](https://nvlpubs.nist.gov/nistpubs/specialpublications/NIST.sp.800-162.pdf) that works well with [Express](https://www.npmjs.com/package/express) and [Passport](https://www.npmjs.com/package/passport) or independently. 9 | 10 | _Get a travel Visa, take your Passport and grab next Express train._ 11 | 12 | ## Quick start 13 | 14 | ### Installation 15 | 16 | ``` 17 | npm install visa-js -S 18 | ``` 19 | 20 | ### Example 21 | 22 | ```js 23 | const visa = require("visa-js"); 24 | 25 | visa.policy({ 26 | objects: { 27 | account: { 28 | mapRefsToObjects: (refs) => Account.find({ _id: { $in: refs } }), 29 | operations: { 30 | open: (subject) => subject.role == "teller", 31 | read: (subject, account) => account && subject.id === account.ownerId, 32 | }, 33 | }, 34 | }, 35 | }); 36 | 37 | app.post( 38 | "/api/account", 39 | passport.authenticate("local"), 40 | visa.authorize(visa.user.can.open.account), 41 | (req, res) => { 42 | // authorized to open account 43 | } 44 | ); 45 | 46 | app.get( 47 | "/api/account/:id", 48 | passport.authenticate("local"), 49 | visa.authorize(visa.user.can.read.account), 50 | (req, res) => { 51 | // authorized to read specific account 52 | } 53 | ); 54 | ``` 55 | 56 | ## Introduction 57 | 58 | When talking about WEB app security, Node.js has great [Passport](https://www.npmjs.com/package/passport) module that takes care of user authentication - process of actually confirming user identity. It is simple and flexible and already supports multiple strategies. In case strategy is missing it is easy to add one. 59 | 60 | Visa.js tries to follow same principles for another security aspect - authorization - process to actually confirming identified user has access rights to resources provided by WEB app. Visa.js implements [Attribute Based Access Control (ABAC) paradigm](https://nvlpubs.nist.gov/nistpubs/specialpublications/NIST.sp.800-162.pdf) that is a flexible way to control access based on policies. Policy is a set of rules that evaluate attributes of identified user (subject), accessed resource (object) and environment (context) to grant/deny access. There are few initiatives such as [node-abac](https://www.npmjs.com/package/node-abac) or [abac](https://www.npmjs.com/package/abac), however, they only cover some very specific cases. 61 | 62 | [Role Based Access Control (RBAC)](https://csrc.nist.gov/CSRC/media/Publications/conference-paper/1992/10/13/role-based-access-controls/documents/ferraiolo-kuhn-92.pdf) is another popular paradigm. There are popular Node.js modules based on RBAC such as [acl](https://www.npmjs.com/package/acl) or [accesscontrol](https://www.npmjs.com/package/accesscontrol). However, RBAC is user-centric and do not take into account additional parameters such as resource information, relationship between the user (the requesting entity) and the resource, and dynamic information e.g. time of the day or user IP. RBAC is in some ways special case of ABAC in terms of the attributes used. RBAC works on the attribute of “role”. The key difference with ABAC is the concept of policies that express a complex Boolean rule set that can evaluate many different attributes. 63 | 64 | ## Definitions 65 | 66 | ### Subject 67 | 68 | A human user or non-person entity (NPE), such as other systems or bots. Assume that subject and user are synonymous. 69 | 70 | ### Object 71 | 72 | A system resource for which access is managed, such as files, API endpoints or processes. 73 | 74 | ### Operation 75 | 76 | The execution of a function at the request of a subject upon an object, such as read, write, edit, delete, copy, execute, or modify. 77 | 78 | ### Environment conditions 79 | 80 | Operational or situational context in which access requests occur. Environment characteristics are independent of subject or object, and may include the current time, day of the week, location of a user, or the current threat level. 81 | 82 | ### Policy 83 | 84 | Policy is the representation of rules or relationships that makes it possible to determine if a requested access should be allowed, given the values of the attributes of the subject, object, and possibly 85 | environment conditions. 86 | 87 | ### Access Control Mechanism (ACM) 88 | 89 | Assembles authorization information, which may include information about the object being protected, the subject requesting access, the policies governing access to the resource, and any contextual information needed to make a decision regarding access. 90 | 91 | ## Define policies 92 | 93 | Use `visa.policy()` function to define policies (function can be called multiple times for multiple policies): 94 | 95 | ```js 96 | visa.policy({ 97 | objects: { 98 | 'object1': { 99 | mapRefsToObjects: refs => // resolve object references to objects 100 | operations: { 101 | 'operation1': (subject, object, context) => // rule 102 | 'operation2': (subject, object, context) => // rule 103 | } 104 | }, 105 | 'object2': { 106 | // mapping and operations 107 | } 108 | } 109 | }); 110 | ``` 111 | 112 | - Policy contains rules to grant/deny access to perform operations on objects. 113 | - Rule is a function that evaluates user permission to perform operation on object. It receives subject, object (if provided) and context (if provided). Function code use attributes (properties) received to return `true` in case permission is granted and `false` in case denied (`Promise`, callback and sync return are supported). 114 | - `mapRefsToObjects` is a function that receives an array of references to objects and returns an array of objects (`Promise`, callback and sync return are supported). Each object should contain attributes (properties) required to evaluate rules. The function is called when access to object by reference is checked. 115 | 116 | ## Check permission 117 | 118 | Use `visa.check()` function to check permission. The outcome is a `Promise` that is resolved if access is granted and rejected with `visa.Unauthorized` error if access is denied. 119 | 120 | Check for `subject` permission to perform `operation1` on any instance of `object1`: 121 | 122 | ```js 123 | visa.check(subject).can.operation1.object1(); 124 | ``` 125 | 126 | Check for `subject` permission to perform `operation1` on specific `object1`: 127 | 128 | ```js 129 | visa.check(subject).can.operation1.object1({object: object1); 130 | ``` 131 | 132 | Check for `subject` permission to perform `operation1` on specific instances of `object1`: 133 | 134 | ```js 135 | visa.check(subject).can.operation1.object1({objects: [object1_1, object1_2]); 136 | ``` 137 | 138 | Check for `subject` permission to perform `operation1` on specific `object1` with reference `1` (to resolve reference to object `mapRefsToObjects` function must be defined for `object1` in policy): 139 | 140 | ```js 141 | visa.check(subject).can.operation1.object1({ ref: 1 }); 142 | ``` 143 | 144 | Check for `subject` permission to perform `operation1` on specific instances of `object1` with references `1` and `2` (to resolve references to objects `mapRefsToObjects` function must be defined for `object1` in policy): 145 | 146 | ```js 147 | visa.check(subject).can.operation1.object1({ refs: [1, 2] }); 148 | ``` 149 | 150 | Context can optionally be passed in case it is required to evaluate policy rules. For instance, Express `req` object is passed as context: 151 | 152 | ```js 153 | visa.check(subject).can.operation1.object1({ context: req }); 154 | ``` 155 | 156 | ## Ask if has permission 157 | 158 | Use `visa.ask()` function to ask if user has permission. The outcome is a `Promise` that is resolved to `true` if access is granted, `false` if access is denied. 159 | 160 | `visa.ask()` function follows the same structure as `visa.check()` function. For example, ask `subject` if he has permission to perform `operation1` on any instance of `object1`: 161 | 162 | ```js 163 | visa.ask(subject).can.operation1.object1(); 164 | ``` 165 | 166 | ## Check/Ask if does NOT have permission 167 | 168 | Use `can.not` structure in `visa.check()` or `visa.ask()` function to check/ask if user does NOT have permission: 169 | 170 | ```js 171 | visa.check(subject).can.not.operation1.object1(); 172 | ``` 173 | 174 | ```js 175 | visa.ask(subject).can.not.operation1.object1(); 176 | ``` 177 | 178 | ## Integration with Express and Passport 179 | 180 | Use `visa.authorize()` as an Express middleware. Middleware use subject as `req.user` property, therefore, `passport.authenticate()` middleware should be used before `visa.authorize()`: 181 | 182 | ```js 183 | app.get( 184 | "/api/object", 185 | passport.authenticate(/* strategy and options */), 186 | visa.authorize(visa.user.can.operation1.object1), 187 | (req, res) => { 188 | // authorized to perform operation1 on any object1 189 | } 190 | ); 191 | ``` 192 | 193 | - `visa.authorize()` function takes permission check as an argument that starts with `visa.user` and follows the same structure as `visa.check()` function. 194 | - Express `req` object is passed to policy rule as `context` argument. 195 | 196 | `visa.authorize()` middleware also use `req.params.id` Express parameter if present to validate rule against object referenced by `req.params.id`: 197 | 198 | ```js 199 | app.get( 200 | "/api/object/:id", 201 | passport.authenticate(/* strategy and options */), 202 | visa.authorize(visa.user.can.operation1.object1), 203 | (req, res) => { 204 | // authorized to perform operation1 on specific object1 referenced by :id 205 | } 206 | ); 207 | ``` 208 | 209 | - `req.params.id` is passed as `refs` array item to `mapRefsToObjects` function defined in policy for `object`. 210 | - Express `req` object is passed to policy rule as `context` argument. 211 | 212 | `visa.authorize()` middleware also accepts second argument as function that gets Express `req` object as argument and should return reference to object: 213 | 214 | ```js 215 | app.get( 216 | "/api/object/:objectId", 217 | passport.authenticate(/* strategy and options */), 218 | visa.authorize( 219 | visa.user.can.operation1.object1, 220 | (req) => req.params.objectId 221 | ), 222 | (req, res) => { 223 | // authorized to perform operation1 on specific object1 referenced by :objectId parameter 224 | } 225 | ); 226 | ``` 227 | 228 | - `req.params.objectId` is passed as `refs` array item to `mapRefsToObjects` function defined in policy for `object`. 229 | - Express `req` object is passed to policy rule as `context` argument. 230 | 231 | Use `visa.unauthorizedErrorHandler` middleware to handle `visa.Unauthorized` error and to return HTTP status 401: 232 | 233 | ```js 234 | app.use(visa.unauthorizedErrorHandler); // client receives HTTP status 401 in case access is denied 235 | ``` 236 | 237 | - Make sure that `visa.unauthorizedErrorHandler` usage is declared AFTER all the routes where authorization is used. 238 | 239 | ## Example 240 | 241 | Let's define policy for sample bank app. 242 | 243 | ```js 244 | visa.policy({ 245 | objects: { 246 | account: { 247 | mapRefsToObjects: (refs) => Account.find({ _id: { $in: refs } }), 248 | operations: { 249 | open: (user) => user.role == "teller", 250 | read: (user, account) => account && user.id === account.ownerId, 251 | }, 252 | }, 253 | transaction: { 254 | mapRefsToObjects: (refs) => Transaction.find({ _id: { $in: refs } }), 255 | operations: { 256 | create: async (user, _, req) => { 257 | if (user.role === "manager") { 258 | return true; 259 | } 260 | const fromAccount = Account.find({ _id: req.body.fromAccountId }); 261 | return fromAccount.ownerId === user.id; 262 | }, 263 | revert: (user, transaction, req) => 264 | user.role === "cfo" && 265 | transaction.date > moment().subtract(1, "day") && 266 | req.ip === "10.0.0.99", 267 | }, 268 | }, 269 | }, 270 | }); 271 | ``` 272 | 273 | - any `account` can only be `open`ed by `teller` 274 | - specific `account` can only be `read` by owner 275 | - any `transaction` can only be `create`d by `manager` OR by owner of account money are transferred from 276 | - `transaction` that was created less than 1 day ago can only be `revert`ed by CFO from specific machine 277 | 278 | Let's secure the sample bank app API: 279 | 280 | ```js 281 | app.post( 282 | '/api/account', 283 | passport.authenticate('test', { session: false }), 284 | visa.authorize(visa.user.can.open.account), 285 | (req, res) => // authorized to open account 286 | ); 287 | app.get( 288 | '/api/account/:id', 289 | passport.authenticate('test', { session: false }), 290 | visa.authorize(visa.user.can.read.account), 291 | (req, res) => // authorized to read account referenced by :id parameter 292 | ); 293 | app.post( 294 | '/api/transaction', 295 | passport.authenticate('test', { session: false }), 296 | visa.authorize(visa.user.can.create.transaction), 297 | (req, res) => // authorized to create transaction 298 | ); 299 | app.delete( 300 | '/api/transaction/:id', 301 | passport.authenticate('test', { session: false }), 302 | visa.authorize(visa.user.can.revert.transaction), 303 | (req, res) => // authorized to revert transaction referenced by :id parameter 304 | ); 305 | ``` 306 | 307 | Let's check for permissions somewhere in the code of app: 308 | 309 | ```js 310 | visa.check(req.user).can.open.account() 311 | .then(() => { 312 | /* authorized to open account */ 313 | }) 314 | .catch(error => { 315 | if (error instanceof visa.Unauthorized) { 316 | /* unauthorized to open account */ 317 | } else { 318 | /* handle error */ 319 | } 320 | }; 321 | ``` 322 | 323 | ```js 324 | await visa.check(req.user).can.read.account({ ref: req.params.id }); 325 | ``` 326 | 327 | ```js 328 | await visa.check(req.user).can.create.transaction({ context: req }); 329 | ``` 330 | 331 | ```js 332 | await visa.check(req.user).can.revert.transaction({ 333 | ref: req.params.id, 334 | context: req, 335 | }); 336 | ``` 337 | 338 | Let's ask if user has permissions somewhere in the code of app: 339 | 340 | ```js 341 | visa.ask(req.user).can.open.account() 342 | .then(answer => { 343 | if (answer) { 344 | /* authorized to open account */ 345 | } else { 346 | /* unauthorized to open account */ 347 | } 348 | }) 349 | .catch(error => { 350 | /* handle error */ 351 | }; 352 | ``` 353 | 354 | ```js 355 | const answer = await visa 356 | .ask(req.user) 357 | .can.read.account({ ref: req.params.id }); 358 | ``` 359 | 360 | ```js 361 | const answer = await visa 362 | .ask(req.user) 363 | .can.create.transaction({ context: req }); 364 | ``` 365 | 366 | ```js 367 | const answer = await visa.ask(req.user).can.revert.transaction({ 368 | ref: req.params.id, 369 | context: req, 370 | }); 371 | ``` 372 | 373 | Please note that in order to avoid unhandled promise rejection error in Express routes when using `visa.check()` function, it is recommended to use middleware such as [express-async-errors](https://www.npmjs.com/package/express-async-errors) to support `Promise`. 374 | 375 | ## Multiple Access Control Mechanisms (ACMs) 376 | 377 | visa.js by default use single global Access Control Mechanism (ACM). In some cases multiple ACMs within same application might be useful. Use `visa.buildACM()` function to build new ACM. ACM instance returned by the function has same functions as `visa` module, except for: `visa.authorize()`, `visa.reset()`, `visa.Unauthorized` and `visa.unauthorizedErrorHandler` (these are not coupled with specific ACM instance). 378 | -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | const { build, Unauthorized } = require('./lib/acm'); 2 | const { authorize, unauthorizedErrorHandler } = require('./lib/middleware'); 3 | 4 | let acm = build(); 5 | 6 | Object.defineProperty(module.exports, 'user', { get: acm.check }); 7 | 8 | module.exports.Unauthorized = Unauthorized; 9 | module.exports.policy = acm.policy; 10 | module.exports.check = acm.check; 11 | module.exports.ask = acm.ask; 12 | module.exports.reset = () => acm = build(); 13 | 14 | module.exports.authorize = authorize; 15 | module.exports.unauthorizedErrorHandler = unauthorizedErrorHandler; 16 | 17 | module.exports.buildACM = build; 18 | -------------------------------------------------------------------------------- /lib/acm.js: -------------------------------------------------------------------------------- 1 | const async = require('async'); 2 | const { wrapToPromise } = require('./utils'); 3 | 4 | class Unauthorized extends Error { 5 | } 6 | 7 | class VisaError extends Error { 8 | constructor(message) { 9 | super(message); 10 | } 11 | } 12 | 13 | const evaluateRule = (objectName, operationName, rule, subject, object, context) => 14 | wrapToPromise(rule, subject, object, context) 15 | .catch(error => { 16 | error.message = `visa.js: object '${objectName}' operation '${operationName}': rule failed with error: ${error.message}`; 17 | throw error; 18 | }) 19 | .then(authorized => { 20 | if (!authorized) { 21 | throw new Unauthorized(); 22 | } 23 | }); 24 | 25 | const evaluateObjects = (objectName, operationName, objects, refs) => { 26 | if (!Array.isArray(objects)) { 27 | return Promise.reject(new VisaError(`visa.js: object '${objectName}' operation '${operationName}': 'mapRefsToObjects' function should return array of objects`)); 28 | } 29 | if (objects.length !== refs.length) { 30 | return Promise.reject(new Unauthorized()); 31 | } 32 | if (objects.findIndex(object => object === null || object === undefined) !== -1) { 33 | return Promise.reject(new Unauthorized()); 34 | } 35 | return Promise.resolve(objects); 36 | }; 37 | 38 | const evaluateMapRefsToObjects = (objectName, operationName, rule, subject, mapRefsToObjects, args) => { 39 | const refs = []; 40 | if (args.ref !== null && args.ref !== undefined) { 41 | refs.push(args.ref); 42 | } 43 | if (args.refs) { 44 | args.refs.forEach(ref => refs.push(ref)); 45 | } 46 | if (!refs.length) { 47 | return Promise.resolve(refs); 48 | } 49 | if (!mapRefsToObjects) { 50 | return Promise.reject(new VisaError(`visa.js: object '${objectName}' operation '${operationName}': object references (${refs}) can not be resolved because 'mapRefsToObjects' method is not provided`)); 51 | } 52 | return wrapToPromise(mapRefsToObjects, refs) 53 | .catch(error => { 54 | error.message = `visa.js: object '${objectName}' operation '${operationName}': object references (${refs}) can not be resolved because of 'mapRefsToObjects' error: ${error.message}`; 55 | throw error; 56 | }) 57 | .then(objects => evaluateObjects(objectName, operationName, objects, refs)); 58 | }; 59 | 60 | const evaluateRuleForObjects = (objectName, operationName, rule, subject, objects, context) => 61 | new Promise((resolve, reject) => async.eachLimit( 62 | objects, 63 | 5, 64 | (object, cb) => evaluateRule(objectName, operationName, rule, subject, object, context).then(cb).catch(cb), 65 | error => { 66 | if (error) { 67 | return reject(error); 68 | } 69 | resolve(objects.length === 1 ? objects[0] : objects); 70 | }) 71 | ); 72 | 73 | const evaluateRulesForObjects = (objectName, operationName, rule, subject, mapRefsToObjects, args) => 74 | evaluateMapRefsToObjects(objectName, operationName, rule, subject, mapRefsToObjects, args) 75 | .then(objects => { 76 | if (args.object) { 77 | objects.push(args.object); 78 | } 79 | if (args.objects) { 80 | args.objects.forEach(object => objects.push(object)); 81 | } 82 | return evaluateRuleForObjects(objectName, operationName, rule, subject, objects, args.context) 83 | }); 84 | 85 | const createEvaluationFunction = (rule, mapRefsToObjects, objectName, operationName) => function (args) { 86 | const result = args && (args.ref || args.refs || args.object || args.objects) 87 | ? evaluateRulesForObjects(objectName, operationName, rule, this.subject, mapRefsToObjects, args) 88 | : evaluateRule(objectName, operationName, rule, this.subject, undefined, args && args.context); 89 | if (this.isAsk) { 90 | return result 91 | .then(() => !this.isNot) 92 | .catch(error => { 93 | if (error instanceof Unauthorized) { 94 | return Boolean(this.isNot); 95 | } 96 | throw error; 97 | }); 98 | } 99 | if (this.isNot) { 100 | return result 101 | .then(() => false) 102 | .catch(error => { 103 | if (error instanceof Unauthorized) { 104 | return true; 105 | } 106 | throw error; 107 | }) 108 | .then(result => { 109 | if (!result) { 110 | throw new Unauthorized(); 111 | } 112 | }); 113 | } 114 | return result; 115 | }; 116 | 117 | const createPolicyFunction = (Can) => policy => { 118 | for (let objectName of Object.keys(policy.objects)) { 119 | for (let operationName of Object.keys(policy.objects[objectName].operations)) { 120 | if (!Can.prototype[operationName]) { 121 | class Operation { 122 | constructor(subject, isAsk, isNot) { 123 | this.subject = subject; 124 | this.isAsk = isAsk; 125 | this.isNot = isNot; 126 | } 127 | } 128 | Can.prototype[`_${operationName}Class`] = Operation; 129 | Can.prototype.__defineGetter__(operationName, function () { 130 | return new Operation(this.subject, this.isAsk, this.isNot); 131 | }); 132 | } 133 | Can.prototype[`_${operationName}Class`].prototype[objectName] = createEvaluationFunction( 134 | policy.objects[objectName].operations[operationName], 135 | policy.objects[objectName].mapRefsToObjects, 136 | objectName, 137 | operationName 138 | ); 139 | } 140 | } 141 | }; 142 | 143 | module.exports.Unauthorized = Unauthorized; 144 | 145 | module.exports.build = () => { 146 | let Can = function (subject, isAsk) { 147 | this.subject = subject; 148 | this.isAsk = isAsk; 149 | } 150 | Can.prototype.__defineGetter__('not', function () { 151 | this.isNot = true; 152 | return this; 153 | }); 154 | let Check = function (subject, isAsk) { 155 | this.subject = subject; 156 | this.isAsk = isAsk; 157 | } 158 | Check.prototype.__defineGetter__('can', function () { 159 | return new Can(this.subject, this.isAsk); 160 | }); 161 | return { 162 | policy: createPolicyFunction(Can), 163 | check: subject => new Check(subject, false), 164 | ask: subject => new Check(subject, true), 165 | }; 166 | }; 167 | -------------------------------------------------------------------------------- /lib/middleware.js: -------------------------------------------------------------------------------- 1 | const { Unauthorized } = require('./acm'); 2 | 3 | module.exports.authorize = (rule, getRef) => (req, res, next) => { 4 | const args = [{ context: req }]; 5 | if (getRef) { 6 | args[0].ref = getRef(req); 7 | } else if (req.params.id) { 8 | args[0].ref = req.params.id; 9 | } 10 | rule.apply({ subject: req.user }, args) 11 | .then(() => next()) 12 | .catch(next); 13 | }; 14 | 15 | module.exports.unauthorizedErrorHandler = (error, req, res, next) => { 16 | if (error instanceof Unauthorized) { 17 | return res.sendStatus(401); 18 | } 19 | next(error); 20 | }; -------------------------------------------------------------------------------- /lib/utils.js: -------------------------------------------------------------------------------- 1 | module.exports.wrapToPromise = (func, ...args) => new Promise((resolve, reject) => { 2 | const cb = (error, result) => { 3 | if (error) { 4 | return reject(error); 5 | } 6 | resolve(result); 7 | } 8 | let result = null; 9 | try { 10 | result = func(...args, cb); 11 | } catch (error) { 12 | return reject(error); 13 | } 14 | if (result !== undefined) { 15 | return resolve(result); 16 | } 17 | }); 18 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "visa-js", 3 | "version": "1.0.25", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "visa-js", 9 | "version": "1.0.25", 10 | "license": "MIT", 11 | "dependencies": { 12 | "async": "^3.2.6" 13 | }, 14 | "devDependencies": { 15 | "chai": "^4.2.0", 16 | "eslint": "^8.11.0", 17 | "express": "^4.18.2", 18 | "express-async-errors": "^3.1.1", 19 | "mocha": "^9.2.2", 20 | "nyc": "^15.1.0", 21 | "passport": "^0.6.0", 22 | "passport-strategy": "^1.0.0", 23 | "supertest": "^6.3.3" 24 | } 25 | }, 26 | "node_modules/@ampproject/remapping": { 27 | "version": "2.1.2", 28 | "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.1.2.tgz", 29 | "integrity": "sha512-hoyByceqwKirw7w3Z7gnIIZC3Wx3J484Y3L/cMpXFbr7d9ZQj2mODrirNzcJa+SM3UlpWXYvKV4RlRpFXlWgXg==", 30 | "dev": true, 31 | "dependencies": { 32 | "@jridgewell/trace-mapping": "^0.3.0" 33 | }, 34 | "engines": { 35 | "node": ">=6.0.0" 36 | } 37 | }, 38 | "node_modules/@babel/code-frame": { 39 | "version": "7.26.2", 40 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.26.2.tgz", 41 | "integrity": "sha512-RJlIHRueQgwWitWgF8OdFYGZX328Ax5BCemNGlqHfplnRT9ESi8JkFlvaVYbS+UubVY6dpv87Fs2u5M29iNFVQ==", 42 | "dev": true, 43 | "license": "MIT", 44 | "dependencies": { 45 | "@babel/helper-validator-identifier": "^7.25.9", 46 | "js-tokens": "^4.0.0", 47 | "picocolors": "^1.0.0" 48 | }, 49 | "engines": { 50 | "node": ">=6.9.0" 51 | } 52 | }, 53 | "node_modules/@babel/compat-data": { 54 | "version": "7.17.7", 55 | "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.17.7.tgz", 56 | "integrity": "sha512-p8pdE6j0a29TNGebNm7NzYZWB3xVZJBZ7XGs42uAKzQo8VQ3F0By/cQCtUEABwIqw5zo6WA4NbmxsfzADzMKnQ==", 57 | "dev": true, 58 | "engines": { 59 | "node": ">=6.9.0" 60 | } 61 | }, 62 | "node_modules/@babel/core": { 63 | "version": "7.17.7", 64 | "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.17.7.tgz", 65 | "integrity": "sha512-djHlEfFHnSnTAcPb7dATbiM5HxGOP98+3JLBZtjRb5I7RXrw7kFRoG2dXM8cm3H+o11A8IFH/uprmJpwFynRNQ==", 66 | "dev": true, 67 | "dependencies": { 68 | "@ampproject/remapping": "^2.1.0", 69 | "@babel/code-frame": "^7.16.7", 70 | "@babel/generator": "^7.17.7", 71 | "@babel/helper-compilation-targets": "^7.17.7", 72 | "@babel/helper-module-transforms": "^7.17.7", 73 | "@babel/helpers": "^7.17.7", 74 | "@babel/parser": "^7.17.7", 75 | "@babel/template": "^7.16.7", 76 | "@babel/traverse": "^7.17.3", 77 | "@babel/types": "^7.17.0", 78 | "convert-source-map": "^1.7.0", 79 | "debug": "^4.1.0", 80 | "gensync": "^1.0.0-beta.2", 81 | "json5": "^2.1.2", 82 | "semver": "^6.3.0" 83 | }, 84 | "engines": { 85 | "node": ">=6.9.0" 86 | }, 87 | "funding": { 88 | "type": "opencollective", 89 | "url": "https://opencollective.com/babel" 90 | } 91 | }, 92 | "node_modules/@babel/generator": { 93 | "version": "7.23.6", 94 | "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.23.6.tgz", 95 | "integrity": "sha512-qrSfCYxYQB5owCmGLbl8XRpX1ytXlpueOb0N0UmQwA073KZxejgQTzAmJezxvpwQD9uGtK2shHdi55QT+MbjIw==", 96 | "dev": true, 97 | "dependencies": { 98 | "@babel/types": "^7.23.6", 99 | "@jridgewell/gen-mapping": "^0.3.2", 100 | "@jridgewell/trace-mapping": "^0.3.17", 101 | "jsesc": "^2.5.1" 102 | }, 103 | "engines": { 104 | "node": ">=6.9.0" 105 | } 106 | }, 107 | "node_modules/@babel/helper-compilation-targets": { 108 | "version": "7.17.7", 109 | "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.17.7.tgz", 110 | "integrity": "sha512-UFzlz2jjd8kroj0hmCFV5zr+tQPi1dpC2cRsDV/3IEW8bJfCPrPpmcSN6ZS8RqIq4LXcmpipCQFPddyFA5Yc7w==", 111 | "dev": true, 112 | "dependencies": { 113 | "@babel/compat-data": "^7.17.7", 114 | "@babel/helper-validator-option": "^7.16.7", 115 | "browserslist": "^4.17.5", 116 | "semver": "^6.3.0" 117 | }, 118 | "engines": { 119 | "node": ">=6.9.0" 120 | }, 121 | "peerDependencies": { 122 | "@babel/core": "^7.0.0" 123 | } 124 | }, 125 | "node_modules/@babel/helper-environment-visitor": { 126 | "version": "7.22.20", 127 | "resolved": "https://registry.npmjs.org/@babel/helper-environment-visitor/-/helper-environment-visitor-7.22.20.tgz", 128 | "integrity": "sha512-zfedSIzFhat/gFhWfHtgWvlec0nqB9YEIVrpuwjruLlXfUSnA8cJB0miHKwqDnQ7d32aKo2xt88/xZptwxbfhA==", 129 | "dev": true, 130 | "engines": { 131 | "node": ">=6.9.0" 132 | } 133 | }, 134 | "node_modules/@babel/helper-function-name": { 135 | "version": "7.23.0", 136 | "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.23.0.tgz", 137 | "integrity": "sha512-OErEqsrxjZTJciZ4Oo+eoZqeW9UIiOcuYKRJA4ZAgV9myA+pOXhhmpfNCKjEH/auVfEYVFJ6y1Tc4r0eIApqiw==", 138 | "dev": true, 139 | "dependencies": { 140 | "@babel/template": "^7.22.15", 141 | "@babel/types": "^7.23.0" 142 | }, 143 | "engines": { 144 | "node": ">=6.9.0" 145 | } 146 | }, 147 | "node_modules/@babel/helper-hoist-variables": { 148 | "version": "7.22.5", 149 | "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.22.5.tgz", 150 | "integrity": "sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw==", 151 | "dev": true, 152 | "dependencies": { 153 | "@babel/types": "^7.22.5" 154 | }, 155 | "engines": { 156 | "node": ">=6.9.0" 157 | } 158 | }, 159 | "node_modules/@babel/helper-module-imports": { 160 | "version": "7.16.7", 161 | "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.16.7.tgz", 162 | "integrity": "sha512-LVtS6TqjJHFc+nYeITRo6VLXve70xmq7wPhWTqDJusJEgGmkAACWwMiTNrvfoQo6hEhFwAIixNkvB0jPXDL8Wg==", 163 | "dev": true, 164 | "dependencies": { 165 | "@babel/types": "^7.16.7" 166 | }, 167 | "engines": { 168 | "node": ">=6.9.0" 169 | } 170 | }, 171 | "node_modules/@babel/helper-module-transforms": { 172 | "version": "7.17.7", 173 | "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.17.7.tgz", 174 | "integrity": "sha512-VmZD99F3gNTYB7fJRDTi+u6l/zxY0BE6OIxPSU7a50s6ZUQkHwSDmV92FfM+oCG0pZRVojGYhkR8I0OGeCVREw==", 175 | "dev": true, 176 | "dependencies": { 177 | "@babel/helper-environment-visitor": "^7.16.7", 178 | "@babel/helper-module-imports": "^7.16.7", 179 | "@babel/helper-simple-access": "^7.17.7", 180 | "@babel/helper-split-export-declaration": "^7.16.7", 181 | "@babel/helper-validator-identifier": "^7.16.7", 182 | "@babel/template": "^7.16.7", 183 | "@babel/traverse": "^7.17.3", 184 | "@babel/types": "^7.17.0" 185 | }, 186 | "engines": { 187 | "node": ">=6.9.0" 188 | } 189 | }, 190 | "node_modules/@babel/helper-simple-access": { 191 | "version": "7.17.7", 192 | "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.17.7.tgz", 193 | "integrity": "sha512-txyMCGroZ96i+Pxr3Je3lzEJjqwaRC9buMUgtomcrLe5Nd0+fk1h0LLA+ixUF5OW7AhHuQ7Es1WcQJZmZsz2XA==", 194 | "dev": true, 195 | "dependencies": { 196 | "@babel/types": "^7.17.0" 197 | }, 198 | "engines": { 199 | "node": ">=6.9.0" 200 | } 201 | }, 202 | "node_modules/@babel/helper-split-export-declaration": { 203 | "version": "7.22.6", 204 | "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.22.6.tgz", 205 | "integrity": "sha512-AsUnxuLhRYsisFiaJwvp1QF+I3KjD5FOxut14q/GzovUe6orHLesW2C7d754kRm53h5gqrz6sFl6sxc4BVtE/g==", 206 | "dev": true, 207 | "dependencies": { 208 | "@babel/types": "^7.22.5" 209 | }, 210 | "engines": { 211 | "node": ">=6.9.0" 212 | } 213 | }, 214 | "node_modules/@babel/helper-string-parser": { 215 | "version": "7.25.9", 216 | "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.25.9.tgz", 217 | "integrity": "sha512-4A/SCr/2KLd5jrtOMFzaKjVtAei3+2r/NChoBNoZ3EyP/+GlhoaEGoWOZUmFmoITP7zOJyHIMm+DYRd8o3PvHA==", 218 | "dev": true, 219 | "license": "MIT", 220 | "engines": { 221 | "node": ">=6.9.0" 222 | } 223 | }, 224 | "node_modules/@babel/helper-validator-identifier": { 225 | "version": "7.25.9", 226 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.25.9.tgz", 227 | "integrity": "sha512-Ed61U6XJc3CVRfkERJWDz4dJwKe7iLmmJsbOGu9wSloNSFttHV0I8g6UAgb7qnK5ly5bGLPd4oXZlxCdANBOWQ==", 228 | "dev": true, 229 | "license": "MIT", 230 | "engines": { 231 | "node": ">=6.9.0" 232 | } 233 | }, 234 | "node_modules/@babel/helper-validator-option": { 235 | "version": "7.16.7", 236 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.16.7.tgz", 237 | "integrity": "sha512-TRtenOuRUVo9oIQGPC5G9DgK4743cdxvtOw0weQNpZXaS16SCBi5MNjZF8vba3ETURjZpTbVn7Vvcf2eAwFozQ==", 238 | "dev": true, 239 | "engines": { 240 | "node": ">=6.9.0" 241 | } 242 | }, 243 | "node_modules/@babel/helpers": { 244 | "version": "7.27.0", 245 | "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.27.0.tgz", 246 | "integrity": "sha512-U5eyP/CTFPuNE3qk+WZMxFkp/4zUzdceQlfzf7DdGdhp+Fezd7HD+i8Y24ZuTMKX3wQBld449jijbGq6OdGNQg==", 247 | "dev": true, 248 | "license": "MIT", 249 | "dependencies": { 250 | "@babel/template": "^7.27.0", 251 | "@babel/types": "^7.27.0" 252 | }, 253 | "engines": { 254 | "node": ">=6.9.0" 255 | } 256 | }, 257 | "node_modules/@babel/parser": { 258 | "version": "7.27.0", 259 | "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.27.0.tgz", 260 | "integrity": "sha512-iaepho73/2Pz7w2eMS0Q5f83+0RKI7i4xmiYeBmDzfRVbQtTOG7Ts0S4HzJVsTMGI9keU8rNfuZr8DKfSt7Yyg==", 261 | "dev": true, 262 | "license": "MIT", 263 | "dependencies": { 264 | "@babel/types": "^7.27.0" 265 | }, 266 | "bin": { 267 | "parser": "bin/babel-parser.js" 268 | }, 269 | "engines": { 270 | "node": ">=6.0.0" 271 | } 272 | }, 273 | "node_modules/@babel/template": { 274 | "version": "7.27.0", 275 | "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.27.0.tgz", 276 | "integrity": "sha512-2ncevenBqXI6qRMukPlXwHKHchC7RyMuu4xv5JBXRfOGVcTy1mXCD12qrp7Jsoxll1EV3+9sE4GugBVRjT2jFA==", 277 | "dev": true, 278 | "license": "MIT", 279 | "dependencies": { 280 | "@babel/code-frame": "^7.26.2", 281 | "@babel/parser": "^7.27.0", 282 | "@babel/types": "^7.27.0" 283 | }, 284 | "engines": { 285 | "node": ">=6.9.0" 286 | } 287 | }, 288 | "node_modules/@babel/traverse": { 289 | "version": "7.24.0", 290 | "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.24.0.tgz", 291 | "integrity": "sha512-HfuJlI8qq3dEDmNU5ChzzpZRWq+oxCZQyMzIMEqLho+AQnhMnKQUzH6ydo3RBl/YjPCuk68Y6s0Gx0AeyULiWw==", 292 | "dev": true, 293 | "dependencies": { 294 | "@babel/code-frame": "^7.23.5", 295 | "@babel/generator": "^7.23.6", 296 | "@babel/helper-environment-visitor": "^7.22.20", 297 | "@babel/helper-function-name": "^7.23.0", 298 | "@babel/helper-hoist-variables": "^7.22.5", 299 | "@babel/helper-split-export-declaration": "^7.22.6", 300 | "@babel/parser": "^7.24.0", 301 | "@babel/types": "^7.24.0", 302 | "debug": "^4.3.1", 303 | "globals": "^11.1.0" 304 | }, 305 | "engines": { 306 | "node": ">=6.9.0" 307 | } 308 | }, 309 | "node_modules/@babel/traverse/node_modules/globals": { 310 | "version": "11.12.0", 311 | "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", 312 | "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", 313 | "dev": true, 314 | "engines": { 315 | "node": ">=4" 316 | } 317 | }, 318 | "node_modules/@babel/types": { 319 | "version": "7.27.0", 320 | "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.27.0.tgz", 321 | "integrity": "sha512-H45s8fVLYjbhFH62dIJ3WtmJ6RSPt/3DRO0ZcT2SUiYiQyz3BLVb9ADEnLl91m74aQPS3AzzeajZHYOalWe3bg==", 322 | "dev": true, 323 | "license": "MIT", 324 | "dependencies": { 325 | "@babel/helper-string-parser": "^7.25.9", 326 | "@babel/helper-validator-identifier": "^7.25.9" 327 | }, 328 | "engines": { 329 | "node": ">=6.9.0" 330 | } 331 | }, 332 | "node_modules/@eslint/eslintrc": { 333 | "version": "1.2.1", 334 | "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-1.2.1.tgz", 335 | "integrity": "sha512-bxvbYnBPN1Gibwyp6NrpnFzA3YtRL3BBAyEAFVIpNTm2Rn4Vy87GA5M4aSn3InRrlsbX5N0GW7XIx+U4SAEKdQ==", 336 | "dev": true, 337 | "dependencies": { 338 | "ajv": "^6.12.4", 339 | "debug": "^4.3.2", 340 | "espree": "^9.3.1", 341 | "globals": "^13.9.0", 342 | "ignore": "^5.2.0", 343 | "import-fresh": "^3.2.1", 344 | "js-yaml": "^4.1.0", 345 | "minimatch": "^3.0.4", 346 | "strip-json-comments": "^3.1.1" 347 | }, 348 | "engines": { 349 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 350 | } 351 | }, 352 | "node_modules/@humanwhocodes/config-array": { 353 | "version": "0.9.5", 354 | "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.9.5.tgz", 355 | "integrity": "sha512-ObyMyWxZiCu/yTisA7uzx81s40xR2fD5Cg/2Kq7G02ajkNubJf6BopgDTmDyc3U7sXpNKM8cYOw7s7Tyr+DnCw==", 356 | "dev": true, 357 | "dependencies": { 358 | "@humanwhocodes/object-schema": "^1.2.1", 359 | "debug": "^4.1.1", 360 | "minimatch": "^3.0.4" 361 | }, 362 | "engines": { 363 | "node": ">=10.10.0" 364 | } 365 | }, 366 | "node_modules/@humanwhocodes/object-schema": { 367 | "version": "1.2.1", 368 | "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz", 369 | "integrity": "sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==", 370 | "dev": true 371 | }, 372 | "node_modules/@istanbuljs/load-nyc-config": { 373 | "version": "1.1.0", 374 | "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", 375 | "integrity": "sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ==", 376 | "dev": true, 377 | "dependencies": { 378 | "camelcase": "^5.3.1", 379 | "find-up": "^4.1.0", 380 | "get-package-type": "^0.1.0", 381 | "js-yaml": "^3.13.1", 382 | "resolve-from": "^5.0.0" 383 | }, 384 | "engines": { 385 | "node": ">=8" 386 | } 387 | }, 388 | "node_modules/@istanbuljs/load-nyc-config/node_modules/argparse": { 389 | "version": "1.0.10", 390 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 391 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 392 | "dev": true, 393 | "dependencies": { 394 | "sprintf-js": "~1.0.2" 395 | } 396 | }, 397 | "node_modules/@istanbuljs/load-nyc-config/node_modules/js-yaml": { 398 | "version": "3.14.1", 399 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", 400 | "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", 401 | "dev": true, 402 | "dependencies": { 403 | "argparse": "^1.0.7", 404 | "esprima": "^4.0.0" 405 | }, 406 | "bin": { 407 | "js-yaml": "bin/js-yaml.js" 408 | } 409 | }, 410 | "node_modules/@istanbuljs/load-nyc-config/node_modules/resolve-from": { 411 | "version": "5.0.0", 412 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", 413 | "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", 414 | "dev": true, 415 | "engines": { 416 | "node": ">=8" 417 | } 418 | }, 419 | "node_modules/@istanbuljs/schema": { 420 | "version": "0.1.3", 421 | "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", 422 | "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", 423 | "dev": true, 424 | "engines": { 425 | "node": ">=8" 426 | } 427 | }, 428 | "node_modules/@jridgewell/gen-mapping": { 429 | "version": "0.3.4", 430 | "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.4.tgz", 431 | "integrity": "sha512-Oud2QPM5dHviZNn4y/WhhYKSXksv+1xLEIsNrAbGcFzUN3ubqWRFT5gwPchNc5NuzILOU4tPBDTZ4VwhL8Y7cw==", 432 | "dev": true, 433 | "dependencies": { 434 | "@jridgewell/set-array": "^1.0.1", 435 | "@jridgewell/sourcemap-codec": "^1.4.10", 436 | "@jridgewell/trace-mapping": "^0.3.9" 437 | }, 438 | "engines": { 439 | "node": ">=6.0.0" 440 | } 441 | }, 442 | "node_modules/@jridgewell/resolve-uri": { 443 | "version": "3.1.2", 444 | "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", 445 | "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", 446 | "dev": true, 447 | "engines": { 448 | "node": ">=6.0.0" 449 | } 450 | }, 451 | "node_modules/@jridgewell/set-array": { 452 | "version": "1.2.1", 453 | "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.2.1.tgz", 454 | "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==", 455 | "dev": true, 456 | "engines": { 457 | "node": ">=6.0.0" 458 | } 459 | }, 460 | "node_modules/@jridgewell/sourcemap-codec": { 461 | "version": "1.4.15", 462 | "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz", 463 | "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==", 464 | "dev": true 465 | }, 466 | "node_modules/@jridgewell/trace-mapping": { 467 | "version": "0.3.23", 468 | "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.23.tgz", 469 | "integrity": "sha512-9/4foRoUKp8s96tSkh8DlAAc5A0Ty8vLXld+l9gjKKY6ckwI8G15f0hskGmuLZu78ZlGa1vtsfOa+lnB4vG6Jg==", 470 | "dev": true, 471 | "dependencies": { 472 | "@jridgewell/resolve-uri": "^3.1.0", 473 | "@jridgewell/sourcemap-codec": "^1.4.14" 474 | } 475 | }, 476 | "node_modules/@ungap/promise-all-settled": { 477 | "version": "1.1.2", 478 | "resolved": "https://registry.npmjs.org/@ungap/promise-all-settled/-/promise-all-settled-1.1.2.tgz", 479 | "integrity": "sha512-sL/cEvJWAnClXw0wHk85/2L0G6Sj8UB0Ctc1TEMbKSsmpRosqhwj9gWgFRZSrBr2f9tiXISwNhCPmlfqUqyb9Q==", 480 | "dev": true 481 | }, 482 | "node_modules/accepts": { 483 | "version": "1.3.8", 484 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", 485 | "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", 486 | "dev": true, 487 | "dependencies": { 488 | "mime-types": "~2.1.34", 489 | "negotiator": "0.6.3" 490 | }, 491 | "engines": { 492 | "node": ">= 0.6" 493 | } 494 | }, 495 | "node_modules/accepts/node_modules/mime-db": { 496 | "version": "1.52.0", 497 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 498 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", 499 | "dev": true, 500 | "engines": { 501 | "node": ">= 0.6" 502 | } 503 | }, 504 | "node_modules/accepts/node_modules/mime-types": { 505 | "version": "2.1.35", 506 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 507 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 508 | "dev": true, 509 | "dependencies": { 510 | "mime-db": "1.52.0" 511 | }, 512 | "engines": { 513 | "node": ">= 0.6" 514 | } 515 | }, 516 | "node_modules/acorn": { 517 | "version": "8.7.0", 518 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.7.0.tgz", 519 | "integrity": "sha512-V/LGr1APy+PXIwKebEWrkZPwoeoF+w1jiOBUmuxuiUIaOHtob8Qc9BTrYo7VuI5fR8tqsy+buA2WFooR5olqvQ==", 520 | "dev": true, 521 | "bin": { 522 | "acorn": "bin/acorn" 523 | }, 524 | "engines": { 525 | "node": ">=0.4.0" 526 | } 527 | }, 528 | "node_modules/acorn-jsx": { 529 | "version": "5.3.2", 530 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", 531 | "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", 532 | "dev": true, 533 | "peerDependencies": { 534 | "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" 535 | } 536 | }, 537 | "node_modules/aggregate-error": { 538 | "version": "3.1.0", 539 | "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz", 540 | "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==", 541 | "dev": true, 542 | "dependencies": { 543 | "clean-stack": "^2.0.0", 544 | "indent-string": "^4.0.0" 545 | }, 546 | "engines": { 547 | "node": ">=8" 548 | } 549 | }, 550 | "node_modules/ajv": { 551 | "version": "6.12.6", 552 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", 553 | "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", 554 | "dev": true, 555 | "dependencies": { 556 | "fast-deep-equal": "^3.1.1", 557 | "fast-json-stable-stringify": "^2.0.0", 558 | "json-schema-traverse": "^0.4.1", 559 | "uri-js": "^4.2.2" 560 | }, 561 | "funding": { 562 | "type": "github", 563 | "url": "https://github.com/sponsors/epoberezkin" 564 | } 565 | }, 566 | "node_modules/ansi-colors": { 567 | "version": "4.1.1", 568 | "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", 569 | "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", 570 | "dev": true, 571 | "engines": { 572 | "node": ">=6" 573 | } 574 | }, 575 | "node_modules/ansi-regex": { 576 | "version": "5.0.1", 577 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 578 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 579 | "dev": true, 580 | "engines": { 581 | "node": ">=8" 582 | } 583 | }, 584 | "node_modules/ansi-styles": { 585 | "version": "4.3.0", 586 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 587 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 588 | "dev": true, 589 | "dependencies": { 590 | "color-convert": "^2.0.1" 591 | }, 592 | "engines": { 593 | "node": ">=8" 594 | }, 595 | "funding": { 596 | "url": "https://github.com/chalk/ansi-styles?sponsor=1" 597 | } 598 | }, 599 | "node_modules/anymatch": { 600 | "version": "3.1.2", 601 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 602 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 603 | "dev": true, 604 | "dependencies": { 605 | "normalize-path": "^3.0.0", 606 | "picomatch": "^2.0.4" 607 | }, 608 | "engines": { 609 | "node": ">= 8" 610 | } 611 | }, 612 | "node_modules/append-transform": { 613 | "version": "2.0.0", 614 | "resolved": "https://registry.npmjs.org/append-transform/-/append-transform-2.0.0.tgz", 615 | "integrity": "sha512-7yeyCEurROLQJFv5Xj4lEGTy0borxepjFv1g22oAdqFu//SrAlDl1O1Nxx15SH1RoliUml6p8dwJW9jvZughhg==", 616 | "dev": true, 617 | "dependencies": { 618 | "default-require-extensions": "^3.0.0" 619 | }, 620 | "engines": { 621 | "node": ">=8" 622 | } 623 | }, 624 | "node_modules/archy": { 625 | "version": "1.0.0", 626 | "resolved": "https://registry.npmjs.org/archy/-/archy-1.0.0.tgz", 627 | "integrity": "sha1-+cjBN1fMHde8N5rHeyxipcKGjEA=", 628 | "dev": true 629 | }, 630 | "node_modules/argparse": { 631 | "version": "2.0.1", 632 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", 633 | "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", 634 | "dev": true 635 | }, 636 | "node_modules/array-flatten": { 637 | "version": "1.1.1", 638 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 639 | "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==", 640 | "dev": true 641 | }, 642 | "node_modules/asap": { 643 | "version": "2.0.6", 644 | "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", 645 | "integrity": "sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==", 646 | "dev": true 647 | }, 648 | "node_modules/assertion-error": { 649 | "version": "1.1.0", 650 | "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz", 651 | "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==", 652 | "dev": true, 653 | "engines": { 654 | "node": "*" 655 | } 656 | }, 657 | "node_modules/async": { 658 | "version": "3.2.6", 659 | "resolved": "https://registry.npmjs.org/async/-/async-3.2.6.tgz", 660 | "integrity": "sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA==", 661 | "license": "MIT" 662 | }, 663 | "node_modules/asynckit": { 664 | "version": "0.4.0", 665 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 666 | "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", 667 | "dev": true 668 | }, 669 | "node_modules/balanced-match": { 670 | "version": "1.0.0", 671 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 672 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 673 | "dev": true 674 | }, 675 | "node_modules/binary-extensions": { 676 | "version": "2.2.0", 677 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 678 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 679 | "dev": true, 680 | "engines": { 681 | "node": ">=8" 682 | } 683 | }, 684 | "node_modules/body-parser": { 685 | "version": "1.20.3", 686 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.3.tgz", 687 | "integrity": "sha512-7rAxByjUMqQ3/bHJy7D6OGXvx/MMc4IqBn/X0fcM1QUcAItpZrBEYhWGem+tzXH90c+G01ypMcYJBO9Y30203g==", 688 | "dev": true, 689 | "license": "MIT", 690 | "dependencies": { 691 | "bytes": "3.1.2", 692 | "content-type": "~1.0.5", 693 | "debug": "2.6.9", 694 | "depd": "2.0.0", 695 | "destroy": "1.2.0", 696 | "http-errors": "2.0.0", 697 | "iconv-lite": "0.4.24", 698 | "on-finished": "2.4.1", 699 | "qs": "6.13.0", 700 | "raw-body": "2.5.2", 701 | "type-is": "~1.6.18", 702 | "unpipe": "1.0.0" 703 | }, 704 | "engines": { 705 | "node": ">= 0.8", 706 | "npm": "1.2.8000 || >= 1.4.16" 707 | } 708 | }, 709 | "node_modules/body-parser/node_modules/debug": { 710 | "version": "2.6.9", 711 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 712 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 713 | "dev": true, 714 | "license": "MIT", 715 | "dependencies": { 716 | "ms": "2.0.0" 717 | } 718 | }, 719 | "node_modules/body-parser/node_modules/ms": { 720 | "version": "2.0.0", 721 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 722 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", 723 | "dev": true, 724 | "license": "MIT" 725 | }, 726 | "node_modules/brace-expansion": { 727 | "version": "1.1.11", 728 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 729 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 730 | "dev": true, 731 | "dependencies": { 732 | "balanced-match": "^1.0.0", 733 | "concat-map": "0.0.1" 734 | } 735 | }, 736 | "node_modules/braces": { 737 | "version": "3.0.3", 738 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", 739 | "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", 740 | "dev": true, 741 | "license": "MIT", 742 | "dependencies": { 743 | "fill-range": "^7.1.1" 744 | }, 745 | "engines": { 746 | "node": ">=8" 747 | } 748 | }, 749 | "node_modules/browser-stdout": { 750 | "version": "1.3.1", 751 | "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz", 752 | "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", 753 | "dev": true 754 | }, 755 | "node_modules/browserslist": { 756 | "version": "4.20.0", 757 | "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.20.0.tgz", 758 | "integrity": "sha512-bnpOoa+DownbciXj0jVGENf8VYQnE2LNWomhYuCsMmmx9Jd9lwq0WXODuwpSsp8AVdKM2/HorrzxAfbKvWTByQ==", 759 | "dev": true, 760 | "dependencies": { 761 | "caniuse-lite": "^1.0.30001313", 762 | "electron-to-chromium": "^1.4.76", 763 | "escalade": "^3.1.1", 764 | "node-releases": "^2.0.2", 765 | "picocolors": "^1.0.0" 766 | }, 767 | "bin": { 768 | "browserslist": "cli.js" 769 | }, 770 | "engines": { 771 | "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" 772 | }, 773 | "funding": { 774 | "type": "opencollective", 775 | "url": "https://opencollective.com/browserslist" 776 | } 777 | }, 778 | "node_modules/bytes": { 779 | "version": "3.1.2", 780 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", 781 | "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", 782 | "dev": true, 783 | "license": "MIT", 784 | "engines": { 785 | "node": ">= 0.8" 786 | } 787 | }, 788 | "node_modules/caching-transform": { 789 | "version": "4.0.0", 790 | "resolved": "https://registry.npmjs.org/caching-transform/-/caching-transform-4.0.0.tgz", 791 | "integrity": "sha512-kpqOvwXnjjN44D89K5ccQC+RUrsy7jB/XLlRrx0D7/2HNcTPqzsb6XgYoErwko6QsV184CA2YgS1fxDiiDZMWA==", 792 | "dev": true, 793 | "dependencies": { 794 | "hasha": "^5.0.0", 795 | "make-dir": "^3.0.0", 796 | "package-hash": "^4.0.0", 797 | "write-file-atomic": "^3.0.0" 798 | }, 799 | "engines": { 800 | "node": ">=8" 801 | } 802 | }, 803 | "node_modules/call-bind-apply-helpers": { 804 | "version": "1.0.2", 805 | "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", 806 | "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", 807 | "dev": true, 808 | "license": "MIT", 809 | "dependencies": { 810 | "es-errors": "^1.3.0", 811 | "function-bind": "^1.1.2" 812 | }, 813 | "engines": { 814 | "node": ">= 0.4" 815 | } 816 | }, 817 | "node_modules/call-bound": { 818 | "version": "1.0.4", 819 | "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", 820 | "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", 821 | "dev": true, 822 | "license": "MIT", 823 | "dependencies": { 824 | "call-bind-apply-helpers": "^1.0.2", 825 | "get-intrinsic": "^1.3.0" 826 | }, 827 | "engines": { 828 | "node": ">= 0.4" 829 | }, 830 | "funding": { 831 | "url": "https://github.com/sponsors/ljharb" 832 | } 833 | }, 834 | "node_modules/callsites": { 835 | "version": "3.1.0", 836 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", 837 | "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", 838 | "dev": true, 839 | "engines": { 840 | "node": ">=6" 841 | } 842 | }, 843 | "node_modules/camelcase": { 844 | "version": "5.3.1", 845 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 846 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 847 | "dev": true, 848 | "engines": { 849 | "node": ">=6" 850 | } 851 | }, 852 | "node_modules/caniuse-lite": { 853 | "version": "1.0.30001317", 854 | "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001317.tgz", 855 | "integrity": "sha512-xIZLh8gBm4dqNX0gkzrBeyI86J2eCjWzYAs40q88smG844YIrN4tVQl/RhquHvKEKImWWFIVh1Lxe5n1G/N+GQ==", 856 | "dev": true, 857 | "funding": { 858 | "type": "opencollective", 859 | "url": "https://opencollective.com/browserslist" 860 | } 861 | }, 862 | "node_modules/chai": { 863 | "version": "4.2.0", 864 | "resolved": "https://registry.npmjs.org/chai/-/chai-4.2.0.tgz", 865 | "integrity": "sha512-XQU3bhBukrOsQCuwZndwGcCVQHyZi53fQ6Ys1Fym7E4olpIqqZZhhoFJoaKVvV17lWQoXYwgWN2nF5crA8J2jw==", 866 | "dev": true, 867 | "dependencies": { 868 | "assertion-error": "^1.1.0", 869 | "check-error": "^1.0.2", 870 | "deep-eql": "^3.0.1", 871 | "get-func-name": "^2.0.0", 872 | "pathval": "^1.1.0", 873 | "type-detect": "^4.0.5" 874 | }, 875 | "engines": { 876 | "node": ">=4" 877 | } 878 | }, 879 | "node_modules/chalk": { 880 | "version": "4.1.2", 881 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", 882 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", 883 | "dev": true, 884 | "dependencies": { 885 | "ansi-styles": "^4.1.0", 886 | "supports-color": "^7.1.0" 887 | }, 888 | "engines": { 889 | "node": ">=10" 890 | }, 891 | "funding": { 892 | "url": "https://github.com/chalk/chalk?sponsor=1" 893 | } 894 | }, 895 | "node_modules/check-error": { 896 | "version": "1.0.2", 897 | "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz", 898 | "integrity": "sha1-V00xLt2Iu13YkS6Sht1sCu1KrII=", 899 | "dev": true, 900 | "engines": { 901 | "node": "*" 902 | } 903 | }, 904 | "node_modules/chokidar": { 905 | "version": "3.5.3", 906 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", 907 | "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", 908 | "dev": true, 909 | "funding": [ 910 | { 911 | "type": "individual", 912 | "url": "https://paulmillr.com/funding/" 913 | } 914 | ], 915 | "dependencies": { 916 | "anymatch": "~3.1.2", 917 | "braces": "~3.0.2", 918 | "glob-parent": "~5.1.2", 919 | "is-binary-path": "~2.1.0", 920 | "is-glob": "~4.0.1", 921 | "normalize-path": "~3.0.0", 922 | "readdirp": "~3.6.0" 923 | }, 924 | "engines": { 925 | "node": ">= 8.10.0" 926 | }, 927 | "optionalDependencies": { 928 | "fsevents": "~2.3.2" 929 | } 930 | }, 931 | "node_modules/chokidar/node_modules/glob-parent": { 932 | "version": "5.1.2", 933 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 934 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 935 | "dev": true, 936 | "dependencies": { 937 | "is-glob": "^4.0.1" 938 | }, 939 | "engines": { 940 | "node": ">= 6" 941 | } 942 | }, 943 | "node_modules/clean-stack": { 944 | "version": "2.2.0", 945 | "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", 946 | "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", 947 | "dev": true, 948 | "engines": { 949 | "node": ">=6" 950 | } 951 | }, 952 | "node_modules/cliui": { 953 | "version": "6.0.0", 954 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", 955 | "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", 956 | "dev": true, 957 | "dependencies": { 958 | "string-width": "^4.2.0", 959 | "strip-ansi": "^6.0.0", 960 | "wrap-ansi": "^6.2.0" 961 | } 962 | }, 963 | "node_modules/color-convert": { 964 | "version": "2.0.1", 965 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 966 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 967 | "dev": true, 968 | "dependencies": { 969 | "color-name": "~1.1.4" 970 | }, 971 | "engines": { 972 | "node": ">=7.0.0" 973 | } 974 | }, 975 | "node_modules/color-name": { 976 | "version": "1.1.4", 977 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 978 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 979 | "dev": true 980 | }, 981 | "node_modules/combined-stream": { 982 | "version": "1.0.8", 983 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 984 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 985 | "dev": true, 986 | "dependencies": { 987 | "delayed-stream": "~1.0.0" 988 | }, 989 | "engines": { 990 | "node": ">= 0.8" 991 | } 992 | }, 993 | "node_modules/commondir": { 994 | "version": "1.0.1", 995 | "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", 996 | "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=", 997 | "dev": true 998 | }, 999 | "node_modules/component-emitter": { 1000 | "version": "1.3.0", 1001 | "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", 1002 | "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==", 1003 | "dev": true 1004 | }, 1005 | "node_modules/concat-map": { 1006 | "version": "0.0.1", 1007 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 1008 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 1009 | "dev": true 1010 | }, 1011 | "node_modules/content-disposition": { 1012 | "version": "0.5.4", 1013 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", 1014 | "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", 1015 | "dev": true, 1016 | "dependencies": { 1017 | "safe-buffer": "5.2.1" 1018 | }, 1019 | "engines": { 1020 | "node": ">= 0.6" 1021 | } 1022 | }, 1023 | "node_modules/content-disposition/node_modules/safe-buffer": { 1024 | "version": "5.2.1", 1025 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 1026 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", 1027 | "dev": true, 1028 | "funding": [ 1029 | { 1030 | "type": "github", 1031 | "url": "https://github.com/sponsors/feross" 1032 | }, 1033 | { 1034 | "type": "patreon", 1035 | "url": "https://www.patreon.com/feross" 1036 | }, 1037 | { 1038 | "type": "consulting", 1039 | "url": "https://feross.org/support" 1040 | } 1041 | ] 1042 | }, 1043 | "node_modules/content-type": { 1044 | "version": "1.0.5", 1045 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", 1046 | "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", 1047 | "dev": true, 1048 | "license": "MIT", 1049 | "engines": { 1050 | "node": ">= 0.6" 1051 | } 1052 | }, 1053 | "node_modules/convert-source-map": { 1054 | "version": "1.8.0", 1055 | "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.8.0.tgz", 1056 | "integrity": "sha512-+OQdjP49zViI/6i7nIJpA8rAl4sV/JdPfU9nZs3VqOwGIgizICvuN2ru6fMd+4llL0tar18UYJXfZ/TWtmhUjA==", 1057 | "dev": true, 1058 | "dependencies": { 1059 | "safe-buffer": "~5.1.1" 1060 | } 1061 | }, 1062 | "node_modules/cookie": { 1063 | "version": "0.7.1", 1064 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.1.tgz", 1065 | "integrity": "sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w==", 1066 | "dev": true, 1067 | "license": "MIT", 1068 | "engines": { 1069 | "node": ">= 0.6" 1070 | } 1071 | }, 1072 | "node_modules/cookie-signature": { 1073 | "version": "1.0.6", 1074 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 1075 | "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==", 1076 | "dev": true 1077 | }, 1078 | "node_modules/cookiejar": { 1079 | "version": "2.1.4", 1080 | "resolved": "https://registry.npmjs.org/cookiejar/-/cookiejar-2.1.4.tgz", 1081 | "integrity": "sha512-LDx6oHrK+PhzLKJU9j5S7/Y3jM/mUHvD/DeI1WQmJn652iPC5Y4TBzC9l+5OMOXlyTTA+SmVUPm0HQUwpD5Jqw==", 1082 | "dev": true 1083 | }, 1084 | "node_modules/cross-spawn": { 1085 | "version": "7.0.6", 1086 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", 1087 | "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", 1088 | "dev": true, 1089 | "license": "MIT", 1090 | "dependencies": { 1091 | "path-key": "^3.1.0", 1092 | "shebang-command": "^2.0.0", 1093 | "which": "^2.0.1" 1094 | }, 1095 | "engines": { 1096 | "node": ">= 8" 1097 | } 1098 | }, 1099 | "node_modules/debug": { 1100 | "version": "4.3.4", 1101 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 1102 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 1103 | "dev": true, 1104 | "dependencies": { 1105 | "ms": "2.1.2" 1106 | }, 1107 | "engines": { 1108 | "node": ">=6.0" 1109 | }, 1110 | "peerDependenciesMeta": { 1111 | "supports-color": { 1112 | "optional": true 1113 | } 1114 | } 1115 | }, 1116 | "node_modules/decamelize": { 1117 | "version": "1.2.0", 1118 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", 1119 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", 1120 | "dev": true, 1121 | "engines": { 1122 | "node": ">=0.10.0" 1123 | } 1124 | }, 1125 | "node_modules/deep-eql": { 1126 | "version": "3.0.1", 1127 | "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-3.0.1.tgz", 1128 | "integrity": "sha512-+QeIQyN5ZuO+3Uk5DYh6/1eKO0m0YmJFGNmFHGACpf1ClL1nmlV/p4gNgbl2pJGxgXb4faqo6UE+M5ACEMyVcw==", 1129 | "dev": true, 1130 | "dependencies": { 1131 | "type-detect": "^4.0.0" 1132 | }, 1133 | "engines": { 1134 | "node": ">=0.12" 1135 | } 1136 | }, 1137 | "node_modules/deep-is": { 1138 | "version": "0.1.4", 1139 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", 1140 | "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", 1141 | "dev": true 1142 | }, 1143 | "node_modules/default-require-extensions": { 1144 | "version": "3.0.0", 1145 | "resolved": "https://registry.npmjs.org/default-require-extensions/-/default-require-extensions-3.0.0.tgz", 1146 | "integrity": "sha512-ek6DpXq/SCpvjhpFsLFRVtIxJCRw6fUR42lYMVZuUMK7n8eMz4Uh5clckdBjEpLhn/gEBZo7hDJnJcwdKLKQjg==", 1147 | "dev": true, 1148 | "dependencies": { 1149 | "strip-bom": "^4.0.0" 1150 | }, 1151 | "engines": { 1152 | "node": ">=8" 1153 | } 1154 | }, 1155 | "node_modules/delayed-stream": { 1156 | "version": "1.0.0", 1157 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 1158 | "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", 1159 | "dev": true, 1160 | "engines": { 1161 | "node": ">=0.4.0" 1162 | } 1163 | }, 1164 | "node_modules/depd": { 1165 | "version": "2.0.0", 1166 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 1167 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", 1168 | "dev": true, 1169 | "license": "MIT", 1170 | "engines": { 1171 | "node": ">= 0.8" 1172 | } 1173 | }, 1174 | "node_modules/destroy": { 1175 | "version": "1.2.0", 1176 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", 1177 | "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", 1178 | "dev": true, 1179 | "license": "MIT", 1180 | "engines": { 1181 | "node": ">= 0.8", 1182 | "npm": "1.2.8000 || >= 1.4.16" 1183 | } 1184 | }, 1185 | "node_modules/dezalgo": { 1186 | "version": "1.0.4", 1187 | "resolved": "https://registry.npmjs.org/dezalgo/-/dezalgo-1.0.4.tgz", 1188 | "integrity": "sha512-rXSP0bf+5n0Qonsb+SVVfNfIsimO4HEtmnIpPHY8Q1UCzKlQrDMfdobr8nJOOsRgWCyMRqeSBQzmWUMq7zvVig==", 1189 | "dev": true, 1190 | "dependencies": { 1191 | "asap": "^2.0.0", 1192 | "wrappy": "1" 1193 | } 1194 | }, 1195 | "node_modules/diff": { 1196 | "version": "5.0.0", 1197 | "resolved": "https://registry.npmjs.org/diff/-/diff-5.0.0.tgz", 1198 | "integrity": "sha512-/VTCrvm5Z0JGty/BWHljh+BAiw3IK+2j87NGMu8Nwc/f48WoDAC395uomO9ZD117ZOBaHmkX1oyLvkVM/aIT3w==", 1199 | "dev": true, 1200 | "engines": { 1201 | "node": ">=0.3.1" 1202 | } 1203 | }, 1204 | "node_modules/doctrine": { 1205 | "version": "3.0.0", 1206 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", 1207 | "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", 1208 | "dev": true, 1209 | "dependencies": { 1210 | "esutils": "^2.0.2" 1211 | }, 1212 | "engines": { 1213 | "node": ">=6.0.0" 1214 | } 1215 | }, 1216 | "node_modules/dunder-proto": { 1217 | "version": "1.0.1", 1218 | "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", 1219 | "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", 1220 | "dev": true, 1221 | "license": "MIT", 1222 | "dependencies": { 1223 | "call-bind-apply-helpers": "^1.0.1", 1224 | "es-errors": "^1.3.0", 1225 | "gopd": "^1.2.0" 1226 | }, 1227 | "engines": { 1228 | "node": ">= 0.4" 1229 | } 1230 | }, 1231 | "node_modules/ee-first": { 1232 | "version": "1.1.1", 1233 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 1234 | "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", 1235 | "dev": true, 1236 | "license": "MIT" 1237 | }, 1238 | "node_modules/electron-to-chromium": { 1239 | "version": "1.4.84", 1240 | "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.84.tgz", 1241 | "integrity": "sha512-b+DdcyOiZtLXHdgEG8lncYJdxbdJWJvclPNMg0eLUDcSOSO876WA/pYjdSblUTd7eJdIs4YdIxHWGazx7UPSJw==", 1242 | "dev": true 1243 | }, 1244 | "node_modules/emoji-regex": { 1245 | "version": "8.0.0", 1246 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 1247 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 1248 | "dev": true 1249 | }, 1250 | "node_modules/encodeurl": { 1251 | "version": "2.0.0", 1252 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", 1253 | "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", 1254 | "dev": true, 1255 | "license": "MIT", 1256 | "engines": { 1257 | "node": ">= 0.8" 1258 | } 1259 | }, 1260 | "node_modules/es-define-property": { 1261 | "version": "1.0.1", 1262 | "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", 1263 | "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", 1264 | "dev": true, 1265 | "license": "MIT", 1266 | "engines": { 1267 | "node": ">= 0.4" 1268 | } 1269 | }, 1270 | "node_modules/es-errors": { 1271 | "version": "1.3.0", 1272 | "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", 1273 | "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", 1274 | "dev": true, 1275 | "license": "MIT", 1276 | "engines": { 1277 | "node": ">= 0.4" 1278 | } 1279 | }, 1280 | "node_modules/es-object-atoms": { 1281 | "version": "1.1.1", 1282 | "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", 1283 | "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", 1284 | "dev": true, 1285 | "license": "MIT", 1286 | "dependencies": { 1287 | "es-errors": "^1.3.0" 1288 | }, 1289 | "engines": { 1290 | "node": ">= 0.4" 1291 | } 1292 | }, 1293 | "node_modules/es6-error": { 1294 | "version": "4.1.1", 1295 | "resolved": "https://registry.npmjs.org/es6-error/-/es6-error-4.1.1.tgz", 1296 | "integrity": "sha512-Um/+FxMr9CISWh0bi5Zv0iOD+4cFh5qLeks1qhAopKVAJw3drgKbKySikp7wGhDL0HPeaja0P5ULZrxLkniUVg==", 1297 | "dev": true 1298 | }, 1299 | "node_modules/escalade": { 1300 | "version": "3.1.1", 1301 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", 1302 | "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", 1303 | "dev": true, 1304 | "engines": { 1305 | "node": ">=6" 1306 | } 1307 | }, 1308 | "node_modules/escape-html": { 1309 | "version": "1.0.3", 1310 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 1311 | "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==", 1312 | "dev": true, 1313 | "license": "MIT" 1314 | }, 1315 | "node_modules/escape-string-regexp": { 1316 | "version": "4.0.0", 1317 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", 1318 | "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", 1319 | "dev": true, 1320 | "engines": { 1321 | "node": ">=10" 1322 | }, 1323 | "funding": { 1324 | "url": "https://github.com/sponsors/sindresorhus" 1325 | } 1326 | }, 1327 | "node_modules/eslint": { 1328 | "version": "8.11.0", 1329 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.11.0.tgz", 1330 | "integrity": "sha512-/KRpd9mIRg2raGxHRGwW9ZywYNAClZrHjdueHcrVDuO3a6bj83eoTirCCk0M0yPwOjWYKHwRVRid+xK4F/GHgA==", 1331 | "dev": true, 1332 | "dependencies": { 1333 | "@eslint/eslintrc": "^1.2.1", 1334 | "@humanwhocodes/config-array": "^0.9.2", 1335 | "ajv": "^6.10.0", 1336 | "chalk": "^4.0.0", 1337 | "cross-spawn": "^7.0.2", 1338 | "debug": "^4.3.2", 1339 | "doctrine": "^3.0.0", 1340 | "escape-string-regexp": "^4.0.0", 1341 | "eslint-scope": "^7.1.1", 1342 | "eslint-utils": "^3.0.0", 1343 | "eslint-visitor-keys": "^3.3.0", 1344 | "espree": "^9.3.1", 1345 | "esquery": "^1.4.0", 1346 | "esutils": "^2.0.2", 1347 | "fast-deep-equal": "^3.1.3", 1348 | "file-entry-cache": "^6.0.1", 1349 | "functional-red-black-tree": "^1.0.1", 1350 | "glob-parent": "^6.0.1", 1351 | "globals": "^13.6.0", 1352 | "ignore": "^5.2.0", 1353 | "import-fresh": "^3.0.0", 1354 | "imurmurhash": "^0.1.4", 1355 | "is-glob": "^4.0.0", 1356 | "js-yaml": "^4.1.0", 1357 | "json-stable-stringify-without-jsonify": "^1.0.1", 1358 | "levn": "^0.4.1", 1359 | "lodash.merge": "^4.6.2", 1360 | "minimatch": "^3.0.4", 1361 | "natural-compare": "^1.4.0", 1362 | "optionator": "^0.9.1", 1363 | "regexpp": "^3.2.0", 1364 | "strip-ansi": "^6.0.1", 1365 | "strip-json-comments": "^3.1.0", 1366 | "text-table": "^0.2.0", 1367 | "v8-compile-cache": "^2.0.3" 1368 | }, 1369 | "bin": { 1370 | "eslint": "bin/eslint.js" 1371 | }, 1372 | "engines": { 1373 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1374 | }, 1375 | "funding": { 1376 | "url": "https://opencollective.com/eslint" 1377 | } 1378 | }, 1379 | "node_modules/eslint-scope": { 1380 | "version": "7.1.1", 1381 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.1.1.tgz", 1382 | "integrity": "sha512-QKQM/UXpIiHcLqJ5AOyIW7XZmzjkzQXYE54n1++wb0u9V/abW3l9uQnxX8Z5Xd18xyKIMTUAyQ0k1e8pz6LUrw==", 1383 | "dev": true, 1384 | "dependencies": { 1385 | "esrecurse": "^4.3.0", 1386 | "estraverse": "^5.2.0" 1387 | }, 1388 | "engines": { 1389 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1390 | } 1391 | }, 1392 | "node_modules/eslint-utils": { 1393 | "version": "3.0.0", 1394 | "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-3.0.0.tgz", 1395 | "integrity": "sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==", 1396 | "dev": true, 1397 | "dependencies": { 1398 | "eslint-visitor-keys": "^2.0.0" 1399 | }, 1400 | "engines": { 1401 | "node": "^10.0.0 || ^12.0.0 || >= 14.0.0" 1402 | }, 1403 | "funding": { 1404 | "url": "https://github.com/sponsors/mysticatea" 1405 | }, 1406 | "peerDependencies": { 1407 | "eslint": ">=5" 1408 | } 1409 | }, 1410 | "node_modules/eslint-utils/node_modules/eslint-visitor-keys": { 1411 | "version": "2.1.0", 1412 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz", 1413 | "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==", 1414 | "dev": true, 1415 | "engines": { 1416 | "node": ">=10" 1417 | } 1418 | }, 1419 | "node_modules/eslint-visitor-keys": { 1420 | "version": "3.3.0", 1421 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.3.0.tgz", 1422 | "integrity": "sha512-mQ+suqKJVyeuwGYHAdjMFqjCyfl8+Ldnxuyp3ldiMBFKkvytrXUZWaiPCEav8qDHKty44bD+qV1IP4T+w+xXRA==", 1423 | "dev": true, 1424 | "engines": { 1425 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1426 | } 1427 | }, 1428 | "node_modules/espree": { 1429 | "version": "9.3.1", 1430 | "resolved": "https://registry.npmjs.org/espree/-/espree-9.3.1.tgz", 1431 | "integrity": "sha512-bvdyLmJMfwkV3NCRl5ZhJf22zBFo1y8bYh3VYb+bfzqNB4Je68P2sSuXyuFquzWLebHpNd2/d5uv7yoP9ISnGQ==", 1432 | "dev": true, 1433 | "dependencies": { 1434 | "acorn": "^8.7.0", 1435 | "acorn-jsx": "^5.3.1", 1436 | "eslint-visitor-keys": "^3.3.0" 1437 | }, 1438 | "engines": { 1439 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1440 | } 1441 | }, 1442 | "node_modules/esprima": { 1443 | "version": "4.0.1", 1444 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 1445 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", 1446 | "dev": true, 1447 | "bin": { 1448 | "esparse": "bin/esparse.js", 1449 | "esvalidate": "bin/esvalidate.js" 1450 | }, 1451 | "engines": { 1452 | "node": ">=4" 1453 | } 1454 | }, 1455 | "node_modules/esquery": { 1456 | "version": "1.4.0", 1457 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.4.0.tgz", 1458 | "integrity": "sha512-cCDispWt5vHHtwMY2YrAQ4ibFkAL8RbH5YGBnZBc90MolvvfkkQcJro/aZiAQUlQ3qgrYS6D6v8Gc5G5CQsc9w==", 1459 | "dev": true, 1460 | "dependencies": { 1461 | "estraverse": "^5.1.0" 1462 | }, 1463 | "engines": { 1464 | "node": ">=0.10" 1465 | } 1466 | }, 1467 | "node_modules/esrecurse": { 1468 | "version": "4.3.0", 1469 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", 1470 | "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", 1471 | "dev": true, 1472 | "dependencies": { 1473 | "estraverse": "^5.2.0" 1474 | }, 1475 | "engines": { 1476 | "node": ">=4.0" 1477 | } 1478 | }, 1479 | "node_modules/estraverse": { 1480 | "version": "5.3.0", 1481 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", 1482 | "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", 1483 | "dev": true, 1484 | "engines": { 1485 | "node": ">=4.0" 1486 | } 1487 | }, 1488 | "node_modules/esutils": { 1489 | "version": "2.0.2", 1490 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz", 1491 | "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=", 1492 | "dev": true, 1493 | "engines": { 1494 | "node": ">=0.10.0" 1495 | } 1496 | }, 1497 | "node_modules/etag": { 1498 | "version": "1.8.1", 1499 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 1500 | "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", 1501 | "dev": true, 1502 | "license": "MIT", 1503 | "engines": { 1504 | "node": ">= 0.6" 1505 | } 1506 | }, 1507 | "node_modules/express": { 1508 | "version": "4.21.2", 1509 | "resolved": "https://registry.npmjs.org/express/-/express-4.21.2.tgz", 1510 | "integrity": "sha512-28HqgMZAmih1Czt9ny7qr6ek2qddF4FclbMzwhCREB6OFfH+rXAnuNCwo1/wFvrtbgsQDb4kSbX9de9lFbrXnA==", 1511 | "dev": true, 1512 | "license": "MIT", 1513 | "dependencies": { 1514 | "accepts": "~1.3.8", 1515 | "array-flatten": "1.1.1", 1516 | "body-parser": "1.20.3", 1517 | "content-disposition": "0.5.4", 1518 | "content-type": "~1.0.4", 1519 | "cookie": "0.7.1", 1520 | "cookie-signature": "1.0.6", 1521 | "debug": "2.6.9", 1522 | "depd": "2.0.0", 1523 | "encodeurl": "~2.0.0", 1524 | "escape-html": "~1.0.3", 1525 | "etag": "~1.8.1", 1526 | "finalhandler": "1.3.1", 1527 | "fresh": "0.5.2", 1528 | "http-errors": "2.0.0", 1529 | "merge-descriptors": "1.0.3", 1530 | "methods": "~1.1.2", 1531 | "on-finished": "2.4.1", 1532 | "parseurl": "~1.3.3", 1533 | "path-to-regexp": "0.1.12", 1534 | "proxy-addr": "~2.0.7", 1535 | "qs": "6.13.0", 1536 | "range-parser": "~1.2.1", 1537 | "safe-buffer": "5.2.1", 1538 | "send": "0.19.0", 1539 | "serve-static": "1.16.2", 1540 | "setprototypeof": "1.2.0", 1541 | "statuses": "2.0.1", 1542 | "type-is": "~1.6.18", 1543 | "utils-merge": "1.0.1", 1544 | "vary": "~1.1.2" 1545 | }, 1546 | "engines": { 1547 | "node": ">= 0.10.0" 1548 | }, 1549 | "funding": { 1550 | "type": "opencollective", 1551 | "url": "https://opencollective.com/express" 1552 | } 1553 | }, 1554 | "node_modules/express-async-errors": { 1555 | "version": "3.1.1", 1556 | "resolved": "https://registry.npmjs.org/express-async-errors/-/express-async-errors-3.1.1.tgz", 1557 | "integrity": "sha512-h6aK1da4tpqWSbyCa3FxB/V6Ehd4EEB15zyQq9qe75OZBp0krinNKuH4rAY+S/U/2I36vdLAUFSjQJ+TFmODng==", 1558 | "dev": true, 1559 | "peerDependencies": { 1560 | "express": "^4.16.2" 1561 | } 1562 | }, 1563 | "node_modules/express/node_modules/debug": { 1564 | "version": "2.6.9", 1565 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 1566 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 1567 | "dev": true, 1568 | "dependencies": { 1569 | "ms": "2.0.0" 1570 | } 1571 | }, 1572 | "node_modules/express/node_modules/ms": { 1573 | "version": "2.0.0", 1574 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1575 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", 1576 | "dev": true 1577 | }, 1578 | "node_modules/express/node_modules/safe-buffer": { 1579 | "version": "5.2.1", 1580 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 1581 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", 1582 | "dev": true, 1583 | "funding": [ 1584 | { 1585 | "type": "github", 1586 | "url": "https://github.com/sponsors/feross" 1587 | }, 1588 | { 1589 | "type": "patreon", 1590 | "url": "https://www.patreon.com/feross" 1591 | }, 1592 | { 1593 | "type": "consulting", 1594 | "url": "https://feross.org/support" 1595 | } 1596 | ] 1597 | }, 1598 | "node_modules/fast-deep-equal": { 1599 | "version": "3.1.3", 1600 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", 1601 | "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", 1602 | "dev": true 1603 | }, 1604 | "node_modules/fast-json-stable-stringify": { 1605 | "version": "2.1.0", 1606 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", 1607 | "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", 1608 | "dev": true 1609 | }, 1610 | "node_modules/fast-levenshtein": { 1611 | "version": "2.0.6", 1612 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", 1613 | "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", 1614 | "dev": true 1615 | }, 1616 | "node_modules/fast-safe-stringify": { 1617 | "version": "2.1.1", 1618 | "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.1.1.tgz", 1619 | "integrity": "sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA==", 1620 | "dev": true 1621 | }, 1622 | "node_modules/file-entry-cache": { 1623 | "version": "6.0.1", 1624 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", 1625 | "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", 1626 | "dev": true, 1627 | "dependencies": { 1628 | "flat-cache": "^3.0.4" 1629 | }, 1630 | "engines": { 1631 | "node": "^10.12.0 || >=12.0.0" 1632 | } 1633 | }, 1634 | "node_modules/fill-range": { 1635 | "version": "7.1.1", 1636 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", 1637 | "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", 1638 | "dev": true, 1639 | "license": "MIT", 1640 | "dependencies": { 1641 | "to-regex-range": "^5.0.1" 1642 | }, 1643 | "engines": { 1644 | "node": ">=8" 1645 | } 1646 | }, 1647 | "node_modules/finalhandler": { 1648 | "version": "1.3.1", 1649 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.1.tgz", 1650 | "integrity": "sha512-6BN9trH7bp3qvnrRyzsBz+g3lZxTNZTbVO2EV1CS0WIcDbawYVdYvGflME/9QP0h0pYlCDBCTjYa9nZzMDpyxQ==", 1651 | "dev": true, 1652 | "license": "MIT", 1653 | "dependencies": { 1654 | "debug": "2.6.9", 1655 | "encodeurl": "~2.0.0", 1656 | "escape-html": "~1.0.3", 1657 | "on-finished": "2.4.1", 1658 | "parseurl": "~1.3.3", 1659 | "statuses": "2.0.1", 1660 | "unpipe": "~1.0.0" 1661 | }, 1662 | "engines": { 1663 | "node": ">= 0.8" 1664 | } 1665 | }, 1666 | "node_modules/finalhandler/node_modules/debug": { 1667 | "version": "2.6.9", 1668 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 1669 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 1670 | "dev": true, 1671 | "license": "MIT", 1672 | "dependencies": { 1673 | "ms": "2.0.0" 1674 | } 1675 | }, 1676 | "node_modules/finalhandler/node_modules/ms": { 1677 | "version": "2.0.0", 1678 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1679 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", 1680 | "dev": true, 1681 | "license": "MIT" 1682 | }, 1683 | "node_modules/find-cache-dir": { 1684 | "version": "3.3.2", 1685 | "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-3.3.2.tgz", 1686 | "integrity": "sha512-wXZV5emFEjrridIgED11OoUKLxiYjAcqot/NJdAkOhlJ+vGzwhOAfcG5OX1jP+S0PcjEn8bdMJv+g2jwQ3Onig==", 1687 | "dev": true, 1688 | "dependencies": { 1689 | "commondir": "^1.0.1", 1690 | "make-dir": "^3.0.2", 1691 | "pkg-dir": "^4.1.0" 1692 | }, 1693 | "engines": { 1694 | "node": ">=8" 1695 | }, 1696 | "funding": { 1697 | "url": "https://github.com/avajs/find-cache-dir?sponsor=1" 1698 | } 1699 | }, 1700 | "node_modules/find-up": { 1701 | "version": "4.1.0", 1702 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", 1703 | "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", 1704 | "dev": true, 1705 | "dependencies": { 1706 | "locate-path": "^5.0.0", 1707 | "path-exists": "^4.0.0" 1708 | }, 1709 | "engines": { 1710 | "node": ">=8" 1711 | } 1712 | }, 1713 | "node_modules/flat": { 1714 | "version": "5.0.2", 1715 | "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz", 1716 | "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==", 1717 | "dev": true, 1718 | "bin": { 1719 | "flat": "cli.js" 1720 | } 1721 | }, 1722 | "node_modules/flat-cache": { 1723 | "version": "3.0.4", 1724 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz", 1725 | "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==", 1726 | "dev": true, 1727 | "dependencies": { 1728 | "flatted": "^3.1.0", 1729 | "rimraf": "^3.0.2" 1730 | }, 1731 | "engines": { 1732 | "node": "^10.12.0 || >=12.0.0" 1733 | } 1734 | }, 1735 | "node_modules/flatted": { 1736 | "version": "3.2.5", 1737 | "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.5.tgz", 1738 | "integrity": "sha512-WIWGi2L3DyTUvUrwRKgGi9TwxQMUEqPOPQBVi71R96jZXJdFskXEmf54BoZaS1kknGODoIGASGEzBUYdyMCBJg==", 1739 | "dev": true 1740 | }, 1741 | "node_modules/foreground-child": { 1742 | "version": "2.0.0", 1743 | "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-2.0.0.tgz", 1744 | "integrity": "sha512-dCIq9FpEcyQyXKCkyzmlPTFNgrCzPudOe+mhvJU5zAtlBnGVy2yKxtfsxK2tQBThwq225jcvBjpw1Gr40uzZCA==", 1745 | "dev": true, 1746 | "dependencies": { 1747 | "cross-spawn": "^7.0.0", 1748 | "signal-exit": "^3.0.2" 1749 | }, 1750 | "engines": { 1751 | "node": ">=8.0.0" 1752 | } 1753 | }, 1754 | "node_modules/form-data": { 1755 | "version": "4.0.0", 1756 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", 1757 | "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", 1758 | "dev": true, 1759 | "dependencies": { 1760 | "asynckit": "^0.4.0", 1761 | "combined-stream": "^1.0.8", 1762 | "mime-types": "^2.1.12" 1763 | }, 1764 | "engines": { 1765 | "node": ">= 6" 1766 | } 1767 | }, 1768 | "node_modules/formidable": { 1769 | "version": "2.1.1", 1770 | "resolved": "https://registry.npmjs.org/formidable/-/formidable-2.1.1.tgz", 1771 | "integrity": "sha512-0EcS9wCFEzLvfiks7omJ+SiYJAiD+TzK4Pcw1UlUoGnhUxDcMKjt0P7x8wEb0u6OHu8Nb98WG3nxtlF5C7bvUQ==", 1772 | "dev": true, 1773 | "dependencies": { 1774 | "dezalgo": "^1.0.4", 1775 | "hexoid": "^1.0.0", 1776 | "once": "^1.4.0", 1777 | "qs": "^6.11.0" 1778 | }, 1779 | "funding": { 1780 | "url": "https://ko-fi.com/tunnckoCore/commissions" 1781 | } 1782 | }, 1783 | "node_modules/forwarded": { 1784 | "version": "0.2.0", 1785 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 1786 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", 1787 | "dev": true, 1788 | "engines": { 1789 | "node": ">= 0.6" 1790 | } 1791 | }, 1792 | "node_modules/fresh": { 1793 | "version": "0.5.2", 1794 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 1795 | "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", 1796 | "dev": true, 1797 | "license": "MIT", 1798 | "engines": { 1799 | "node": ">= 0.6" 1800 | } 1801 | }, 1802 | "node_modules/fromentries": { 1803 | "version": "1.3.2", 1804 | "resolved": "https://registry.npmjs.org/fromentries/-/fromentries-1.3.2.tgz", 1805 | "integrity": "sha512-cHEpEQHUg0f8XdtZCc2ZAhrHzKzT0MrFUTcvx+hfxYu7rGMDc5SKoXFh+n4YigxsHXRzc6OrCshdR1bWH6HHyg==", 1806 | "dev": true, 1807 | "funding": [ 1808 | { 1809 | "type": "github", 1810 | "url": "https://github.com/sponsors/feross" 1811 | }, 1812 | { 1813 | "type": "patreon", 1814 | "url": "https://www.patreon.com/feross" 1815 | }, 1816 | { 1817 | "type": "consulting", 1818 | "url": "https://feross.org/support" 1819 | } 1820 | ] 1821 | }, 1822 | "node_modules/fs.realpath": { 1823 | "version": "1.0.0", 1824 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 1825 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", 1826 | "dev": true 1827 | }, 1828 | "node_modules/fsevents": { 1829 | "version": "2.3.2", 1830 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 1831 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 1832 | "dev": true, 1833 | "hasInstallScript": true, 1834 | "optional": true, 1835 | "os": [ 1836 | "darwin" 1837 | ], 1838 | "engines": { 1839 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 1840 | } 1841 | }, 1842 | "node_modules/function-bind": { 1843 | "version": "1.1.2", 1844 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", 1845 | "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", 1846 | "dev": true, 1847 | "license": "MIT", 1848 | "funding": { 1849 | "url": "https://github.com/sponsors/ljharb" 1850 | } 1851 | }, 1852 | "node_modules/functional-red-black-tree": { 1853 | "version": "1.0.1", 1854 | "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", 1855 | "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", 1856 | "dev": true 1857 | }, 1858 | "node_modules/gensync": { 1859 | "version": "1.0.0-beta.2", 1860 | "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", 1861 | "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", 1862 | "dev": true, 1863 | "engines": { 1864 | "node": ">=6.9.0" 1865 | } 1866 | }, 1867 | "node_modules/get-caller-file": { 1868 | "version": "2.0.5", 1869 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 1870 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 1871 | "dev": true, 1872 | "engines": { 1873 | "node": "6.* || 8.* || >= 10.*" 1874 | } 1875 | }, 1876 | "node_modules/get-func-name": { 1877 | "version": "2.0.2", 1878 | "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.2.tgz", 1879 | "integrity": "sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==", 1880 | "dev": true, 1881 | "engines": { 1882 | "node": "*" 1883 | } 1884 | }, 1885 | "node_modules/get-intrinsic": { 1886 | "version": "1.3.0", 1887 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", 1888 | "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", 1889 | "dev": true, 1890 | "license": "MIT", 1891 | "dependencies": { 1892 | "call-bind-apply-helpers": "^1.0.2", 1893 | "es-define-property": "^1.0.1", 1894 | "es-errors": "^1.3.0", 1895 | "es-object-atoms": "^1.1.1", 1896 | "function-bind": "^1.1.2", 1897 | "get-proto": "^1.0.1", 1898 | "gopd": "^1.2.0", 1899 | "has-symbols": "^1.1.0", 1900 | "hasown": "^2.0.2", 1901 | "math-intrinsics": "^1.1.0" 1902 | }, 1903 | "engines": { 1904 | "node": ">= 0.4" 1905 | }, 1906 | "funding": { 1907 | "url": "https://github.com/sponsors/ljharb" 1908 | } 1909 | }, 1910 | "node_modules/get-package-type": { 1911 | "version": "0.1.0", 1912 | "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", 1913 | "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", 1914 | "dev": true, 1915 | "engines": { 1916 | "node": ">=8.0.0" 1917 | } 1918 | }, 1919 | "node_modules/get-proto": { 1920 | "version": "1.0.1", 1921 | "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", 1922 | "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", 1923 | "dev": true, 1924 | "license": "MIT", 1925 | "dependencies": { 1926 | "dunder-proto": "^1.0.1", 1927 | "es-object-atoms": "^1.0.0" 1928 | }, 1929 | "engines": { 1930 | "node": ">= 0.4" 1931 | } 1932 | }, 1933 | "node_modules/glob": { 1934 | "version": "7.2.0", 1935 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", 1936 | "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", 1937 | "dev": true, 1938 | "dependencies": { 1939 | "fs.realpath": "^1.0.0", 1940 | "inflight": "^1.0.4", 1941 | "inherits": "2", 1942 | "minimatch": "^3.0.4", 1943 | "once": "^1.3.0", 1944 | "path-is-absolute": "^1.0.0" 1945 | }, 1946 | "engines": { 1947 | "node": "*" 1948 | }, 1949 | "funding": { 1950 | "url": "https://github.com/sponsors/isaacs" 1951 | } 1952 | }, 1953 | "node_modules/glob-parent": { 1954 | "version": "6.0.2", 1955 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", 1956 | "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", 1957 | "dev": true, 1958 | "dependencies": { 1959 | "is-glob": "^4.0.3" 1960 | }, 1961 | "engines": { 1962 | "node": ">=10.13.0" 1963 | } 1964 | }, 1965 | "node_modules/globals": { 1966 | "version": "13.12.1", 1967 | "resolved": "https://registry.npmjs.org/globals/-/globals-13.12.1.tgz", 1968 | "integrity": "sha512-317dFlgY2pdJZ9rspXDks7073GpDmXdfbM3vYYp0HAMKGDh1FfWPleI2ljVNLQX5M5lXcAslTcPTrOrMEFOjyw==", 1969 | "dev": true, 1970 | "dependencies": { 1971 | "type-fest": "^0.20.2" 1972 | }, 1973 | "engines": { 1974 | "node": ">=8" 1975 | }, 1976 | "funding": { 1977 | "url": "https://github.com/sponsors/sindresorhus" 1978 | } 1979 | }, 1980 | "node_modules/gopd": { 1981 | "version": "1.2.0", 1982 | "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", 1983 | "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", 1984 | "dev": true, 1985 | "license": "MIT", 1986 | "engines": { 1987 | "node": ">= 0.4" 1988 | }, 1989 | "funding": { 1990 | "url": "https://github.com/sponsors/ljharb" 1991 | } 1992 | }, 1993 | "node_modules/graceful-fs": { 1994 | "version": "4.2.9", 1995 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.9.tgz", 1996 | "integrity": "sha512-NtNxqUcXgpW2iMrfqSfR73Glt39K+BLwWsPs94yR63v45T0Wbej7eRmL5cWfwEgqXnmjQp3zaJTshdRW/qC2ZQ==", 1997 | "dev": true 1998 | }, 1999 | "node_modules/growl": { 2000 | "version": "1.10.5", 2001 | "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz", 2002 | "integrity": "sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==", 2003 | "dev": true, 2004 | "engines": { 2005 | "node": ">=4.x" 2006 | } 2007 | }, 2008 | "node_modules/has-flag": { 2009 | "version": "4.0.0", 2010 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 2011 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 2012 | "dev": true, 2013 | "engines": { 2014 | "node": ">=8" 2015 | } 2016 | }, 2017 | "node_modules/has-symbols": { 2018 | "version": "1.1.0", 2019 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", 2020 | "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", 2021 | "dev": true, 2022 | "license": "MIT", 2023 | "engines": { 2024 | "node": ">= 0.4" 2025 | }, 2026 | "funding": { 2027 | "url": "https://github.com/sponsors/ljharb" 2028 | } 2029 | }, 2030 | "node_modules/hasha": { 2031 | "version": "5.2.2", 2032 | "resolved": "https://registry.npmjs.org/hasha/-/hasha-5.2.2.tgz", 2033 | "integrity": "sha512-Hrp5vIK/xr5SkeN2onO32H0MgNZ0f17HRNH39WfL0SYUNOTZ5Lz1TJ8Pajo/87dYGEFlLMm7mIc/k/s6Bvz9HQ==", 2034 | "dev": true, 2035 | "dependencies": { 2036 | "is-stream": "^2.0.0", 2037 | "type-fest": "^0.8.0" 2038 | }, 2039 | "engines": { 2040 | "node": ">=8" 2041 | }, 2042 | "funding": { 2043 | "url": "https://github.com/sponsors/sindresorhus" 2044 | } 2045 | }, 2046 | "node_modules/hasha/node_modules/type-fest": { 2047 | "version": "0.8.1", 2048 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 2049 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", 2050 | "dev": true, 2051 | "engines": { 2052 | "node": ">=8" 2053 | } 2054 | }, 2055 | "node_modules/hasown": { 2056 | "version": "2.0.2", 2057 | "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", 2058 | "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", 2059 | "dev": true, 2060 | "license": "MIT", 2061 | "dependencies": { 2062 | "function-bind": "^1.1.2" 2063 | }, 2064 | "engines": { 2065 | "node": ">= 0.4" 2066 | } 2067 | }, 2068 | "node_modules/he": { 2069 | "version": "1.2.0", 2070 | "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", 2071 | "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", 2072 | "dev": true, 2073 | "bin": { 2074 | "he": "bin/he" 2075 | } 2076 | }, 2077 | "node_modules/hexoid": { 2078 | "version": "1.0.0", 2079 | "resolved": "https://registry.npmjs.org/hexoid/-/hexoid-1.0.0.tgz", 2080 | "integrity": "sha512-QFLV0taWQOZtvIRIAdBChesmogZrtuXvVWsFHZTk2SU+anspqZ2vMnoLg7IE1+Uk16N19APic1BuF8bC8c2m5g==", 2081 | "dev": true, 2082 | "engines": { 2083 | "node": ">=8" 2084 | } 2085 | }, 2086 | "node_modules/html-escaper": { 2087 | "version": "2.0.2", 2088 | "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", 2089 | "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", 2090 | "dev": true 2091 | }, 2092 | "node_modules/http-errors": { 2093 | "version": "2.0.0", 2094 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", 2095 | "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", 2096 | "dev": true, 2097 | "license": "MIT", 2098 | "dependencies": { 2099 | "depd": "2.0.0", 2100 | "inherits": "2.0.4", 2101 | "setprototypeof": "1.2.0", 2102 | "statuses": "2.0.1", 2103 | "toidentifier": "1.0.1" 2104 | }, 2105 | "engines": { 2106 | "node": ">= 0.8" 2107 | } 2108 | }, 2109 | "node_modules/iconv-lite": { 2110 | "version": "0.4.24", 2111 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 2112 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 2113 | "dev": true, 2114 | "license": "MIT", 2115 | "dependencies": { 2116 | "safer-buffer": ">= 2.1.2 < 3" 2117 | }, 2118 | "engines": { 2119 | "node": ">=0.10.0" 2120 | } 2121 | }, 2122 | "node_modules/ignore": { 2123 | "version": "5.2.0", 2124 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.0.tgz", 2125 | "integrity": "sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ==", 2126 | "dev": true, 2127 | "engines": { 2128 | "node": ">= 4" 2129 | } 2130 | }, 2131 | "node_modules/import-fresh": { 2132 | "version": "3.3.0", 2133 | "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", 2134 | "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", 2135 | "dev": true, 2136 | "dependencies": { 2137 | "parent-module": "^1.0.0", 2138 | "resolve-from": "^4.0.0" 2139 | }, 2140 | "engines": { 2141 | "node": ">=6" 2142 | }, 2143 | "funding": { 2144 | "url": "https://github.com/sponsors/sindresorhus" 2145 | } 2146 | }, 2147 | "node_modules/imurmurhash": { 2148 | "version": "0.1.4", 2149 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 2150 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 2151 | "dev": true, 2152 | "engines": { 2153 | "node": ">=0.8.19" 2154 | } 2155 | }, 2156 | "node_modules/indent-string": { 2157 | "version": "4.0.0", 2158 | "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", 2159 | "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", 2160 | "dev": true, 2161 | "engines": { 2162 | "node": ">=8" 2163 | } 2164 | }, 2165 | "node_modules/inflight": { 2166 | "version": "1.0.6", 2167 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 2168 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 2169 | "dev": true, 2170 | "dependencies": { 2171 | "once": "^1.3.0", 2172 | "wrappy": "1" 2173 | } 2174 | }, 2175 | "node_modules/inherits": { 2176 | "version": "2.0.4", 2177 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 2178 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 2179 | "dev": true 2180 | }, 2181 | "node_modules/ipaddr.js": { 2182 | "version": "1.9.1", 2183 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 2184 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", 2185 | "dev": true, 2186 | "engines": { 2187 | "node": ">= 0.10" 2188 | } 2189 | }, 2190 | "node_modules/is-binary-path": { 2191 | "version": "2.1.0", 2192 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 2193 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 2194 | "dev": true, 2195 | "dependencies": { 2196 | "binary-extensions": "^2.0.0" 2197 | }, 2198 | "engines": { 2199 | "node": ">=8" 2200 | } 2201 | }, 2202 | "node_modules/is-extglob": { 2203 | "version": "2.1.1", 2204 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 2205 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 2206 | "dev": true, 2207 | "engines": { 2208 | "node": ">=0.10.0" 2209 | } 2210 | }, 2211 | "node_modules/is-fullwidth-code-point": { 2212 | "version": "3.0.0", 2213 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 2214 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 2215 | "dev": true, 2216 | "engines": { 2217 | "node": ">=8" 2218 | } 2219 | }, 2220 | "node_modules/is-glob": { 2221 | "version": "4.0.3", 2222 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 2223 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 2224 | "dev": true, 2225 | "dependencies": { 2226 | "is-extglob": "^2.1.1" 2227 | }, 2228 | "engines": { 2229 | "node": ">=0.10.0" 2230 | } 2231 | }, 2232 | "node_modules/is-number": { 2233 | "version": "7.0.0", 2234 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 2235 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 2236 | "dev": true, 2237 | "license": "MIT", 2238 | "engines": { 2239 | "node": ">=0.12.0" 2240 | } 2241 | }, 2242 | "node_modules/is-plain-obj": { 2243 | "version": "2.1.0", 2244 | "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-2.1.0.tgz", 2245 | "integrity": "sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==", 2246 | "dev": true, 2247 | "engines": { 2248 | "node": ">=8" 2249 | } 2250 | }, 2251 | "node_modules/is-stream": { 2252 | "version": "2.0.1", 2253 | "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", 2254 | "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", 2255 | "dev": true, 2256 | "engines": { 2257 | "node": ">=8" 2258 | }, 2259 | "funding": { 2260 | "url": "https://github.com/sponsors/sindresorhus" 2261 | } 2262 | }, 2263 | "node_modules/is-typedarray": { 2264 | "version": "1.0.0", 2265 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 2266 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", 2267 | "dev": true 2268 | }, 2269 | "node_modules/is-unicode-supported": { 2270 | "version": "0.1.0", 2271 | "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", 2272 | "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==", 2273 | "dev": true, 2274 | "engines": { 2275 | "node": ">=10" 2276 | }, 2277 | "funding": { 2278 | "url": "https://github.com/sponsors/sindresorhus" 2279 | } 2280 | }, 2281 | "node_modules/is-windows": { 2282 | "version": "1.0.2", 2283 | "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", 2284 | "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", 2285 | "dev": true, 2286 | "engines": { 2287 | "node": ">=0.10.0" 2288 | } 2289 | }, 2290 | "node_modules/isexe": { 2291 | "version": "2.0.0", 2292 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 2293 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 2294 | "dev": true 2295 | }, 2296 | "node_modules/istanbul-lib-coverage": { 2297 | "version": "3.2.0", 2298 | "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.0.tgz", 2299 | "integrity": "sha512-eOeJ5BHCmHYvQK7xt9GkdHuzuCGS1Y6g9Gvnx3Ym33fz/HpLRYxiS0wHNr+m/MBC8B647Xt608vCDEvhl9c6Mw==", 2300 | "dev": true, 2301 | "engines": { 2302 | "node": ">=8" 2303 | } 2304 | }, 2305 | "node_modules/istanbul-lib-hook": { 2306 | "version": "3.0.0", 2307 | "resolved": "https://registry.npmjs.org/istanbul-lib-hook/-/istanbul-lib-hook-3.0.0.tgz", 2308 | "integrity": "sha512-Pt/uge1Q9s+5VAZ+pCo16TYMWPBIl+oaNIjgLQxcX0itS6ueeaA+pEfThZpH8WxhFgCiEb8sAJY6MdUKgiIWaQ==", 2309 | "dev": true, 2310 | "dependencies": { 2311 | "append-transform": "^2.0.0" 2312 | }, 2313 | "engines": { 2314 | "node": ">=8" 2315 | } 2316 | }, 2317 | "node_modules/istanbul-lib-instrument": { 2318 | "version": "4.0.3", 2319 | "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-4.0.3.tgz", 2320 | "integrity": "sha512-BXgQl9kf4WTCPCCpmFGoJkz/+uhvm7h7PFKUYxh7qarQd3ER33vHG//qaE8eN25l07YqZPpHXU9I09l/RD5aGQ==", 2321 | "dev": true, 2322 | "dependencies": { 2323 | "@babel/core": "^7.7.5", 2324 | "@istanbuljs/schema": "^0.1.2", 2325 | "istanbul-lib-coverage": "^3.0.0", 2326 | "semver": "^6.3.0" 2327 | }, 2328 | "engines": { 2329 | "node": ">=8" 2330 | } 2331 | }, 2332 | "node_modules/istanbul-lib-processinfo": { 2333 | "version": "2.0.2", 2334 | "resolved": "https://registry.npmjs.org/istanbul-lib-processinfo/-/istanbul-lib-processinfo-2.0.2.tgz", 2335 | "integrity": "sha512-kOwpa7z9hme+IBPZMzQ5vdQj8srYgAtaRqeI48NGmAQ+/5yKiHLV0QbYqQpxsdEF0+w14SoB8YbnHKcXE2KnYw==", 2336 | "dev": true, 2337 | "dependencies": { 2338 | "archy": "^1.0.0", 2339 | "cross-spawn": "^7.0.0", 2340 | "istanbul-lib-coverage": "^3.0.0-alpha.1", 2341 | "make-dir": "^3.0.0", 2342 | "p-map": "^3.0.0", 2343 | "rimraf": "^3.0.0", 2344 | "uuid": "^3.3.3" 2345 | }, 2346 | "engines": { 2347 | "node": ">=8" 2348 | } 2349 | }, 2350 | "node_modules/istanbul-lib-report": { 2351 | "version": "3.0.0", 2352 | "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", 2353 | "integrity": "sha512-wcdi+uAKzfiGT2abPpKZ0hSU1rGQjUQnLvtY5MpQ7QCTahD3VODhcu4wcfY1YtkGaDD5yuydOLINXsfbus9ROw==", 2354 | "dev": true, 2355 | "dependencies": { 2356 | "istanbul-lib-coverage": "^3.0.0", 2357 | "make-dir": "^3.0.0", 2358 | "supports-color": "^7.1.0" 2359 | }, 2360 | "engines": { 2361 | "node": ">=8" 2362 | } 2363 | }, 2364 | "node_modules/istanbul-lib-source-maps": { 2365 | "version": "4.0.1", 2366 | "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", 2367 | "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", 2368 | "dev": true, 2369 | "dependencies": { 2370 | "debug": "^4.1.1", 2371 | "istanbul-lib-coverage": "^3.0.0", 2372 | "source-map": "^0.6.1" 2373 | }, 2374 | "engines": { 2375 | "node": ">=10" 2376 | } 2377 | }, 2378 | "node_modules/istanbul-lib-source-maps/node_modules/source-map": { 2379 | "version": "0.6.1", 2380 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 2381 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", 2382 | "dev": true, 2383 | "engines": { 2384 | "node": ">=0.10.0" 2385 | } 2386 | }, 2387 | "node_modules/istanbul-reports": { 2388 | "version": "3.1.4", 2389 | "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.4.tgz", 2390 | "integrity": "sha512-r1/DshN4KSE7xWEknZLLLLDn5CJybV3nw01VTkp6D5jzLuELlcbudfj/eSQFvrKsJuTVCGnePO7ho82Nw9zzfw==", 2391 | "dev": true, 2392 | "dependencies": { 2393 | "html-escaper": "^2.0.0", 2394 | "istanbul-lib-report": "^3.0.0" 2395 | }, 2396 | "engines": { 2397 | "node": ">=8" 2398 | } 2399 | }, 2400 | "node_modules/js-tokens": { 2401 | "version": "4.0.0", 2402 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 2403 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 2404 | "dev": true, 2405 | "license": "MIT" 2406 | }, 2407 | "node_modules/js-yaml": { 2408 | "version": "4.1.0", 2409 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", 2410 | "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", 2411 | "dev": true, 2412 | "dependencies": { 2413 | "argparse": "^2.0.1" 2414 | }, 2415 | "bin": { 2416 | "js-yaml": "bin/js-yaml.js" 2417 | } 2418 | }, 2419 | "node_modules/jsesc": { 2420 | "version": "2.5.2", 2421 | "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", 2422 | "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", 2423 | "dev": true, 2424 | "bin": { 2425 | "jsesc": "bin/jsesc" 2426 | }, 2427 | "engines": { 2428 | "node": ">=4" 2429 | } 2430 | }, 2431 | "node_modules/json-schema-traverse": { 2432 | "version": "0.4.1", 2433 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", 2434 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", 2435 | "dev": true 2436 | }, 2437 | "node_modules/json-stable-stringify-without-jsonify": { 2438 | "version": "1.0.1", 2439 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", 2440 | "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", 2441 | "dev": true 2442 | }, 2443 | "node_modules/json5": { 2444 | "version": "2.2.3", 2445 | "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", 2446 | "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", 2447 | "dev": true, 2448 | "bin": { 2449 | "json5": "lib/cli.js" 2450 | }, 2451 | "engines": { 2452 | "node": ">=6" 2453 | } 2454 | }, 2455 | "node_modules/levn": { 2456 | "version": "0.4.1", 2457 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", 2458 | "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", 2459 | "dev": true, 2460 | "dependencies": { 2461 | "prelude-ls": "^1.2.1", 2462 | "type-check": "~0.4.0" 2463 | }, 2464 | "engines": { 2465 | "node": ">= 0.8.0" 2466 | } 2467 | }, 2468 | "node_modules/locate-path": { 2469 | "version": "5.0.0", 2470 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", 2471 | "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", 2472 | "dev": true, 2473 | "dependencies": { 2474 | "p-locate": "^4.1.0" 2475 | }, 2476 | "engines": { 2477 | "node": ">=8" 2478 | } 2479 | }, 2480 | "node_modules/lodash.flattendeep": { 2481 | "version": "4.4.0", 2482 | "resolved": "https://registry.npmjs.org/lodash.flattendeep/-/lodash.flattendeep-4.4.0.tgz", 2483 | "integrity": "sha1-+wMJF/hqMTTlvJvsDWngAT3f7bI=", 2484 | "dev": true 2485 | }, 2486 | "node_modules/lodash.merge": { 2487 | "version": "4.6.2", 2488 | "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", 2489 | "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", 2490 | "dev": true 2491 | }, 2492 | "node_modules/log-symbols": { 2493 | "version": "4.1.0", 2494 | "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz", 2495 | "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", 2496 | "dev": true, 2497 | "dependencies": { 2498 | "chalk": "^4.1.0", 2499 | "is-unicode-supported": "^0.1.0" 2500 | }, 2501 | "engines": { 2502 | "node": ">=10" 2503 | }, 2504 | "funding": { 2505 | "url": "https://github.com/sponsors/sindresorhus" 2506 | } 2507 | }, 2508 | "node_modules/lru-cache": { 2509 | "version": "6.0.0", 2510 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", 2511 | "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", 2512 | "dev": true, 2513 | "dependencies": { 2514 | "yallist": "^4.0.0" 2515 | }, 2516 | "engines": { 2517 | "node": ">=10" 2518 | } 2519 | }, 2520 | "node_modules/make-dir": { 2521 | "version": "3.1.0", 2522 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 2523 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 2524 | "dev": true, 2525 | "dependencies": { 2526 | "semver": "^6.0.0" 2527 | }, 2528 | "engines": { 2529 | "node": ">=8" 2530 | }, 2531 | "funding": { 2532 | "url": "https://github.com/sponsors/sindresorhus" 2533 | } 2534 | }, 2535 | "node_modules/math-intrinsics": { 2536 | "version": "1.1.0", 2537 | "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", 2538 | "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", 2539 | "dev": true, 2540 | "license": "MIT", 2541 | "engines": { 2542 | "node": ">= 0.4" 2543 | } 2544 | }, 2545 | "node_modules/media-typer": { 2546 | "version": "0.3.0", 2547 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 2548 | "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", 2549 | "dev": true, 2550 | "license": "MIT", 2551 | "engines": { 2552 | "node": ">= 0.6" 2553 | } 2554 | }, 2555 | "node_modules/merge-descriptors": { 2556 | "version": "1.0.3", 2557 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", 2558 | "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", 2559 | "dev": true, 2560 | "license": "MIT", 2561 | "funding": { 2562 | "url": "https://github.com/sponsors/sindresorhus" 2563 | } 2564 | }, 2565 | "node_modules/methods": { 2566 | "version": "1.1.2", 2567 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 2568 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=", 2569 | "dev": true, 2570 | "engines": { 2571 | "node": ">= 0.6" 2572 | } 2573 | }, 2574 | "node_modules/mime": { 2575 | "version": "1.6.0", 2576 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 2577 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", 2578 | "dev": true, 2579 | "license": "MIT", 2580 | "bin": { 2581 | "mime": "cli.js" 2582 | }, 2583 | "engines": { 2584 | "node": ">=4" 2585 | } 2586 | }, 2587 | "node_modules/mime-db": { 2588 | "version": "1.40.0", 2589 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.40.0.tgz", 2590 | "integrity": "sha512-jYdeOMPy9vnxEqFRRo6ZvTZ8d9oPb+k18PKoYNYUe2stVEBPPwsln/qWzdbmaIvnhZ9v2P+CuecK+fpUfsV2mA==", 2591 | "dev": true, 2592 | "engines": { 2593 | "node": ">= 0.6" 2594 | } 2595 | }, 2596 | "node_modules/mime-types": { 2597 | "version": "2.1.24", 2598 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.24.tgz", 2599 | "integrity": "sha512-WaFHS3MCl5fapm3oLxU4eYDw77IQM2ACcxQ9RIxfaC3ooc6PFuBMGZZsYpvoXS5D5QTWPieo1jjLdAm3TBP3cQ==", 2600 | "dev": true, 2601 | "dependencies": { 2602 | "mime-db": "1.40.0" 2603 | }, 2604 | "engines": { 2605 | "node": ">= 0.6" 2606 | } 2607 | }, 2608 | "node_modules/minimatch": { 2609 | "version": "3.1.2", 2610 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 2611 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 2612 | "dev": true, 2613 | "dependencies": { 2614 | "brace-expansion": "^1.1.7" 2615 | }, 2616 | "engines": { 2617 | "node": "*" 2618 | } 2619 | }, 2620 | "node_modules/mocha": { 2621 | "version": "9.2.2", 2622 | "resolved": "https://registry.npmjs.org/mocha/-/mocha-9.2.2.tgz", 2623 | "integrity": "sha512-L6XC3EdwT6YrIk0yXpavvLkn8h+EU+Y5UcCHKECyMbdUIxyMuZj4bX4U9e1nvnvUUvQVsV2VHQr5zLdcUkhW/g==", 2624 | "dev": true, 2625 | "dependencies": { 2626 | "@ungap/promise-all-settled": "1.1.2", 2627 | "ansi-colors": "4.1.1", 2628 | "browser-stdout": "1.3.1", 2629 | "chokidar": "3.5.3", 2630 | "debug": "4.3.3", 2631 | "diff": "5.0.0", 2632 | "escape-string-regexp": "4.0.0", 2633 | "find-up": "5.0.0", 2634 | "glob": "7.2.0", 2635 | "growl": "1.10.5", 2636 | "he": "1.2.0", 2637 | "js-yaml": "4.1.0", 2638 | "log-symbols": "4.1.0", 2639 | "minimatch": "4.2.1", 2640 | "ms": "2.1.3", 2641 | "nanoid": "3.3.1", 2642 | "serialize-javascript": "6.0.0", 2643 | "strip-json-comments": "3.1.1", 2644 | "supports-color": "8.1.1", 2645 | "which": "2.0.2", 2646 | "workerpool": "6.2.0", 2647 | "yargs": "16.2.0", 2648 | "yargs-parser": "20.2.4", 2649 | "yargs-unparser": "2.0.0" 2650 | }, 2651 | "bin": { 2652 | "_mocha": "bin/_mocha", 2653 | "mocha": "bin/mocha" 2654 | }, 2655 | "engines": { 2656 | "node": ">= 12.0.0" 2657 | }, 2658 | "funding": { 2659 | "type": "opencollective", 2660 | "url": "https://opencollective.com/mochajs" 2661 | } 2662 | }, 2663 | "node_modules/mocha/node_modules/cliui": { 2664 | "version": "7.0.4", 2665 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", 2666 | "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", 2667 | "dev": true, 2668 | "dependencies": { 2669 | "string-width": "^4.2.0", 2670 | "strip-ansi": "^6.0.0", 2671 | "wrap-ansi": "^7.0.0" 2672 | } 2673 | }, 2674 | "node_modules/mocha/node_modules/debug": { 2675 | "version": "4.3.3", 2676 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.3.tgz", 2677 | "integrity": "sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q==", 2678 | "dev": true, 2679 | "dependencies": { 2680 | "ms": "2.1.2" 2681 | }, 2682 | "engines": { 2683 | "node": ">=6.0" 2684 | }, 2685 | "peerDependenciesMeta": { 2686 | "supports-color": { 2687 | "optional": true 2688 | } 2689 | } 2690 | }, 2691 | "node_modules/mocha/node_modules/debug/node_modules/ms": { 2692 | "version": "2.1.2", 2693 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 2694 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 2695 | "dev": true 2696 | }, 2697 | "node_modules/mocha/node_modules/find-up": { 2698 | "version": "5.0.0", 2699 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", 2700 | "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", 2701 | "dev": true, 2702 | "dependencies": { 2703 | "locate-path": "^6.0.0", 2704 | "path-exists": "^4.0.0" 2705 | }, 2706 | "engines": { 2707 | "node": ">=10" 2708 | }, 2709 | "funding": { 2710 | "url": "https://github.com/sponsors/sindresorhus" 2711 | } 2712 | }, 2713 | "node_modules/mocha/node_modules/locate-path": { 2714 | "version": "6.0.0", 2715 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", 2716 | "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", 2717 | "dev": true, 2718 | "dependencies": { 2719 | "p-locate": "^5.0.0" 2720 | }, 2721 | "engines": { 2722 | "node": ">=10" 2723 | }, 2724 | "funding": { 2725 | "url": "https://github.com/sponsors/sindresorhus" 2726 | } 2727 | }, 2728 | "node_modules/mocha/node_modules/minimatch": { 2729 | "version": "4.2.1", 2730 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-4.2.1.tgz", 2731 | "integrity": "sha512-9Uq1ChtSZO+Mxa/CL1eGizn2vRn3MlLgzhT0Iz8zaY8NdvxvB0d5QdPFmCKf7JKA9Lerx5vRrnwO03jsSfGG9g==", 2732 | "dev": true, 2733 | "dependencies": { 2734 | "brace-expansion": "^1.1.7" 2735 | }, 2736 | "engines": { 2737 | "node": ">=10" 2738 | } 2739 | }, 2740 | "node_modules/mocha/node_modules/ms": { 2741 | "version": "2.1.3", 2742 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 2743 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 2744 | "dev": true 2745 | }, 2746 | "node_modules/mocha/node_modules/p-limit": { 2747 | "version": "3.1.0", 2748 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", 2749 | "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", 2750 | "dev": true, 2751 | "dependencies": { 2752 | "yocto-queue": "^0.1.0" 2753 | }, 2754 | "engines": { 2755 | "node": ">=10" 2756 | }, 2757 | "funding": { 2758 | "url": "https://github.com/sponsors/sindresorhus" 2759 | } 2760 | }, 2761 | "node_modules/mocha/node_modules/p-locate": { 2762 | "version": "5.0.0", 2763 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", 2764 | "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", 2765 | "dev": true, 2766 | "dependencies": { 2767 | "p-limit": "^3.0.2" 2768 | }, 2769 | "engines": { 2770 | "node": ">=10" 2771 | }, 2772 | "funding": { 2773 | "url": "https://github.com/sponsors/sindresorhus" 2774 | } 2775 | }, 2776 | "node_modules/mocha/node_modules/supports-color": { 2777 | "version": "8.1.1", 2778 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", 2779 | "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", 2780 | "dev": true, 2781 | "dependencies": { 2782 | "has-flag": "^4.0.0" 2783 | }, 2784 | "engines": { 2785 | "node": ">=10" 2786 | }, 2787 | "funding": { 2788 | "url": "https://github.com/chalk/supports-color?sponsor=1" 2789 | } 2790 | }, 2791 | "node_modules/mocha/node_modules/wrap-ansi": { 2792 | "version": "7.0.0", 2793 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", 2794 | "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", 2795 | "dev": true, 2796 | "dependencies": { 2797 | "ansi-styles": "^4.0.0", 2798 | "string-width": "^4.1.0", 2799 | "strip-ansi": "^6.0.0" 2800 | }, 2801 | "engines": { 2802 | "node": ">=10" 2803 | }, 2804 | "funding": { 2805 | "url": "https://github.com/chalk/wrap-ansi?sponsor=1" 2806 | } 2807 | }, 2808 | "node_modules/mocha/node_modules/y18n": { 2809 | "version": "5.0.8", 2810 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", 2811 | "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", 2812 | "dev": true, 2813 | "engines": { 2814 | "node": ">=10" 2815 | } 2816 | }, 2817 | "node_modules/mocha/node_modules/yargs": { 2818 | "version": "16.2.0", 2819 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", 2820 | "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", 2821 | "dev": true, 2822 | "dependencies": { 2823 | "cliui": "^7.0.2", 2824 | "escalade": "^3.1.1", 2825 | "get-caller-file": "^2.0.5", 2826 | "require-directory": "^2.1.1", 2827 | "string-width": "^4.2.0", 2828 | "y18n": "^5.0.5", 2829 | "yargs-parser": "^20.2.2" 2830 | }, 2831 | "engines": { 2832 | "node": ">=10" 2833 | } 2834 | }, 2835 | "node_modules/mocha/node_modules/yargs-parser": { 2836 | "version": "20.2.4", 2837 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.4.tgz", 2838 | "integrity": "sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==", 2839 | "dev": true, 2840 | "engines": { 2841 | "node": ">=10" 2842 | } 2843 | }, 2844 | "node_modules/ms": { 2845 | "version": "2.1.2", 2846 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 2847 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 2848 | "dev": true 2849 | }, 2850 | "node_modules/nanoid": { 2851 | "version": "3.3.1", 2852 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.1.tgz", 2853 | "integrity": "sha512-n6Vs/3KGyxPQd6uO0eH4Bv0ojGSUvuLlIHtC3Y0kEO23YRge8H9x1GCzLn28YX0H66pMkxuaeESFq4tKISKwdw==", 2854 | "dev": true, 2855 | "bin": { 2856 | "nanoid": "bin/nanoid.cjs" 2857 | }, 2858 | "engines": { 2859 | "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" 2860 | } 2861 | }, 2862 | "node_modules/natural-compare": { 2863 | "version": "1.4.0", 2864 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", 2865 | "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", 2866 | "dev": true 2867 | }, 2868 | "node_modules/negotiator": { 2869 | "version": "0.6.3", 2870 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", 2871 | "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", 2872 | "dev": true, 2873 | "engines": { 2874 | "node": ">= 0.6" 2875 | } 2876 | }, 2877 | "node_modules/node-preload": { 2878 | "version": "0.2.1", 2879 | "resolved": "https://registry.npmjs.org/node-preload/-/node-preload-0.2.1.tgz", 2880 | "integrity": "sha512-RM5oyBy45cLEoHqCeh+MNuFAxO0vTFBLskvQbOKnEE7YTTSN4tbN8QWDIPQ6L+WvKsB/qLEGpYe2ZZ9d4W9OIQ==", 2881 | "dev": true, 2882 | "dependencies": { 2883 | "process-on-spawn": "^1.0.0" 2884 | }, 2885 | "engines": { 2886 | "node": ">=8" 2887 | } 2888 | }, 2889 | "node_modules/node-releases": { 2890 | "version": "2.0.2", 2891 | "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.2.tgz", 2892 | "integrity": "sha512-XxYDdcQ6eKqp/YjI+tb2C5WM2LgjnZrfYg4vgQt49EK268b6gYCHsBLrK2qvJo4FmCtqmKezb0WZFK4fkrZNsg==", 2893 | "dev": true 2894 | }, 2895 | "node_modules/normalize-path": { 2896 | "version": "3.0.0", 2897 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 2898 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 2899 | "dev": true, 2900 | "engines": { 2901 | "node": ">=0.10.0" 2902 | } 2903 | }, 2904 | "node_modules/nyc": { 2905 | "version": "15.1.0", 2906 | "resolved": "https://registry.npmjs.org/nyc/-/nyc-15.1.0.tgz", 2907 | "integrity": "sha512-jMW04n9SxKdKi1ZMGhvUTHBN0EICCRkHemEoE5jm6mTYcqcdas0ATzgUgejlQUHMvpnOZqGB5Xxsv9KxJW1j8A==", 2908 | "dev": true, 2909 | "dependencies": { 2910 | "@istanbuljs/load-nyc-config": "^1.0.0", 2911 | "@istanbuljs/schema": "^0.1.2", 2912 | "caching-transform": "^4.0.0", 2913 | "convert-source-map": "^1.7.0", 2914 | "decamelize": "^1.2.0", 2915 | "find-cache-dir": "^3.2.0", 2916 | "find-up": "^4.1.0", 2917 | "foreground-child": "^2.0.0", 2918 | "get-package-type": "^0.1.0", 2919 | "glob": "^7.1.6", 2920 | "istanbul-lib-coverage": "^3.0.0", 2921 | "istanbul-lib-hook": "^3.0.0", 2922 | "istanbul-lib-instrument": "^4.0.0", 2923 | "istanbul-lib-processinfo": "^2.0.2", 2924 | "istanbul-lib-report": "^3.0.0", 2925 | "istanbul-lib-source-maps": "^4.0.0", 2926 | "istanbul-reports": "^3.0.2", 2927 | "make-dir": "^3.0.0", 2928 | "node-preload": "^0.2.1", 2929 | "p-map": "^3.0.0", 2930 | "process-on-spawn": "^1.0.0", 2931 | "resolve-from": "^5.0.0", 2932 | "rimraf": "^3.0.0", 2933 | "signal-exit": "^3.0.2", 2934 | "spawn-wrap": "^2.0.0", 2935 | "test-exclude": "^6.0.0", 2936 | "yargs": "^15.0.2" 2937 | }, 2938 | "bin": { 2939 | "nyc": "bin/nyc.js" 2940 | }, 2941 | "engines": { 2942 | "node": ">=8.9" 2943 | } 2944 | }, 2945 | "node_modules/nyc/node_modules/resolve-from": { 2946 | "version": "5.0.0", 2947 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", 2948 | "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", 2949 | "dev": true, 2950 | "engines": { 2951 | "node": ">=8" 2952 | } 2953 | }, 2954 | "node_modules/object-inspect": { 2955 | "version": "1.13.4", 2956 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", 2957 | "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", 2958 | "dev": true, 2959 | "license": "MIT", 2960 | "engines": { 2961 | "node": ">= 0.4" 2962 | }, 2963 | "funding": { 2964 | "url": "https://github.com/sponsors/ljharb" 2965 | } 2966 | }, 2967 | "node_modules/on-finished": { 2968 | "version": "2.4.1", 2969 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", 2970 | "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", 2971 | "dev": true, 2972 | "license": "MIT", 2973 | "dependencies": { 2974 | "ee-first": "1.1.1" 2975 | }, 2976 | "engines": { 2977 | "node": ">= 0.8" 2978 | } 2979 | }, 2980 | "node_modules/once": { 2981 | "version": "1.4.0", 2982 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 2983 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 2984 | "dev": true, 2985 | "dependencies": { 2986 | "wrappy": "1" 2987 | } 2988 | }, 2989 | "node_modules/optionator": { 2990 | "version": "0.9.1", 2991 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.1.tgz", 2992 | "integrity": "sha512-74RlY5FCnhq4jRxVUPKDaRwrVNXMqsGsiW6AJw4XK8hmtm10wC0ypZBLw5IIp85NZMr91+qd1RvvENwg7jjRFw==", 2993 | "dev": true, 2994 | "dependencies": { 2995 | "deep-is": "^0.1.3", 2996 | "fast-levenshtein": "^2.0.6", 2997 | "levn": "^0.4.1", 2998 | "prelude-ls": "^1.2.1", 2999 | "type-check": "^0.4.0", 3000 | "word-wrap": "^1.2.3" 3001 | }, 3002 | "engines": { 3003 | "node": ">= 0.8.0" 3004 | } 3005 | }, 3006 | "node_modules/p-limit": { 3007 | "version": "2.3.0", 3008 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", 3009 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", 3010 | "dev": true, 3011 | "dependencies": { 3012 | "p-try": "^2.0.0" 3013 | }, 3014 | "engines": { 3015 | "node": ">=6" 3016 | }, 3017 | "funding": { 3018 | "url": "https://github.com/sponsors/sindresorhus" 3019 | } 3020 | }, 3021 | "node_modules/p-locate": { 3022 | "version": "4.1.0", 3023 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", 3024 | "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", 3025 | "dev": true, 3026 | "dependencies": { 3027 | "p-limit": "^2.2.0" 3028 | }, 3029 | "engines": { 3030 | "node": ">=8" 3031 | } 3032 | }, 3033 | "node_modules/p-map": { 3034 | "version": "3.0.0", 3035 | "resolved": "https://registry.npmjs.org/p-map/-/p-map-3.0.0.tgz", 3036 | "integrity": "sha512-d3qXVTF/s+W+CdJ5A29wywV2n8CQQYahlgz2bFiA+4eVNJbHJodPZ+/gXwPGh0bOqA+j8S+6+ckmvLGPk1QpxQ==", 3037 | "dev": true, 3038 | "dependencies": { 3039 | "aggregate-error": "^3.0.0" 3040 | }, 3041 | "engines": { 3042 | "node": ">=8" 3043 | } 3044 | }, 3045 | "node_modules/p-try": { 3046 | "version": "2.2.0", 3047 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", 3048 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", 3049 | "dev": true, 3050 | "engines": { 3051 | "node": ">=6" 3052 | } 3053 | }, 3054 | "node_modules/package-hash": { 3055 | "version": "4.0.0", 3056 | "resolved": "https://registry.npmjs.org/package-hash/-/package-hash-4.0.0.tgz", 3057 | "integrity": "sha512-whdkPIooSu/bASggZ96BWVvZTRMOFxnyUG5PnTSGKoJE2gd5mbVNmR2Nj20QFzxYYgAXpoqC+AiXzl+UMRh7zQ==", 3058 | "dev": true, 3059 | "dependencies": { 3060 | "graceful-fs": "^4.1.15", 3061 | "hasha": "^5.0.0", 3062 | "lodash.flattendeep": "^4.4.0", 3063 | "release-zalgo": "^1.0.0" 3064 | }, 3065 | "engines": { 3066 | "node": ">=8" 3067 | } 3068 | }, 3069 | "node_modules/parent-module": { 3070 | "version": "1.0.1", 3071 | "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", 3072 | "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", 3073 | "dev": true, 3074 | "dependencies": { 3075 | "callsites": "^3.0.0" 3076 | }, 3077 | "engines": { 3078 | "node": ">=6" 3079 | } 3080 | }, 3081 | "node_modules/parseurl": { 3082 | "version": "1.3.3", 3083 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 3084 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", 3085 | "dev": true, 3086 | "license": "MIT", 3087 | "engines": { 3088 | "node": ">= 0.8" 3089 | } 3090 | }, 3091 | "node_modules/passport": { 3092 | "version": "0.6.0", 3093 | "resolved": "https://registry.npmjs.org/passport/-/passport-0.6.0.tgz", 3094 | "integrity": "sha512-0fe+p3ZnrWRW74fe8+SvCyf4a3Pb2/h7gFkQ8yTJpAO50gDzlfjZUZTO1k5Eg9kUct22OxHLqDZoKUWRHOh9ug==", 3095 | "dev": true, 3096 | "dependencies": { 3097 | "passport-strategy": "1.x.x", 3098 | "pause": "0.0.1", 3099 | "utils-merge": "^1.0.1" 3100 | }, 3101 | "engines": { 3102 | "node": ">= 0.4.0" 3103 | }, 3104 | "funding": { 3105 | "type": "github", 3106 | "url": "https://github.com/sponsors/jaredhanson" 3107 | } 3108 | }, 3109 | "node_modules/passport-strategy": { 3110 | "version": "1.0.0", 3111 | "resolved": "https://registry.npmjs.org/passport-strategy/-/passport-strategy-1.0.0.tgz", 3112 | "integrity": "sha1-tVOaqPwiWj0a0XlHbd8ja0QPUuQ=", 3113 | "dev": true, 3114 | "engines": { 3115 | "node": ">= 0.4.0" 3116 | } 3117 | }, 3118 | "node_modules/path-exists": { 3119 | "version": "4.0.0", 3120 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 3121 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 3122 | "dev": true, 3123 | "engines": { 3124 | "node": ">=8" 3125 | } 3126 | }, 3127 | "node_modules/path-is-absolute": { 3128 | "version": "1.0.1", 3129 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 3130 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 3131 | "dev": true, 3132 | "engines": { 3133 | "node": ">=0.10.0" 3134 | } 3135 | }, 3136 | "node_modules/path-key": { 3137 | "version": "3.1.1", 3138 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", 3139 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", 3140 | "dev": true, 3141 | "engines": { 3142 | "node": ">=8" 3143 | } 3144 | }, 3145 | "node_modules/path-to-regexp": { 3146 | "version": "0.1.12", 3147 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.12.tgz", 3148 | "integrity": "sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==", 3149 | "dev": true, 3150 | "license": "MIT" 3151 | }, 3152 | "node_modules/pathval": { 3153 | "version": "1.1.1", 3154 | "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.1.tgz", 3155 | "integrity": "sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==", 3156 | "dev": true, 3157 | "engines": { 3158 | "node": "*" 3159 | } 3160 | }, 3161 | "node_modules/pause": { 3162 | "version": "0.0.1", 3163 | "resolved": "https://registry.npmjs.org/pause/-/pause-0.0.1.tgz", 3164 | "integrity": "sha512-KG8UEiEVkR3wGEb4m5yZkVCzigAD+cVEJck2CzYZO37ZGJfctvVptVO192MwrtPhzONn6go8ylnOdMhKqi4nfg==", 3165 | "dev": true 3166 | }, 3167 | "node_modules/picocolors": { 3168 | "version": "1.0.0", 3169 | "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", 3170 | "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", 3171 | "dev": true 3172 | }, 3173 | "node_modules/picomatch": { 3174 | "version": "2.3.1", 3175 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 3176 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 3177 | "dev": true, 3178 | "engines": { 3179 | "node": ">=8.6" 3180 | }, 3181 | "funding": { 3182 | "url": "https://github.com/sponsors/jonschlinkert" 3183 | } 3184 | }, 3185 | "node_modules/pkg-dir": { 3186 | "version": "4.2.0", 3187 | "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", 3188 | "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", 3189 | "dev": true, 3190 | "dependencies": { 3191 | "find-up": "^4.0.0" 3192 | }, 3193 | "engines": { 3194 | "node": ">=8" 3195 | } 3196 | }, 3197 | "node_modules/prelude-ls": { 3198 | "version": "1.2.1", 3199 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", 3200 | "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", 3201 | "dev": true, 3202 | "engines": { 3203 | "node": ">= 0.8.0" 3204 | } 3205 | }, 3206 | "node_modules/process-on-spawn": { 3207 | "version": "1.0.0", 3208 | "resolved": "https://registry.npmjs.org/process-on-spawn/-/process-on-spawn-1.0.0.tgz", 3209 | "integrity": "sha512-1WsPDsUSMmZH5LeMLegqkPDrsGgsWwk1Exipy2hvB0o/F0ASzbpIctSCcZIK1ykJvtTJULEH+20WOFjMvGnCTg==", 3210 | "dev": true, 3211 | "dependencies": { 3212 | "fromentries": "^1.2.0" 3213 | }, 3214 | "engines": { 3215 | "node": ">=8" 3216 | } 3217 | }, 3218 | "node_modules/proxy-addr": { 3219 | "version": "2.0.7", 3220 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 3221 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 3222 | "dev": true, 3223 | "dependencies": { 3224 | "forwarded": "0.2.0", 3225 | "ipaddr.js": "1.9.1" 3226 | }, 3227 | "engines": { 3228 | "node": ">= 0.10" 3229 | } 3230 | }, 3231 | "node_modules/punycode": { 3232 | "version": "2.1.1", 3233 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 3234 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", 3235 | "dev": true, 3236 | "engines": { 3237 | "node": ">=6" 3238 | } 3239 | }, 3240 | "node_modules/qs": { 3241 | "version": "6.13.0", 3242 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.0.tgz", 3243 | "integrity": "sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==", 3244 | "dev": true, 3245 | "license": "BSD-3-Clause", 3246 | "dependencies": { 3247 | "side-channel": "^1.0.6" 3248 | }, 3249 | "engines": { 3250 | "node": ">=0.6" 3251 | }, 3252 | "funding": { 3253 | "url": "https://github.com/sponsors/ljharb" 3254 | } 3255 | }, 3256 | "node_modules/randombytes": { 3257 | "version": "2.1.0", 3258 | "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", 3259 | "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", 3260 | "dev": true, 3261 | "dependencies": { 3262 | "safe-buffer": "^5.1.0" 3263 | } 3264 | }, 3265 | "node_modules/range-parser": { 3266 | "version": "1.2.1", 3267 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 3268 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", 3269 | "dev": true, 3270 | "license": "MIT", 3271 | "engines": { 3272 | "node": ">= 0.6" 3273 | } 3274 | }, 3275 | "node_modules/raw-body": { 3276 | "version": "2.5.2", 3277 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz", 3278 | "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==", 3279 | "dev": true, 3280 | "license": "MIT", 3281 | "dependencies": { 3282 | "bytes": "3.1.2", 3283 | "http-errors": "2.0.0", 3284 | "iconv-lite": "0.4.24", 3285 | "unpipe": "1.0.0" 3286 | }, 3287 | "engines": { 3288 | "node": ">= 0.8" 3289 | } 3290 | }, 3291 | "node_modules/readdirp": { 3292 | "version": "3.6.0", 3293 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 3294 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 3295 | "dev": true, 3296 | "dependencies": { 3297 | "picomatch": "^2.2.1" 3298 | }, 3299 | "engines": { 3300 | "node": ">=8.10.0" 3301 | } 3302 | }, 3303 | "node_modules/regexpp": { 3304 | "version": "3.2.0", 3305 | "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz", 3306 | "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==", 3307 | "dev": true, 3308 | "engines": { 3309 | "node": ">=8" 3310 | }, 3311 | "funding": { 3312 | "url": "https://github.com/sponsors/mysticatea" 3313 | } 3314 | }, 3315 | "node_modules/release-zalgo": { 3316 | "version": "1.0.0", 3317 | "resolved": "https://registry.npmjs.org/release-zalgo/-/release-zalgo-1.0.0.tgz", 3318 | "integrity": "sha1-CXALflB0Mpc5Mw5TXFqQ+2eFFzA=", 3319 | "dev": true, 3320 | "dependencies": { 3321 | "es6-error": "^4.0.1" 3322 | }, 3323 | "engines": { 3324 | "node": ">=4" 3325 | } 3326 | }, 3327 | "node_modules/require-directory": { 3328 | "version": "2.1.1", 3329 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 3330 | "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", 3331 | "dev": true, 3332 | "engines": { 3333 | "node": ">=0.10.0" 3334 | } 3335 | }, 3336 | "node_modules/require-main-filename": { 3337 | "version": "2.0.0", 3338 | "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", 3339 | "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", 3340 | "dev": true 3341 | }, 3342 | "node_modules/resolve-from": { 3343 | "version": "4.0.0", 3344 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", 3345 | "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", 3346 | "dev": true, 3347 | "engines": { 3348 | "node": ">=4" 3349 | } 3350 | }, 3351 | "node_modules/rimraf": { 3352 | "version": "3.0.2", 3353 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", 3354 | "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", 3355 | "dev": true, 3356 | "dependencies": { 3357 | "glob": "^7.1.3" 3358 | }, 3359 | "bin": { 3360 | "rimraf": "bin.js" 3361 | }, 3362 | "funding": { 3363 | "url": "https://github.com/sponsors/isaacs" 3364 | } 3365 | }, 3366 | "node_modules/safe-buffer": { 3367 | "version": "5.1.2", 3368 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 3369 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", 3370 | "dev": true 3371 | }, 3372 | "node_modules/safer-buffer": { 3373 | "version": "2.1.2", 3374 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 3375 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", 3376 | "dev": true, 3377 | "license": "MIT" 3378 | }, 3379 | "node_modules/semver": { 3380 | "version": "6.3.1", 3381 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", 3382 | "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", 3383 | "dev": true, 3384 | "bin": { 3385 | "semver": "bin/semver.js" 3386 | } 3387 | }, 3388 | "node_modules/send": { 3389 | "version": "0.19.0", 3390 | "resolved": "https://registry.npmjs.org/send/-/send-0.19.0.tgz", 3391 | "integrity": "sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw==", 3392 | "dev": true, 3393 | "license": "MIT", 3394 | "dependencies": { 3395 | "debug": "2.6.9", 3396 | "depd": "2.0.0", 3397 | "destroy": "1.2.0", 3398 | "encodeurl": "~1.0.2", 3399 | "escape-html": "~1.0.3", 3400 | "etag": "~1.8.1", 3401 | "fresh": "0.5.2", 3402 | "http-errors": "2.0.0", 3403 | "mime": "1.6.0", 3404 | "ms": "2.1.3", 3405 | "on-finished": "2.4.1", 3406 | "range-parser": "~1.2.1", 3407 | "statuses": "2.0.1" 3408 | }, 3409 | "engines": { 3410 | "node": ">= 0.8.0" 3411 | } 3412 | }, 3413 | "node_modules/send/node_modules/debug": { 3414 | "version": "2.6.9", 3415 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 3416 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 3417 | "dev": true, 3418 | "license": "MIT", 3419 | "dependencies": { 3420 | "ms": "2.0.0" 3421 | } 3422 | }, 3423 | "node_modules/send/node_modules/debug/node_modules/ms": { 3424 | "version": "2.0.0", 3425 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 3426 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", 3427 | "dev": true, 3428 | "license": "MIT" 3429 | }, 3430 | "node_modules/send/node_modules/encodeurl": { 3431 | "version": "1.0.2", 3432 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 3433 | "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", 3434 | "dev": true, 3435 | "license": "MIT", 3436 | "engines": { 3437 | "node": ">= 0.8" 3438 | } 3439 | }, 3440 | "node_modules/send/node_modules/ms": { 3441 | "version": "2.1.3", 3442 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 3443 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 3444 | "dev": true, 3445 | "license": "MIT" 3446 | }, 3447 | "node_modules/serialize-javascript": { 3448 | "version": "6.0.0", 3449 | "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.0.tgz", 3450 | "integrity": "sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==", 3451 | "dev": true, 3452 | "dependencies": { 3453 | "randombytes": "^2.1.0" 3454 | } 3455 | }, 3456 | "node_modules/serve-static": { 3457 | "version": "1.16.2", 3458 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.2.tgz", 3459 | "integrity": "sha512-VqpjJZKadQB/PEbEwvFdO43Ax5dFBZ2UECszz8bQ7pi7wt//PWe1P6MN7eCnjsatYtBT6EuiClbjSWP2WrIoTw==", 3460 | "dev": true, 3461 | "license": "MIT", 3462 | "dependencies": { 3463 | "encodeurl": "~2.0.0", 3464 | "escape-html": "~1.0.3", 3465 | "parseurl": "~1.3.3", 3466 | "send": "0.19.0" 3467 | }, 3468 | "engines": { 3469 | "node": ">= 0.8.0" 3470 | } 3471 | }, 3472 | "node_modules/set-blocking": { 3473 | "version": "2.0.0", 3474 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", 3475 | "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", 3476 | "dev": true 3477 | }, 3478 | "node_modules/setprototypeof": { 3479 | "version": "1.2.0", 3480 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", 3481 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", 3482 | "dev": true, 3483 | "license": "ISC" 3484 | }, 3485 | "node_modules/shebang-command": { 3486 | "version": "2.0.0", 3487 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", 3488 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", 3489 | "dev": true, 3490 | "dependencies": { 3491 | "shebang-regex": "^3.0.0" 3492 | }, 3493 | "engines": { 3494 | "node": ">=8" 3495 | } 3496 | }, 3497 | "node_modules/shebang-regex": { 3498 | "version": "3.0.0", 3499 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", 3500 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", 3501 | "dev": true, 3502 | "engines": { 3503 | "node": ">=8" 3504 | } 3505 | }, 3506 | "node_modules/side-channel": { 3507 | "version": "1.1.0", 3508 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", 3509 | "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", 3510 | "dev": true, 3511 | "license": "MIT", 3512 | "dependencies": { 3513 | "es-errors": "^1.3.0", 3514 | "object-inspect": "^1.13.3", 3515 | "side-channel-list": "^1.0.0", 3516 | "side-channel-map": "^1.0.1", 3517 | "side-channel-weakmap": "^1.0.2" 3518 | }, 3519 | "engines": { 3520 | "node": ">= 0.4" 3521 | }, 3522 | "funding": { 3523 | "url": "https://github.com/sponsors/ljharb" 3524 | } 3525 | }, 3526 | "node_modules/side-channel-list": { 3527 | "version": "1.0.0", 3528 | "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", 3529 | "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", 3530 | "dev": true, 3531 | "license": "MIT", 3532 | "dependencies": { 3533 | "es-errors": "^1.3.0", 3534 | "object-inspect": "^1.13.3" 3535 | }, 3536 | "engines": { 3537 | "node": ">= 0.4" 3538 | }, 3539 | "funding": { 3540 | "url": "https://github.com/sponsors/ljharb" 3541 | } 3542 | }, 3543 | "node_modules/side-channel-map": { 3544 | "version": "1.0.1", 3545 | "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", 3546 | "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", 3547 | "dev": true, 3548 | "license": "MIT", 3549 | "dependencies": { 3550 | "call-bound": "^1.0.2", 3551 | "es-errors": "^1.3.0", 3552 | "get-intrinsic": "^1.2.5", 3553 | "object-inspect": "^1.13.3" 3554 | }, 3555 | "engines": { 3556 | "node": ">= 0.4" 3557 | }, 3558 | "funding": { 3559 | "url": "https://github.com/sponsors/ljharb" 3560 | } 3561 | }, 3562 | "node_modules/side-channel-weakmap": { 3563 | "version": "1.0.2", 3564 | "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", 3565 | "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", 3566 | "dev": true, 3567 | "license": "MIT", 3568 | "dependencies": { 3569 | "call-bound": "^1.0.2", 3570 | "es-errors": "^1.3.0", 3571 | "get-intrinsic": "^1.2.5", 3572 | "object-inspect": "^1.13.3", 3573 | "side-channel-map": "^1.0.1" 3574 | }, 3575 | "engines": { 3576 | "node": ">= 0.4" 3577 | }, 3578 | "funding": { 3579 | "url": "https://github.com/sponsors/ljharb" 3580 | } 3581 | }, 3582 | "node_modules/signal-exit": { 3583 | "version": "3.0.7", 3584 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", 3585 | "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", 3586 | "dev": true 3587 | }, 3588 | "node_modules/spawn-wrap": { 3589 | "version": "2.0.0", 3590 | "resolved": "https://registry.npmjs.org/spawn-wrap/-/spawn-wrap-2.0.0.tgz", 3591 | "integrity": "sha512-EeajNjfN9zMnULLwhZZQU3GWBoFNkbngTUPfaawT4RkMiviTxcX0qfhVbGey39mfctfDHkWtuecgQ8NJcyQWHg==", 3592 | "dev": true, 3593 | "dependencies": { 3594 | "foreground-child": "^2.0.0", 3595 | "is-windows": "^1.0.2", 3596 | "make-dir": "^3.0.0", 3597 | "rimraf": "^3.0.0", 3598 | "signal-exit": "^3.0.2", 3599 | "which": "^2.0.1" 3600 | }, 3601 | "engines": { 3602 | "node": ">=8" 3603 | } 3604 | }, 3605 | "node_modules/sprintf-js": { 3606 | "version": "1.0.3", 3607 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 3608 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", 3609 | "dev": true 3610 | }, 3611 | "node_modules/statuses": { 3612 | "version": "2.0.1", 3613 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", 3614 | "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", 3615 | "dev": true, 3616 | "license": "MIT", 3617 | "engines": { 3618 | "node": ">= 0.8" 3619 | } 3620 | }, 3621 | "node_modules/string-width": { 3622 | "version": "4.2.3", 3623 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 3624 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 3625 | "dev": true, 3626 | "dependencies": { 3627 | "emoji-regex": "^8.0.0", 3628 | "is-fullwidth-code-point": "^3.0.0", 3629 | "strip-ansi": "^6.0.1" 3630 | }, 3631 | "engines": { 3632 | "node": ">=8" 3633 | } 3634 | }, 3635 | "node_modules/strip-ansi": { 3636 | "version": "6.0.1", 3637 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 3638 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 3639 | "dev": true, 3640 | "dependencies": { 3641 | "ansi-regex": "^5.0.1" 3642 | }, 3643 | "engines": { 3644 | "node": ">=8" 3645 | } 3646 | }, 3647 | "node_modules/strip-bom": { 3648 | "version": "4.0.0", 3649 | "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", 3650 | "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", 3651 | "dev": true, 3652 | "engines": { 3653 | "node": ">=8" 3654 | } 3655 | }, 3656 | "node_modules/strip-json-comments": { 3657 | "version": "3.1.1", 3658 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", 3659 | "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", 3660 | "dev": true, 3661 | "engines": { 3662 | "node": ">=8" 3663 | }, 3664 | "funding": { 3665 | "url": "https://github.com/sponsors/sindresorhus" 3666 | } 3667 | }, 3668 | "node_modules/superagent": { 3669 | "version": "8.0.6", 3670 | "resolved": "https://registry.npmjs.org/superagent/-/superagent-8.0.6.tgz", 3671 | "integrity": "sha512-HqSe6DSIh3hEn6cJvCkaM1BLi466f1LHi4yubR0tpewlMpk4RUFFy35bKz8SsPBwYfIIJy5eclp+3tCYAuX0bw==", 3672 | "dev": true, 3673 | "dependencies": { 3674 | "component-emitter": "^1.3.0", 3675 | "cookiejar": "^2.1.3", 3676 | "debug": "^4.3.4", 3677 | "fast-safe-stringify": "^2.1.1", 3678 | "form-data": "^4.0.0", 3679 | "formidable": "^2.1.1", 3680 | "methods": "^1.1.2", 3681 | "mime": "2.6.0", 3682 | "qs": "^6.11.0", 3683 | "semver": "^7.3.8" 3684 | }, 3685 | "engines": { 3686 | "node": ">=6.4.0 <13 || >=14" 3687 | } 3688 | }, 3689 | "node_modules/superagent/node_modules/mime": { 3690 | "version": "2.6.0", 3691 | "resolved": "https://registry.npmjs.org/mime/-/mime-2.6.0.tgz", 3692 | "integrity": "sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg==", 3693 | "dev": true, 3694 | "bin": { 3695 | "mime": "cli.js" 3696 | }, 3697 | "engines": { 3698 | "node": ">=4.0.0" 3699 | } 3700 | }, 3701 | "node_modules/superagent/node_modules/semver": { 3702 | "version": "7.5.4", 3703 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", 3704 | "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", 3705 | "dev": true, 3706 | "dependencies": { 3707 | "lru-cache": "^6.0.0" 3708 | }, 3709 | "bin": { 3710 | "semver": "bin/semver.js" 3711 | }, 3712 | "engines": { 3713 | "node": ">=10" 3714 | } 3715 | }, 3716 | "node_modules/supertest": { 3717 | "version": "6.3.3", 3718 | "resolved": "https://registry.npmjs.org/supertest/-/supertest-6.3.3.tgz", 3719 | "integrity": "sha512-EMCG6G8gDu5qEqRQ3JjjPs6+FYT1a7Hv5ApHvtSghmOFJYtsU5S+pSb6Y2EUeCEY3CmEL3mmQ8YWlPOzQomabA==", 3720 | "dev": true, 3721 | "dependencies": { 3722 | "methods": "^1.1.2", 3723 | "superagent": "^8.0.5" 3724 | }, 3725 | "engines": { 3726 | "node": ">=6.4.0" 3727 | } 3728 | }, 3729 | "node_modules/supports-color": { 3730 | "version": "7.2.0", 3731 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 3732 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 3733 | "dev": true, 3734 | "dependencies": { 3735 | "has-flag": "^4.0.0" 3736 | }, 3737 | "engines": { 3738 | "node": ">=8" 3739 | } 3740 | }, 3741 | "node_modules/test-exclude": { 3742 | "version": "6.0.0", 3743 | "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", 3744 | "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", 3745 | "dev": true, 3746 | "dependencies": { 3747 | "@istanbuljs/schema": "^0.1.2", 3748 | "glob": "^7.1.4", 3749 | "minimatch": "^3.0.4" 3750 | }, 3751 | "engines": { 3752 | "node": ">=8" 3753 | } 3754 | }, 3755 | "node_modules/text-table": { 3756 | "version": "0.2.0", 3757 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", 3758 | "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", 3759 | "dev": true 3760 | }, 3761 | "node_modules/to-regex-range": { 3762 | "version": "5.0.1", 3763 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 3764 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 3765 | "dev": true, 3766 | "license": "MIT", 3767 | "dependencies": { 3768 | "is-number": "^7.0.0" 3769 | }, 3770 | "engines": { 3771 | "node": ">=8.0" 3772 | } 3773 | }, 3774 | "node_modules/toidentifier": { 3775 | "version": "1.0.1", 3776 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", 3777 | "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", 3778 | "dev": true, 3779 | "license": "MIT", 3780 | "engines": { 3781 | "node": ">=0.6" 3782 | } 3783 | }, 3784 | "node_modules/type-check": { 3785 | "version": "0.4.0", 3786 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", 3787 | "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", 3788 | "dev": true, 3789 | "dependencies": { 3790 | "prelude-ls": "^1.2.1" 3791 | }, 3792 | "engines": { 3793 | "node": ">= 0.8.0" 3794 | } 3795 | }, 3796 | "node_modules/type-detect": { 3797 | "version": "4.0.8", 3798 | "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", 3799 | "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", 3800 | "dev": true, 3801 | "engines": { 3802 | "node": ">=4" 3803 | } 3804 | }, 3805 | "node_modules/type-fest": { 3806 | "version": "0.20.2", 3807 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", 3808 | "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", 3809 | "dev": true, 3810 | "engines": { 3811 | "node": ">=10" 3812 | }, 3813 | "funding": { 3814 | "url": "https://github.com/sponsors/sindresorhus" 3815 | } 3816 | }, 3817 | "node_modules/type-is": { 3818 | "version": "1.6.18", 3819 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 3820 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 3821 | "dev": true, 3822 | "license": "MIT", 3823 | "dependencies": { 3824 | "media-typer": "0.3.0", 3825 | "mime-types": "~2.1.24" 3826 | }, 3827 | "engines": { 3828 | "node": ">= 0.6" 3829 | } 3830 | }, 3831 | "node_modules/typedarray-to-buffer": { 3832 | "version": "3.1.5", 3833 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", 3834 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", 3835 | "dev": true, 3836 | "dependencies": { 3837 | "is-typedarray": "^1.0.0" 3838 | } 3839 | }, 3840 | "node_modules/unpipe": { 3841 | "version": "1.0.0", 3842 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 3843 | "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", 3844 | "dev": true, 3845 | "license": "MIT", 3846 | "engines": { 3847 | "node": ">= 0.8" 3848 | } 3849 | }, 3850 | "node_modules/uri-js": { 3851 | "version": "4.4.1", 3852 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", 3853 | "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", 3854 | "dev": true, 3855 | "dependencies": { 3856 | "punycode": "^2.1.0" 3857 | } 3858 | }, 3859 | "node_modules/utils-merge": { 3860 | "version": "1.0.1", 3861 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 3862 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=", 3863 | "dev": true, 3864 | "engines": { 3865 | "node": ">= 0.4.0" 3866 | } 3867 | }, 3868 | "node_modules/uuid": { 3869 | "version": "3.4.0", 3870 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", 3871 | "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", 3872 | "deprecated": "Please upgrade to version 7 or higher. Older versions may use Math.random() in certain circumstances, which is known to be problematic. See https://v8.dev/blog/math-random for details.", 3873 | "dev": true, 3874 | "bin": { 3875 | "uuid": "bin/uuid" 3876 | } 3877 | }, 3878 | "node_modules/v8-compile-cache": { 3879 | "version": "2.1.0", 3880 | "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.1.0.tgz", 3881 | "integrity": "sha512-usZBT3PW+LOjM25wbqIlZwPeJV+3OSz3M1k1Ws8snlW39dZyYL9lOGC5FgPVHfk0jKmjiDV8Z0mIbVQPiwFs7g==", 3882 | "dev": true 3883 | }, 3884 | "node_modules/vary": { 3885 | "version": "1.1.2", 3886 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 3887 | "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", 3888 | "dev": true, 3889 | "engines": { 3890 | "node": ">= 0.8" 3891 | } 3892 | }, 3893 | "node_modules/which": { 3894 | "version": "2.0.2", 3895 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 3896 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 3897 | "dev": true, 3898 | "dependencies": { 3899 | "isexe": "^2.0.0" 3900 | }, 3901 | "bin": { 3902 | "node-which": "bin/node-which" 3903 | }, 3904 | "engines": { 3905 | "node": ">= 8" 3906 | } 3907 | }, 3908 | "node_modules/which-module": { 3909 | "version": "2.0.0", 3910 | "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", 3911 | "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", 3912 | "dev": true 3913 | }, 3914 | "node_modules/word-wrap": { 3915 | "version": "1.2.5", 3916 | "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", 3917 | "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", 3918 | "dev": true, 3919 | "engines": { 3920 | "node": ">=0.10.0" 3921 | } 3922 | }, 3923 | "node_modules/workerpool": { 3924 | "version": "6.2.0", 3925 | "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.2.0.tgz", 3926 | "integrity": "sha512-Rsk5qQHJ9eowMH28Jwhe8HEbmdYDX4lwoMWshiCXugjtHqMD9ZbiqSDLxcsfdqsETPzVUtX5s1Z5kStiIM6l4A==", 3927 | "dev": true 3928 | }, 3929 | "node_modules/wrap-ansi": { 3930 | "version": "6.2.0", 3931 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", 3932 | "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", 3933 | "dev": true, 3934 | "dependencies": { 3935 | "ansi-styles": "^4.0.0", 3936 | "string-width": "^4.1.0", 3937 | "strip-ansi": "^6.0.0" 3938 | }, 3939 | "engines": { 3940 | "node": ">=8" 3941 | } 3942 | }, 3943 | "node_modules/wrappy": { 3944 | "version": "1.0.2", 3945 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 3946 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 3947 | "dev": true 3948 | }, 3949 | "node_modules/write-file-atomic": { 3950 | "version": "3.0.3", 3951 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", 3952 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", 3953 | "dev": true, 3954 | "dependencies": { 3955 | "imurmurhash": "^0.1.4", 3956 | "is-typedarray": "^1.0.0", 3957 | "signal-exit": "^3.0.2", 3958 | "typedarray-to-buffer": "^3.1.5" 3959 | } 3960 | }, 3961 | "node_modules/y18n": { 3962 | "version": "4.0.3", 3963 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.3.tgz", 3964 | "integrity": "sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ==", 3965 | "dev": true 3966 | }, 3967 | "node_modules/yallist": { 3968 | "version": "4.0.0", 3969 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", 3970 | "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", 3971 | "dev": true 3972 | }, 3973 | "node_modules/yargs": { 3974 | "version": "15.4.1", 3975 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", 3976 | "integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==", 3977 | "dev": true, 3978 | "dependencies": { 3979 | "cliui": "^6.0.0", 3980 | "decamelize": "^1.2.0", 3981 | "find-up": "^4.1.0", 3982 | "get-caller-file": "^2.0.1", 3983 | "require-directory": "^2.1.1", 3984 | "require-main-filename": "^2.0.0", 3985 | "set-blocking": "^2.0.0", 3986 | "string-width": "^4.2.0", 3987 | "which-module": "^2.0.0", 3988 | "y18n": "^4.0.0", 3989 | "yargs-parser": "^18.1.2" 3990 | }, 3991 | "engines": { 3992 | "node": ">=8" 3993 | } 3994 | }, 3995 | "node_modules/yargs-parser": { 3996 | "version": "18.1.3", 3997 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz", 3998 | "integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==", 3999 | "dev": true, 4000 | "dependencies": { 4001 | "camelcase": "^5.0.0", 4002 | "decamelize": "^1.2.0" 4003 | }, 4004 | "engines": { 4005 | "node": ">=6" 4006 | } 4007 | }, 4008 | "node_modules/yargs-unparser": { 4009 | "version": "2.0.0", 4010 | "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-2.0.0.tgz", 4011 | "integrity": "sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==", 4012 | "dev": true, 4013 | "dependencies": { 4014 | "camelcase": "^6.0.0", 4015 | "decamelize": "^4.0.0", 4016 | "flat": "^5.0.2", 4017 | "is-plain-obj": "^2.1.0" 4018 | }, 4019 | "engines": { 4020 | "node": ">=10" 4021 | } 4022 | }, 4023 | "node_modules/yargs-unparser/node_modules/camelcase": { 4024 | "version": "6.2.0", 4025 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.2.0.tgz", 4026 | "integrity": "sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg==", 4027 | "dev": true, 4028 | "engines": { 4029 | "node": ">=10" 4030 | }, 4031 | "funding": { 4032 | "url": "https://github.com/sponsors/sindresorhus" 4033 | } 4034 | }, 4035 | "node_modules/yargs-unparser/node_modules/decamelize": { 4036 | "version": "4.0.0", 4037 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-4.0.0.tgz", 4038 | "integrity": "sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==", 4039 | "dev": true, 4040 | "engines": { 4041 | "node": ">=10" 4042 | }, 4043 | "funding": { 4044 | "url": "https://github.com/sponsors/sindresorhus" 4045 | } 4046 | }, 4047 | "node_modules/yocto-queue": { 4048 | "version": "0.1.0", 4049 | "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", 4050 | "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", 4051 | "dev": true, 4052 | "engines": { 4053 | "node": ">=10" 4054 | }, 4055 | "funding": { 4056 | "url": "https://github.com/sponsors/sindresorhus" 4057 | } 4058 | } 4059 | } 4060 | } 4061 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "visa-js", 3 | "version": "1.0.25", 4 | "description": "Simple and flexible Access Control Manager based on Attribute Based Access Control (ABAC) paradigm that works well with Express and Passport or independently", 5 | "main": "index.js", 6 | "scripts": { 7 | "test": "nyc --reporter=lcov mocha", 8 | "lint": "eslint lib" 9 | }, 10 | "repository": { 11 | "type": "git", 12 | "url": "git+ssh://git@github.com/ZenitechSoftware/visa-js.git" 13 | }, 14 | "keywords": [ 15 | "access", 16 | "control", 17 | "ABAC", 18 | "authorization", 19 | "authorize", 20 | "visa", 21 | "visajs", 22 | "visa.js" 23 | ], 24 | "author": "Zenitech.co.uk", 25 | "license": "MIT", 26 | "bugs": { 27 | "url": "https://github.com/ZenitechSoftware/visa-js/issues" 28 | }, 29 | "homepage": "https://github.com/ZenitechSoftware/visa-js#readme", 30 | "devDependencies": { 31 | "chai": "^4.2.0", 32 | "eslint": "^8.11.0", 33 | "express": "^4.18.2", 34 | "express-async-errors": "^3.1.1", 35 | "mocha": "^9.2.2", 36 | "nyc": "^15.1.0", 37 | "passport": "^0.6.0", 38 | "passport-strategy": "^1.0.0", 39 | "supertest": "^6.3.3" 40 | }, 41 | "dependencies": { 42 | "async": "^3.2.6" 43 | } 44 | } 45 | -------------------------------------------------------------------------------- /test/askTest.js: -------------------------------------------------------------------------------- 1 | const visa = require('../'); 2 | 3 | require('chai').should(); 4 | 5 | describe('visa.js ask', () => { 6 | 7 | beforeEach(() => visa.reset()); 8 | 9 | context('Errors', () => { 10 | context('Rule returns error', () => { 11 | it('should catch error and enhance with details', () => { 12 | visa.policy({ 13 | objects: { 14 | 'account': { 15 | operations: { 16 | 'open': () => { throw new Error('Some rule error') }, 17 | } 18 | }, 19 | } 20 | }); 21 | return visa.ask().can.open.account() 22 | .then(() => { throw new Error('Test should fail') }) 23 | .catch(error => error.message.should.equal(`visa.js: object 'account' operation 'open': rule failed with error: Some rule error`)); 24 | }); 25 | }); 26 | }); 27 | context('Rule returns true', () => { 28 | it('should be authorized', () => { 29 | visa.policy({ 30 | objects: { 31 | 'account': { 32 | operations: { 33 | 'open': subject => subject.role === 'teller', 34 | } 35 | } 36 | } 37 | }); 38 | const subject = { role: 'teller' }; 39 | return visa.ask(subject).can.open.account().then(answer => answer.should.equal(true)); 40 | }); 41 | }); 42 | context('Rule returns false', () => { 43 | it('should NOT be authorized', () => { 44 | visa.policy({ 45 | objects: { 46 | 'account': { 47 | operations: { 48 | 'open': subject => subject.role === 'teller', 49 | } 50 | } 51 | } 52 | }); 53 | const subject = { role: 'manager' }; 54 | return visa.ask(subject).can.open.account().then(answer => answer.should.equal(false)); 55 | }); 56 | }); 57 | context('can.not', () => { 58 | context('Rule fails', () => { 59 | it('should return true', () => { 60 | visa.policy({ 61 | objects: { 62 | 'account': { 63 | operations: { 64 | 'open': () => false 65 | } 66 | } 67 | } 68 | }); 69 | return visa.ask().can.not.open.account().then(answer => answer.should.equal(true)); 70 | }); 71 | }); 72 | context('Rule passes', () => { 73 | it('should return false', () => { 74 | visa.policy({ 75 | objects: { 76 | 'account': { 77 | operations: { 78 | 'open': () => true 79 | } 80 | } 81 | } 82 | }); 83 | return visa.ask().can.not.open.account().then(answer => answer.should.equal(false)); 84 | }); 85 | }); 86 | }); 87 | }); 88 | -------------------------------------------------------------------------------- /test/checkTest.js: -------------------------------------------------------------------------------- 1 | const visa = require('../'); 2 | 3 | require('chai').should(); 4 | 5 | describe('visa.js check', () => { 6 | 7 | beforeEach(() => visa.reset()); 8 | 9 | context('Errors', () => { 10 | context('Rule returns error', () => { 11 | it('should catch error and enhance with details', () => { 12 | visa.policy({ 13 | objects: { 14 | 'account': { 15 | operations: { 16 | 'open': () => { throw new Error('Some rule error') }, 17 | } 18 | }, 19 | } 20 | }); 21 | return visa.check().can.open.account() 22 | .then(() => { throw new Error('Test should fail') }) 23 | .catch(error => error.message.should.equal(`visa.js: object 'account' operation 'open': rule failed with error: Some rule error`)); 24 | }); 25 | }); 26 | }); 27 | context('Same operation name used in 2 different objects', () => { 28 | it('should be authorized first object and NOT authorize second object', () => { 29 | visa.policy({ 30 | objects: { 31 | 'account': { 32 | operations: { 33 | 'open': subject => subject.role === 'teller', 34 | } 35 | }, 36 | } 37 | }); 38 | visa.policy({ 39 | objects: { 40 | 'loan': { 41 | operations: { 42 | 'open': subject => Promise.resolve(subject.role === 'manager'), 43 | } 44 | }, 45 | } 46 | }); 47 | const subject = { role: 'teller' }; 48 | return Promise.all([ 49 | visa.check(subject).can.open.account(), 50 | visa.check(subject).can.open.loan() 51 | .then(() => { throw new Error('Test should fail') }) 52 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }) 53 | ]); 54 | }); 55 | }); 56 | context('Subject attributes', () => { 57 | context('Subject attributes are matching', () => { 58 | it('should be authorized', () => { 59 | visa.policy({ 60 | objects: { 61 | 'account': { 62 | operations: { 63 | 'open': subject => subject.role === 'teller', 64 | } 65 | } 66 | } 67 | }); 68 | const subject = { role: 'teller' }; 69 | return visa.check(subject).can.open.account(); 70 | }); 71 | }); 72 | context('Subject attributes are NOT matching', () => { 73 | it('should NOT be authorized', () => { 74 | visa.policy({ 75 | objects: { 76 | 'account': { 77 | operations: { 78 | 'open': subject => subject.role === 'teller', 79 | } 80 | } 81 | } 82 | }); 83 | const subject = { role: 'customer' }; 84 | return visa.check(subject).can.open.account() 85 | .then(() => { throw new Error('Test should fail') }) 86 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }) 87 | }); 88 | }); 89 | }); 90 | context('Object attributes', () => { 91 | context('Subject and object attributes are matching', () => { 92 | it('should be authorized', () => { 93 | visa.policy({ 94 | objects: { 95 | 'account': { 96 | operations: { 97 | 'open': (subject, account) => subject.role === 'teller' && account.country === 'LT', 98 | } 99 | } 100 | } 101 | }); 102 | const subject = { role: 'teller' }; 103 | const account = { country: 'LT' }; 104 | return visa.check(subject).can.open.account({ object: account }); 105 | }); 106 | }); 107 | context('Subject and multiple objects attributes are matching', () => { 108 | it('should be authorized', () => { 109 | visa.policy({ 110 | objects: { 111 | 'account': { 112 | operations: { 113 | 'open': (subject, account) => subject.role === 'teller' && account.country === 'LT', 114 | } 115 | } 116 | } 117 | }); 118 | const subject = { role: 'teller' }; 119 | const account = { country: 'LT' }; 120 | return visa.check(subject).can.open.account({ objects: [account, account] }); 121 | }); 122 | }); 123 | context('Subject attributes are matching but object attributes are NOT matching', () => { 124 | it('should NOT be authorized', () => { 125 | visa.policy({ 126 | objects: { 127 | 'account': { 128 | operations: { 129 | 'open': (subject, account) => subject.role === 'teller' && account.country === 'LT', 130 | } 131 | } 132 | } 133 | }); 134 | const subject = { role: 'teller' }; 135 | const account = { country: 'GB' }; 136 | return visa.check(subject).can.open.account({ object: account }) 137 | .then(() => { throw new Error('Test should fail') }) 138 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }) 139 | }); 140 | }); 141 | context('Subject and first object attributes are matching but second object attributes are NOT matching', () => { 142 | it('should NOT be authorized', () => { 143 | visa.policy({ 144 | objects: { 145 | 'account': { 146 | operations: { 147 | 'open': (subject, account) => subject.role === 'teller' && account.country === 'LT', 148 | } 149 | } 150 | } 151 | }); 152 | const subject = { role: 'teller' }; 153 | const account1 = { country: 'LT' }; 154 | const account2 = { country: 'GB' }; 155 | return visa.check(subject).can.open.account({ objects: [account1, account2] }) 156 | .then(() => { throw new Error('Test should fail') }) 157 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }) 158 | }); 159 | }); 160 | context('Rule returns error', () => { 161 | it('should catch error and enhance with details', () => { 162 | visa.policy({ 163 | objects: { 164 | 'account': { 165 | operations: { 166 | 'open': () => { throw new Error('Some rule error') }, 167 | } 168 | }, 169 | } 170 | }); 171 | const account = { country: 'GB' }; 172 | return visa.check().can.open.account({ object: account }) 173 | .then(() => { throw new Error('Test should fail') }) 174 | .catch(error => error.message.should.equal(`visa.js: object 'account' operation 'open': rule failed with error: Some rule error`)); 175 | }); 176 | }); 177 | }); 178 | context('Referenced object attributes', () => { 179 | context('Subject and object reference attributes are matching', () => { 180 | it('should be authorized', () => { 181 | visa.policy({ 182 | objects: { 183 | 'account': { 184 | mapRefsToObjects: refs => Promise.resolve(refs.map(() => ({ country: 'LT' }))), 185 | operations: { 186 | 'open': (subject, account) => subject.role === 'teller' && account.country === 'LT', 187 | } 188 | } 189 | } 190 | }); 191 | const subject = { role: 'teller' }; 192 | return visa.check(subject).can.open.account({ ref: 1 }) 193 | .then(account => account.should.deep.equal({ country: 'LT' })); 194 | }); 195 | }); 196 | context('Subject and multiple objects reference attributes are matching', () => { 197 | it('should be authorized', () => { 198 | visa.policy({ 199 | objects: { 200 | 'account': { 201 | mapRefsToObjects: (refs, cb) => cb(null, refs.map(() => ({ country: 'LT' }))), 202 | operations: { 203 | 'open': (subject, account) => subject.role === 'teller' && account.country === 'LT', 204 | } 205 | } 206 | } 207 | }); 208 | const subject = { role: 'teller' }; 209 | return visa.check(subject).can.open.account({ refs: [1, 2] }) 210 | .then(accounts => accounts.should.deep.equal([{ country: 'LT' }, { country: 'LT' }])); 211 | }); 212 | }); 213 | context('Subject attributes are matching but object reference attributes are not matching', () => { 214 | it('should NOT be authorized', () => { 215 | visa.policy({ 216 | objects: { 217 | 'account': { 218 | mapRefsToObjects: refs => refs.map(() => ({ country: 'EN' })), 219 | operations: { 220 | 'open': (subject, account) => subject.role === 'teller' && account.country === 'LT', 221 | } 222 | } 223 | } 224 | }); 225 | const subject = { role: 'teller' }; 226 | return visa.check(subject).can.open.account({ ref: 1 }) 227 | .then(() => { throw new Error('Test should fail') }) 228 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }) 229 | }); 230 | }); 231 | context('Subject attributes are matching but object multiple references attributes are not matching', () => { 232 | it('should NOT be authorized', () => { 233 | visa.policy({ 234 | objects: { 235 | 'account': { 236 | mapRefsToObjects: refs => refs.map(() => ({ country: 'EN' })), 237 | operations: { 238 | 'open': (subject, account) => subject.role === 'teller' && account.country === 'LT', 239 | } 240 | } 241 | } 242 | }); 243 | const subject = { role: 'teller' }; 244 | return visa.check(subject).can.open.account({ refs: [1, 2] }) 245 | .then(() => { throw new Error('Test should fail') }) 246 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }) 247 | }); 248 | }); 249 | context('Object reference is provided but mapRefsToObjects function is not provided', () => { 250 | it('should fail with details provided', () => { 251 | visa.policy({ 252 | objects: { 253 | 'account': { 254 | operations: { 255 | 'open': () => true, 256 | } 257 | } 258 | } 259 | }); 260 | return visa.check().can.open.account({ ref: 1 }) 261 | .then(() => { throw new Error('Test should fail') }) 262 | .catch(error => error.message.should.equal(`visa.js: object 'account' operation 'open': object references (1) can not be resolved because 'mapRefsToObjects' method is not provided`)); 263 | }); 264 | }); 265 | context('mapRefsToObjects function throws error', () => { 266 | it('should fail with details provided', () => { 267 | visa.policy({ 268 | objects: { 269 | 'account': { 270 | mapRefsToObjects: () => { throw new Error('mapRefsToObjects error') }, 271 | operations: { 272 | 'open': () => true, 273 | } 274 | } 275 | } 276 | }); 277 | return visa.check().can.open.account({ ref: 1 }) 278 | .then(() => { throw new Error('Test should fail') }) 279 | .catch(error => error.message.should.equal(`visa.js: object 'account' operation 'open': object references (1) can not be resolved because of 'mapRefsToObjects' error: mapRefsToObjects error`)); 280 | }); 281 | }); 282 | context('mapRefsToObjects function return error in cb', () => { 283 | it('should fail with details provided', () => { 284 | visa.policy({ 285 | objects: { 286 | 'account': { 287 | mapRefsToObjects: (refs, cb) => cb(new Error('mapRefsToObjects error')), 288 | operations: { 289 | 'open': () => true, 290 | } 291 | } 292 | } 293 | }); 294 | return visa.check().can.open.account({ ref: 1 }) 295 | .then(() => { throw new Error('Test should fail') }) 296 | .catch(error => error.message.should.equal(`visa.js: object 'account' operation 'open': object references (1) can not be resolved because of 'mapRefsToObjects' error: mapRefsToObjects error`)); 297 | }); 298 | }); 299 | context('mapRefsToObjects function rejects promise', () => { 300 | it('should fail with details provided', () => { 301 | visa.policy({ 302 | objects: { 303 | 'account': { 304 | mapRefsToObjects: () => Promise.reject(new Error('mapRefsToObjects error')), 305 | operations: { 306 | 'open': () => true, 307 | } 308 | } 309 | } 310 | }); 311 | return visa.check().can.open.account({ ref: 1 }) 312 | .then(() => { throw new Error('Test should fail') }) 313 | .catch(error => error.message.should.equal(`visa.js: object 'account' operation 'open': object references (1) can not be resolved because of 'mapRefsToObjects' error: mapRefsToObjects error`)); 314 | }); 315 | }); 316 | context('Provided mapRefsToObjects function returns number of objects that is not equal to number of refs passed', () => { 317 | it('should not authorize', () => { 318 | visa.policy({ 319 | objects: { 320 | 'account': { 321 | mapRefsToObjects: () => [], 322 | operations: { 323 | 'open': () => true, 324 | } 325 | } 326 | } 327 | }); 328 | return visa.check().can.open.account({ ref: 1 }) 329 | .then(() => { throw new Error('Test should fail') }) 330 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }) 331 | }); 332 | }); 333 | context('mapRefsToObjects function return null for account reference', () => { 334 | it('should not authorize', () => { 335 | visa.policy({ 336 | objects: { 337 | 'account': { 338 | mapRefsToObjects: () => [null], 339 | operations: { 340 | 'open': () => true, 341 | } 342 | } 343 | } 344 | }); 345 | return visa.check().can.open.account({ ref: 1 }) 346 | .then(() => { throw new Error('Test should fail') }) 347 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }) 348 | }); 349 | }); 350 | context('mapRefsToObjects function return undefined for account reference', () => { 351 | it('should not authorize', () => { 352 | visa.policy({ 353 | objects: { 354 | 'account': { 355 | mapRefsToObjects: () => [undefined], 356 | operations: { 357 | 'open': () => true, 358 | } 359 | } 360 | } 361 | }); 362 | return visa.check().can.open.account({ ref: 1 }) 363 | .then(() => { throw new Error('Test should fail') }) 364 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }); 365 | }); 366 | }); 367 | context('mapRefsToObjects function returns null', () => { 368 | it('should fail with details provided', () => { 369 | visa.policy({ 370 | objects: { 371 | 'account': { 372 | mapRefsToObjects: () => null, 373 | operations: { 374 | 'open': () => true, 375 | } 376 | } 377 | } 378 | }); 379 | return visa.check().can.open.account({ ref: 1 }) 380 | .then(() => { throw new Error('Test should fail') }) 381 | .catch(error => error.message.should.equal(`visa.js: object 'account' operation 'open': 'mapRefsToObjects' function should return array of objects`)); 382 | }); 383 | }); 384 | }); 385 | context('Context attributes', () => { 386 | context('Subject and context attributes are matching', () => { 387 | it('should be authorized', () => { 388 | visa.policy({ 389 | objects: { 390 | 'account': { 391 | operations: { 392 | 'open': (subject, account, context) => subject.role === 'teller' && context.method === 'POST', 393 | } 394 | } 395 | } 396 | }); 397 | const subject = { role: 'teller' }; 398 | const context = { method: 'POST' }; 399 | return visa.check(subject).can.open.account({ context }); 400 | }); 401 | }); 402 | context('Subject attributes are matching but context attributes are NOT matching', () => { 403 | it('should NOT be authorized', () => { 404 | visa.policy({ 405 | objects: { 406 | 'account': { 407 | operations: { 408 | 'open': (subject, account, context) => subject.role === 'teller' && context.method === 'POST', 409 | } 410 | } 411 | } 412 | }); 413 | const subject = { role: 'teller' }; 414 | const context = { method: 'GET' }; 415 | return visa.check(subject).can.open.account({ context }) 416 | .then(() => { throw new Error('Test should fail') }) 417 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }); 418 | }); 419 | }); 420 | }); 421 | context('can.not', () => { 422 | context('Rule fails', () => { 423 | it('should not fail', () => { 424 | visa.policy({ 425 | objects: { 426 | 'account': { 427 | operations: { 428 | 'open': () => false 429 | } 430 | } 431 | } 432 | }); 433 | return visa.check().can.not.open.account(); 434 | }); 435 | }); 436 | context('Rule passes', () => { 437 | it('should fail', () => { 438 | visa.policy({ 439 | objects: { 440 | 'account': { 441 | operations: { 442 | 'open': () => true 443 | } 444 | } 445 | } 446 | }); 447 | return visa.check().can.not.open.account() 448 | .then(() => { throw new Error('Test should fail') }) 449 | .catch(error => { if (!(error instanceof visa.Unauthorized)) throw error }); 450 | }); 451 | }); 452 | context('Rule throws error', () => { 453 | it('should throw error', () => { 454 | visa.policy({ 455 | objects: { 456 | 'account': { 457 | operations: { 458 | 'open': () => { throw new Error('rule error') }, 459 | } 460 | } 461 | } 462 | }); 463 | return visa.check().can.not.open.account() 464 | .then(() => { throw new Error('Test should fail') }) 465 | .catch(error => error.message.should.equal(`visa.js: object 'account' operation 'open': rule failed with error: rule error`)); 466 | 467 | }); 468 | }); 469 | }); 470 | }); 471 | -------------------------------------------------------------------------------- /test/middlewareTest.js: -------------------------------------------------------------------------------- 1 | const visa = require('../'); 2 | const express = require('express'); 3 | const request = require('supertest'); 4 | const passport = require('passport') 5 | const Strategy = require('passport-strategy').Strategy; 6 | 7 | require('express-async-errors'); 8 | require('chai').should(); 9 | 10 | describe('visa.js middleware', () => { 11 | context('express app is running and passport local strategy is used', () => { 12 | let app = null; 13 | let router = null; 14 | let server = null; 15 | 16 | class TestStrategy extends Strategy { 17 | authenticate() { 18 | this.success({ id: 999, role: 'teller' }); 19 | } 20 | } 21 | passport.use('test', new TestStrategy()); 22 | 23 | beforeEach(() => { 24 | visa.reset(); 25 | router = new express.Router(); 26 | app = express(); 27 | app.use(passport.initialize()); 28 | app.use(router); 29 | app.use(visa.unauthorizedErrorHandler); 30 | server = app.listen(3001); 31 | }); 32 | 33 | afterEach(() => { 34 | server.close(); 35 | }); 36 | 37 | context('user role is matching', () => { 38 | it('should be authorized', () => { 39 | visa.policy({ 40 | objects: { 41 | 'account': { 42 | operations: { 43 | 'open': (subject, _, context) => subject.role === 'teller' 44 | && context.ip === '::ffff:127.0.0.1' 45 | } 46 | }, 47 | } 48 | }); 49 | router.post( 50 | '/api/account', 51 | passport.authenticate('test', { session: false }), 52 | visa.authorize(visa.user.can.open.account), 53 | (req, res) => res.send() 54 | ); 55 | return request(app) 56 | .post('/api/account') 57 | .expect(200); 58 | }); 59 | }); 60 | context('user is the owner of specific object', () => { 61 | it('should be authorized', () => { 62 | visa.policy({ 63 | objects: { 64 | 'account': { 65 | mapRefsToObjects: refs => refs.map(() => ({ ownerId: 999 })), 66 | operations: { 67 | 'close': (subject, account) => account.ownerId === subject.id, 68 | } 69 | }, 70 | } 71 | }); 72 | router.delete( 73 | '/api/account/:id', 74 | passport.authenticate('test', { session: false }), 75 | visa.authorize(visa.user.can.close.account), 76 | (req, res) => res.send() 77 | ); 78 | return request(app) 79 | .delete('/api/account/1') 80 | .expect(200); 81 | }); 82 | }); 83 | context('user is the owner of specific object referenced with :objectId Express param', () => { 84 | it('should be authorized', () => { 85 | visa.policy({ 86 | objects: { 87 | 'account': { 88 | mapRefsToObjects: refs => refs.map(() => ({ ownerId: 999 })), 89 | operations: { 90 | 'close': (subject, account) => account.ownerId === subject.id, 91 | } 92 | }, 93 | } 94 | }); 95 | router.delete( 96 | '/api/account/:objectId', 97 | passport.authenticate('test', { session: false }), 98 | visa.authorize(visa.user.can.close.account, req => req.params.objectId), 99 | (req, res) => res.send() 100 | ); 101 | return request(app) 102 | .delete('/api/account/1') 103 | .expect(200); 104 | }); 105 | }); 106 | context('user role is not matching', () => { 107 | it('should NOT be authorized and return 401', () => { 108 | visa.policy({ 109 | objects: { 110 | 'account': { 111 | operations: { 112 | 'open': (subject) => subject.role === 'manager', 113 | } 114 | }, 115 | } 116 | }); 117 | router.post( 118 | '/api/account', 119 | passport.authenticate('test', { session: false }), 120 | visa.authorize(visa.user.can.open.account), 121 | (req, res) => res.send() 122 | ); 123 | return request(app) 124 | .post('/api/account') 125 | .expect(401); 126 | }); 127 | }); 128 | context('api method throws visa.Unauthorized', () => { 129 | it('should NOT be authorized and return 401', () => { 130 | visa.policy({ 131 | objects: { 132 | 'account': { 133 | operations: { 134 | 'open': (subject) => subject.role === 'manager', 135 | } 136 | }, 137 | } 138 | }); 139 | router.post( 140 | '/api/account', 141 | passport.authenticate('test', { session: false }), 142 | () => visa.check({ role: 'user' }).can.open.account() 143 | ); 144 | return request(app) 145 | .post('/api/account') 146 | .expect(401); 147 | }); 148 | }); 149 | context('visa rule fails', () => { 150 | it('should NOT be authorized and return 500', () => { 151 | visa.policy({ 152 | objects: { 153 | 'account': { 154 | operations: { 155 | 'open': () => { throw new Error('test error') }, 156 | } 157 | }, 158 | } 159 | }); 160 | router.post( 161 | '/api/account', 162 | passport.authenticate('test', { session: false }), 163 | visa.authorize(visa.user.can.open.account), 164 | (req, res) => res.send() 165 | ); 166 | return request(app) 167 | .post('/api/account') 168 | .expect(500); 169 | }); 170 | }); 171 | }); 172 | }); 173 | --------------------------------------------------------------------------------