├── .github
    └── workflows
    │   └── main.yml
├── .gitignore
├── LICENSE
├── examples
    ├── calculator.wasm
    ├── index.html
    ├── node-example.js
    ├── using_calculator.wasm
    ├── using_calculator_async.wasm
    ├── wasi-example.js
    └── wasi_hello.wasm
├── package-lock.json
├── package.json
├── readme.md
├── serve.js
├── src
    ├── index.ts
    ├── linker.ts
    └── store.ts
├── tests
    ├── linker.ts
    ├── store.ts
    └── wat.ts
├── tsconfig.json
└── webpack.config.js
/.github/workflows/main.yml:
--------------------------------------------------------------------------------
 1 | name: Build and Test
 2 | 
 3 | on: [push, pull_request]
 4 | 
 5 | jobs:
 6 |   build:
 7 |     runs-on: ubuntu-latest
 8 | 
 9 |     steps:
10 |       - uses: actions/checkout@v2
11 |       - name: Build and Test
12 |         run: |
13 |           npm install
14 |           npm run build
15 |           npm run test
16 |           npm run examples
17 | 
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | target
2 | Cargo.lock
3 | node_modules/
4 | dist/
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
 1 |    MIT License
 2 | 
 3 |     Copyright (c) Microsoft Corporation. All rights reserved.
 4 | 
 5 |     Permission is hereby granted, free of charge, to any person obtaining a copy
 6 |     of this software and associated documentation files (the "Software"), to deal
 7 |     in the Software without restriction, including without limitation the rights
 8 |     to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 9 |     copies of the Software, and to permit persons to whom the Software is
10 |     furnished to do so, subject to the following conditions:
11 | 
12 |     The above copyright notice and this permission notice shall be included in all
13 |     copies or substantial portions of the Software.
14 | 
15 |     THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 |     IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 |     FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 |     AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 |     LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 |     OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 |     SOFTWARE
22 | 
--------------------------------------------------------------------------------
/examples/calculator.wasm:
--------------------------------------------------------------------------------
1 |  asm   `    /add  multiply subtract divide memory 
2 | !    j    l    k    m \name" addmultiplysubtractdivide1  lhsrhs lhsrhs lhsrhs lhsrhs
--------------------------------------------------------------------------------
/examples/index.html:
--------------------------------------------------------------------------------
 1 | 
 2 | 
 3 |   
10 |   
11 |     
76 |   
77 | 
78 | 
--------------------------------------------------------------------------------
/examples/node-example.js:
--------------------------------------------------------------------------------
  1 | const { Linker } = require("../dist/src/index");
  2 | const { parseText } = require("binaryen");
  3 | const assert = require("assert");
  4 | 
  5 | const usingAdd = `
  6 | (module
  7 |     (import "calculator" "add" (func $calc_add (param i32 i32) (result i32)))
  8 |     
  9 |     (memory 1 1)
 10 |     (export "memory" (memory 0))
 11 |     (export "add" (func $add))
 12 | 
 13 |     (func $add (param i32) (param i32) (result i32)
 14 |         (return
 15 |             (call $calc_add
 16 |                 (local.get 0)
 17 |                 (local.get 1)
 18 |             )
 19 |         )
 20 |     )
 21 | )
 22 | `;
 23 | 
 24 | const add = `
 25 | (module
 26 |   (memory 1 1)
 27 |   (export "memory" (memory 0))
 28 |   (export "add" (func $add))
 29 |   
 30 |   (func $add (param i32) (param i32) (result i32)
 31 |       (return
 32 |           (i32.add
 33 |               (local.get 0)
 34 |               (local.get 1)
 35 |           )
 36 |       )
 37 |   )
 38 | )
 39 | `;
 40 | 
 41 | const usingAddWithAlias = `
 42 | (module
 43 |     (import "calculator1" "add" (func $calc_add1 (param i32 i32) (result i32)))
 44 |     (import "calculator2" "add" (func $calc_add2 (param i32 i32) (result i32)))
 45 | 
 46 |     (memory 1 1)
 47 |     (export "memory" (memory 0))
 48 | 
 49 |     (export "add1" (func $add1))
 50 |     (export "add2" (func $add2))
 51 | 
 52 |     (func $add1 (param i32) (param i32) (result i32)
 53 |         (return
 54 |             (call $calc_add1
 55 |                 (local.get 0)
 56 |                 (local.get 1)
 57 |             )
 58 |         )
 59 |     )
 60 | 
 61 |     (func $add2 (param i32) (param i32) (result i32)
 62 |         (return
 63 |             (call $calc_add2
 64 |                 (local.get 0)
 65 |                 (local.get 1)
 66 |             )
 67 |         )
 68 |     )
 69 | )
 70 | `;
 71 | 
 72 | (async () => {
 73 |   // example of using linker.define
 74 |   {
 75 |     var linker = new Linker();
 76 | 
 77 |     // the "usingAdd" module above imports calculator.add
 78 |     // we define it and provide a JS implementation, then
 79 |     // instantiate it.
 80 |     linker.define("calculator", "add", (a, b) => a + b);
 81 |     var calc = await linker.instantiate(parseText(usingAdd).emitBinary());
 82 | 
 83 |     assert.equal(calc.instance.exports.add(1, 2), 3);
 84 |   }
 85 | 
 86 |   // example of using linker.module
 87 |   {
 88 |     var linker = new Linker();
 89 | 
 90 |     // the "usingAdd" module above imports calculator.add
 91 |     // we link a module that we know exports the functionality
 92 |     // required, then instantiate the module that uses it.
 93 | 
 94 |     await linker.module(
 95 |       "calculator",
 96 |       new WebAssembly.Module(parseText(add).emitBinary())
 97 |     );
 98 |     var calc = await linker.instantiate(parseText(usingAdd).emitBinary());
 99 |     assert.equal(calc.instance.exports.add(1, 41), 42);
100 |   }
101 | 
102 |   // example of using linker.instance
103 |   {
104 |     var linker = new Linker();
105 |     var depsInstance = await WebAssembly.instantiate(
106 |       new WebAssembly.Module(parseText(add).emitBinary())
107 |     );
108 |     linker.instance("calculator", depsInstance);
109 |     var calc = await linker.instantiate(parseText(usingAdd).emitBinary());
110 | 
111 |     assert.equal(42, calc.instance.exports.add(1, 41), 42);
112 |   }
113 | 
114 |   // example of using linker.alias
115 |   {
116 |     var linker = new Linker();
117 |     // we first define the module that implements `add` as `calculator1`
118 |     await linker.module(
119 |       "calculator1",
120 |       new WebAssembly.Module(parseText(add).emitBinary())
121 |     );
122 |     // we alias `calculator1` as `calculator2`
123 |     linker.alias("calculator1", "calculator2");
124 |     var calc = await linker.instantiate(
125 |       parseText(usingAddWithAlias).emitBinary()
126 |     );
127 | 
128 |     assert.equal(calc.instance.exports.add1(1, 99), 100);
129 |     assert.equal(calc.instance.exports.add2(2, 99), 101);
130 |   }
131 | 
132 |   // example of using conventional import objects
133 |   {
134 |     let linker = new Linker();
135 |     var importObject = {
136 |       calculator1: {
137 |         add: (a, b) => a + b,
138 |       },
139 |       calculator2: {
140 |         add: (a, b) => a + b,
141 |       },
142 |     };
143 | 
144 |     linker.imports(importObject);
145 |     var calc = await linker.instantiate(
146 |       parseText(usingAddWithAlias).emitBinary()
147 |     );
148 | 
149 |     assert.equal(calc.instance.exports.add1(1, 99), 100);
150 |     assert.equal(calc.instance.exports.add2(2, 99), 101);
151 |   }
152 | 
153 |   // example of defining an asynchronous import,
154 |   // made possible with Binaryen and asyncify-wasm
155 |   {
156 |     var useAsyncify = true;
157 |     var linker = new Linker(useAsyncify);
158 | 
159 |     // notice how we define an asynchronous import, which
160 |     // will wait for 1.5s before returning the result
161 |     linker.define("calculator", "add", async (a, b) => {
162 |       await sleep(1500);
163 |       return a + b;
164 |     });
165 | 
166 |     let bytes = parseText(usingAdd);
167 | 
168 |     // we perform the asyncify compiler pass from Binaryen
169 |     bytes.runPasses(["asyncify"]);
170 |     var calc = await linker.instantiate(bytes.emitBinary());
171 | 
172 |     assert.equal(await calc.instance.exports.add(1, 2), 3);
173 |   }
174 | })();
175 | 
176 | async function sleep(ms) {
177 |   return new Promise((resolve) => {
178 |     setTimeout(resolve, ms);
179 |   });
180 | }
181 | 
--------------------------------------------------------------------------------
/examples/using_calculator.wasm:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/deislabs/wasm-linker-js/0ab83e04387980c8b6a932452d56cf8ca24d11d0/examples/using_calculator.wasm
--------------------------------------------------------------------------------
/examples/using_calculator_async.wasm:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/deislabs/wasm-linker-js/0ab83e04387980c8b6a932452d56cf8ca24d11d0/examples/using_calculator_async.wasm
--------------------------------------------------------------------------------
/examples/wasi-example.js:
--------------------------------------------------------------------------------
 1 | "use strict";
 2 | const { WASI } = require("wasi");
 3 | const { Linker } = require("../dist/src/index");
 4 | const fs = require("fs");
 5 | const wasi = new WASI();
 6 | 
 7 | (async () => {
 8 |   var linker = new Linker();
 9 |   linker.imports({ wasi_snapshot_preview1: wasi.wasiImport });
10 | 
11 |   const wasi_hello = await WebAssembly.compile(
12 |     fs.readFileSync("./wasi_hello.wasm")
13 |   );
14 | 
15 |   var instance = await linker.instantiate(wasi_hello);
16 |   wasi.start(instance);
17 | })();
18 | 
--------------------------------------------------------------------------------
/examples/wasi_hello.wasm:
--------------------------------------------------------------------------------
1 |  asm   ``  #wasi_snapshot_preview1fd_write   memory _start 
2 |  A A6 AA6 AA AA  Ahello world
3 |  name fd_writemain
--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
   1 | {
   2 |   "name": "@deislabs/wasm-linker-js",
   3 |   "version": "0.2.1",
   4 |   "lockfileVersion": 1,
   5 |   "requires": true,
   6 |   "dependencies": {
   7 |     "@discoveryjs/json-ext": {
   8 |       "version": "0.5.3",
   9 |       "resolved": "https://registry.npmjs.org/@discoveryjs/json-ext/-/json-ext-0.5.3.tgz",
  10 |       "integrity": "sha512-Fxt+AfXgjMoin2maPIYzFZnQjAXjAL0PHscM5pRTtatFqB+vZxAM9tLp2Optnuw3QOQC40jTNeGYFOMvyf7v9g==",
  11 |       "dev": true
  12 |     },
  13 |     "@types/chai": {
  14 |       "version": "4.2.14",
  15 |       "resolved": "https://registry.npmjs.org/@types/chai/-/chai-4.2.14.tgz",
  16 |       "integrity": "sha512-G+ITQPXkwTrslfG5L/BksmbLUA0M1iybEsmCWPqzSxsRRhJZimBKJkoMi8fr/CPygPTj4zO5pJH7I2/cm9M7SQ==",
  17 |       "dev": true
  18 |     },
  19 |     "@types/eslint": {
  20 |       "version": "7.2.13",
  21 |       "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-7.2.13.tgz",
  22 |       "integrity": "sha512-LKmQCWAlnVHvvXq4oasNUMTJJb2GwSyTY8+1C7OH5ILR8mPLaljv1jxL1bXW3xB3jFbQxTKxJAvI8PyjB09aBg==",
  23 |       "dev": true,
  24 |       "requires": {
  25 |         "@types/estree": "*",
  26 |         "@types/json-schema": "*"
  27 |       }
  28 |     },
  29 |     "@types/eslint-scope": {
  30 |       "version": "3.7.0",
  31 |       "resolved": "https://registry.npmjs.org/@types/eslint-scope/-/eslint-scope-3.7.0.tgz",
  32 |       "integrity": "sha512-O/ql2+rrCUe2W2rs7wMR+GqPRcgB6UiqN5RhrR5xruFlY7l9YLMn0ZkDzjoHLeiFkR8MCQZVudUuuvQ2BLC9Qw==",
  33 |       "dev": true,
  34 |       "requires": {
  35 |         "@types/eslint": "*",
  36 |         "@types/estree": "*"
  37 |       }
  38 |     },
  39 |     "@types/estree": {
  40 |       "version": "0.0.47",
  41 |       "resolved": "https://registry.npmjs.org/@types/estree/-/estree-0.0.47.tgz",
  42 |       "integrity": "sha512-c5ciR06jK8u9BstrmJyO97m+klJrrhCf9u3rLu3DEAJBirxRqSCvDQoYKmxuYwQI5SZChAWu+tq9oVlGRuzPAg==",
  43 |       "dev": true
  44 |     },
  45 |     "@types/json-schema": {
  46 |       "version": "7.0.7",
  47 |       "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.7.tgz",
  48 |       "integrity": "sha512-cxWFQVseBm6O9Gbw1IWb8r6OS4OhSt3hPZLkFApLjM8TEXROBuQGLAH2i2gZpcXdLBIrpXuTDhH7Vbm1iXmNGA==",
  49 |       "dev": true
  50 |     },
  51 |     "@types/mocha": {
  52 |       "version": "8.2.0",
  53 |       "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-8.2.0.tgz",
  54 |       "integrity": "sha512-/Sge3BymXo4lKc31C8OINJgXLaw+7vL1/L1pGiBNpGrBiT8FQiaFpSYV0uhTaG4y78vcMBTMFsWaHDvuD+xGzQ==",
  55 |       "dev": true
  56 |     },
  57 |     "@types/node": {
  58 |       "version": "14.14.14",
  59 |       "resolved": "https://registry.npmjs.org/@types/node/-/node-14.14.14.tgz",
  60 |       "integrity": "sha512-UHnOPWVWV1z+VV8k6L1HhG7UbGBgIdghqF3l9Ny9ApPghbjICXkUJSd/b9gOgQfjM1r+37cipdw/HJ3F6ICEnQ==",
  61 |       "dev": true
  62 |     },
  63 |     "@types/node-fetch": {
  64 |       "version": "2.5.7",
  65 |       "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.5.7.tgz",
  66 |       "integrity": "sha512-o2WVNf5UhWRkxlf6eq+jMZDu7kjgpgJfl4xVNlvryc95O/6F2ld8ztKX+qu+Rjyet93WAWm5LjeX9H5FGkODvw==",
  67 |       "dev": true,
  68 |       "requires": {
  69 |         "@types/node": "*",
  70 |         "form-data": "^3.0.0"
  71 |       }
  72 |     },
  73 |     "@types/uuid": {
  74 |       "version": "8.3.0",
  75 |       "resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-8.3.0.tgz",
  76 |       "integrity": "sha512-eQ9qFW/fhfGJF8WKHGEHZEyVWfZxrT+6CLIJGBcZPfxUh/+BnEj+UCGYMlr9qZuX/2AltsvwrGqp0LhEW8D0zQ==",
  77 |       "dev": true
  78 |     },
  79 |     "@ungap/promise-all-settled": {
  80 |       "version": "1.1.2",
  81 |       "resolved": "https://registry.npmjs.org/@ungap/promise-all-settled/-/promise-all-settled-1.1.2.tgz",
  82 |       "integrity": "sha512-sL/cEvJWAnClXw0wHk85/2L0G6Sj8UB0Ctc1TEMbKSsmpRosqhwj9gWgFRZSrBr2f9tiXISwNhCPmlfqUqyb9Q==",
  83 |       "dev": true
  84 |     },
  85 |     "@webassemblyjs/ast": {
  86 |       "version": "1.11.0",
  87 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.11.0.tgz",
  88 |       "integrity": "sha512-kX2W49LWsbthrmIRMbQZuQDhGtjyqXfEmmHyEi4XWnSZtPmxY0+3anPIzsnRb45VH/J55zlOfWvZuY47aJZTJg==",
  89 |       "dev": true,
  90 |       "requires": {
  91 |         "@webassemblyjs/helper-numbers": "1.11.0",
  92 |         "@webassemblyjs/helper-wasm-bytecode": "1.11.0"
  93 |       }
  94 |     },
  95 |     "@webassemblyjs/floating-point-hex-parser": {
  96 |       "version": "1.11.0",
  97 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.11.0.tgz",
  98 |       "integrity": "sha512-Q/aVYs/VnPDVYvsCBL/gSgwmfjeCb4LW8+TMrO3cSzJImgv8lxxEPM2JA5jMrivE7LSz3V+PFqtMbls3m1exDA==",
  99 |       "dev": true
 100 |     },
 101 |     "@webassemblyjs/helper-api-error": {
 102 |       "version": "1.11.0",
 103 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-api-error/-/helper-api-error-1.11.0.tgz",
 104 |       "integrity": "sha512-baT/va95eXiXb2QflSx95QGT5ClzWpGaa8L7JnJbgzoYeaA27FCvuBXU758l+KXWRndEmUXjP0Q5fibhavIn8w==",
 105 |       "dev": true
 106 |     },
 107 |     "@webassemblyjs/helper-buffer": {
 108 |       "version": "1.11.0",
 109 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-buffer/-/helper-buffer-1.11.0.tgz",
 110 |       "integrity": "sha512-u9HPBEl4DS+vA8qLQdEQ6N/eJQ7gT7aNvMIo8AAWvAl/xMrcOSiI2M0MAnMCy3jIFke7bEee/JwdX1nUpCtdyA==",
 111 |       "dev": true
 112 |     },
 113 |     "@webassemblyjs/helper-numbers": {
 114 |       "version": "1.11.0",
 115 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-numbers/-/helper-numbers-1.11.0.tgz",
 116 |       "integrity": "sha512-DhRQKelIj01s5IgdsOJMKLppI+4zpmcMQ3XboFPLwCpSNH6Hqo1ritgHgD0nqHeSYqofA6aBN/NmXuGjM1jEfQ==",
 117 |       "dev": true,
 118 |       "requires": {
 119 |         "@webassemblyjs/floating-point-hex-parser": "1.11.0",
 120 |         "@webassemblyjs/helper-api-error": "1.11.0",
 121 |         "@xtuc/long": "4.2.2"
 122 |       }
 123 |     },
 124 |     "@webassemblyjs/helper-wasm-bytecode": {
 125 |       "version": "1.11.0",
 126 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.11.0.tgz",
 127 |       "integrity": "sha512-MbmhvxXExm542tWREgSFnOVo07fDpsBJg3sIl6fSp9xuu75eGz5lz31q7wTLffwL3Za7XNRCMZy210+tnsUSEA==",
 128 |       "dev": true
 129 |     },
 130 |     "@webassemblyjs/helper-wasm-section": {
 131 |       "version": "1.11.0",
 132 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.11.0.tgz",
 133 |       "integrity": "sha512-3Eb88hcbfY/FCukrg6i3EH8H2UsD7x8Vy47iVJrP967A9JGqgBVL9aH71SETPx1JrGsOUVLo0c7vMCN22ytJew==",
 134 |       "dev": true,
 135 |       "requires": {
 136 |         "@webassemblyjs/ast": "1.11.0",
 137 |         "@webassemblyjs/helper-buffer": "1.11.0",
 138 |         "@webassemblyjs/helper-wasm-bytecode": "1.11.0",
 139 |         "@webassemblyjs/wasm-gen": "1.11.0"
 140 |       }
 141 |     },
 142 |     "@webassemblyjs/ieee754": {
 143 |       "version": "1.11.0",
 144 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/ieee754/-/ieee754-1.11.0.tgz",
 145 |       "integrity": "sha512-KXzOqpcYQwAfeQ6WbF6HXo+0udBNmw0iXDmEK5sFlmQdmND+tr773Ti8/5T/M6Tl/413ArSJErATd8In3B+WBA==",
 146 |       "dev": true,
 147 |       "requires": {
 148 |         "@xtuc/ieee754": "^1.2.0"
 149 |       }
 150 |     },
 151 |     "@webassemblyjs/leb128": {
 152 |       "version": "1.11.0",
 153 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/leb128/-/leb128-1.11.0.tgz",
 154 |       "integrity": "sha512-aqbsHa1mSQAbeeNcl38un6qVY++hh8OpCOzxhixSYgbRfNWcxJNJQwe2rezK9XEcssJbbWIkblaJRwGMS9zp+g==",
 155 |       "dev": true,
 156 |       "requires": {
 157 |         "@xtuc/long": "4.2.2"
 158 |       }
 159 |     },
 160 |     "@webassemblyjs/utf8": {
 161 |       "version": "1.11.0",
 162 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/utf8/-/utf8-1.11.0.tgz",
 163 |       "integrity": "sha512-A/lclGxH6SpSLSyFowMzO/+aDEPU4hvEiooCMXQPcQFPPJaYcPQNKGOCLUySJsYJ4trbpr+Fs08n4jelkVTGVw==",
 164 |       "dev": true
 165 |     },
 166 |     "@webassemblyjs/wasm-edit": {
 167 |       "version": "1.11.0",
 168 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-edit/-/wasm-edit-1.11.0.tgz",
 169 |       "integrity": "sha512-JHQ0damXy0G6J9ucyKVXO2j08JVJ2ntkdJlq1UTiUrIgfGMmA7Ik5VdC/L8hBK46kVJgujkBIoMtT8yVr+yVOQ==",
 170 |       "dev": true,
 171 |       "requires": {
 172 |         "@webassemblyjs/ast": "1.11.0",
 173 |         "@webassemblyjs/helper-buffer": "1.11.0",
 174 |         "@webassemblyjs/helper-wasm-bytecode": "1.11.0",
 175 |         "@webassemblyjs/helper-wasm-section": "1.11.0",
 176 |         "@webassemblyjs/wasm-gen": "1.11.0",
 177 |         "@webassemblyjs/wasm-opt": "1.11.0",
 178 |         "@webassemblyjs/wasm-parser": "1.11.0",
 179 |         "@webassemblyjs/wast-printer": "1.11.0"
 180 |       }
 181 |     },
 182 |     "@webassemblyjs/wasm-gen": {
 183 |       "version": "1.11.0",
 184 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-gen/-/wasm-gen-1.11.0.tgz",
 185 |       "integrity": "sha512-BEUv1aj0WptCZ9kIS30th5ILASUnAPEvE3tVMTrItnZRT9tXCLW2LEXT8ezLw59rqPP9klh9LPmpU+WmRQmCPQ==",
 186 |       "dev": true,
 187 |       "requires": {
 188 |         "@webassemblyjs/ast": "1.11.0",
 189 |         "@webassemblyjs/helper-wasm-bytecode": "1.11.0",
 190 |         "@webassemblyjs/ieee754": "1.11.0",
 191 |         "@webassemblyjs/leb128": "1.11.0",
 192 |         "@webassemblyjs/utf8": "1.11.0"
 193 |       }
 194 |     },
 195 |     "@webassemblyjs/wasm-opt": {
 196 |       "version": "1.11.0",
 197 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-opt/-/wasm-opt-1.11.0.tgz",
 198 |       "integrity": "sha512-tHUSP5F4ywyh3hZ0+fDQuWxKx3mJiPeFufg+9gwTpYp324mPCQgnuVKwzLTZVqj0duRDovnPaZqDwoyhIO8kYg==",
 199 |       "dev": true,
 200 |       "requires": {
 201 |         "@webassemblyjs/ast": "1.11.0",
 202 |         "@webassemblyjs/helper-buffer": "1.11.0",
 203 |         "@webassemblyjs/wasm-gen": "1.11.0",
 204 |         "@webassemblyjs/wasm-parser": "1.11.0"
 205 |       }
 206 |     },
 207 |     "@webassemblyjs/wasm-parser": {
 208 |       "version": "1.11.0",
 209 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-parser/-/wasm-parser-1.11.0.tgz",
 210 |       "integrity": "sha512-6L285Sgu9gphrcpDXINvm0M9BskznnzJTE7gYkjDbxET28shDqp27wpruyx3C2S/dvEwiigBwLA1cz7lNUi0kw==",
 211 |       "dev": true,
 212 |       "requires": {
 213 |         "@webassemblyjs/ast": "1.11.0",
 214 |         "@webassemblyjs/helper-api-error": "1.11.0",
 215 |         "@webassemblyjs/helper-wasm-bytecode": "1.11.0",
 216 |         "@webassemblyjs/ieee754": "1.11.0",
 217 |         "@webassemblyjs/leb128": "1.11.0",
 218 |         "@webassemblyjs/utf8": "1.11.0"
 219 |       }
 220 |     },
 221 |     "@webassemblyjs/wast-printer": {
 222 |       "version": "1.11.0",
 223 |       "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-printer/-/wast-printer-1.11.0.tgz",
 224 |       "integrity": "sha512-Fg5OX46pRdTgB7rKIUojkh9vXaVN6sGYCnEiJN1GYkb0RPwShZXp6KTDqmoMdQPKhcroOXh3fEzmkWmCYaKYhQ==",
 225 |       "dev": true,
 226 |       "requires": {
 227 |         "@webassemblyjs/ast": "1.11.0",
 228 |         "@xtuc/long": "4.2.2"
 229 |       }
 230 |     },
 231 |     "@webpack-cli/configtest": {
 232 |       "version": "1.0.4",
 233 |       "resolved": "https://registry.npmjs.org/@webpack-cli/configtest/-/configtest-1.0.4.tgz",
 234 |       "integrity": "sha512-cs3XLy+UcxiP6bj0A6u7MLLuwdXJ1c3Dtc0RkKg+wiI1g/Ti1om8+/2hc2A2B60NbBNAbMgyBMHvyymWm/j4wQ==",
 235 |       "dev": true
 236 |     },
 237 |     "@webpack-cli/info": {
 238 |       "version": "1.3.0",
 239 |       "resolved": "https://registry.npmjs.org/@webpack-cli/info/-/info-1.3.0.tgz",
 240 |       "integrity": "sha512-ASiVB3t9LOKHs5DyVUcxpraBXDOKubYu/ihHhU+t1UPpxsivg6Od2E2qU4gJCekfEddzRBzHhzA/Acyw/mlK/w==",
 241 |       "dev": true,
 242 |       "requires": {
 243 |         "envinfo": "^7.7.3"
 244 |       }
 245 |     },
 246 |     "@webpack-cli/serve": {
 247 |       "version": "1.5.1",
 248 |       "resolved": "https://registry.npmjs.org/@webpack-cli/serve/-/serve-1.5.1.tgz",
 249 |       "integrity": "sha512-4vSVUiOPJLmr45S8rMGy7WDvpWxfFxfP/Qx/cxZFCfvoypTYpPPL1X8VIZMe0WTA+Jr7blUxwUSEZNkjoMTgSw==",
 250 |       "dev": true
 251 |     },
 252 |     "@xtuc/ieee754": {
 253 |       "version": "1.2.0",
 254 |       "resolved": "https://registry.npmjs.org/@xtuc/ieee754/-/ieee754-1.2.0.tgz",
 255 |       "integrity": "sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==",
 256 |       "dev": true
 257 |     },
 258 |     "@xtuc/long": {
 259 |       "version": "4.2.2",
 260 |       "resolved": "https://registry.npmjs.org/@xtuc/long/-/long-4.2.2.tgz",
 261 |       "integrity": "sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==",
 262 |       "dev": true
 263 |     },
 264 |     "accepts": {
 265 |       "version": "1.3.7",
 266 |       "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
 267 |       "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
 268 |       "dev": true,
 269 |       "requires": {
 270 |         "mime-types": "~2.1.24",
 271 |         "negotiator": "0.6.2"
 272 |       }
 273 |     },
 274 |     "acorn": {
 275 |       "version": "8.4.0",
 276 |       "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.4.0.tgz",
 277 |       "integrity": "sha512-ULr0LDaEqQrMFGyQ3bhJkLsbtrQ8QibAseGZeaSUiT/6zb9IvIkomWHJIvgvwad+hinRAgsI51JcWk2yvwyL+w==",
 278 |       "dev": true
 279 |     },
 280 |     "ajv": {
 281 |       "version": "6.12.6",
 282 |       "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
 283 |       "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
 284 |       "dev": true,
 285 |       "requires": {
 286 |         "fast-deep-equal": "^3.1.1",
 287 |         "fast-json-stable-stringify": "^2.0.0",
 288 |         "json-schema-traverse": "^0.4.1",
 289 |         "uri-js": "^4.2.2"
 290 |       }
 291 |     },
 292 |     "ajv-keywords": {
 293 |       "version": "3.5.2",
 294 |       "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.5.2.tgz",
 295 |       "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==",
 296 |       "dev": true
 297 |     },
 298 |     "ansi-colors": {
 299 |       "version": "4.1.1",
 300 |       "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz",
 301 |       "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==",
 302 |       "dev": true
 303 |     },
 304 |     "ansi-regex": {
 305 |       "version": "3.0.0",
 306 |       "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz",
 307 |       "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=",
 308 |       "dev": true
 309 |     },
 310 |     "ansi-styles": {
 311 |       "version": "3.2.1",
 312 |       "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
 313 |       "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
 314 |       "dev": true,
 315 |       "requires": {
 316 |         "color-convert": "^1.9.0"
 317 |       }
 318 |     },
 319 |     "anymatch": {
 320 |       "version": "3.1.1",
 321 |       "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz",
 322 |       "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==",
 323 |       "dev": true,
 324 |       "requires": {
 325 |         "normalize-path": "^3.0.0",
 326 |         "picomatch": "^2.0.4"
 327 |       }
 328 |     },
 329 |     "arg": {
 330 |       "version": "4.1.3",
 331 |       "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz",
 332 |       "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==",
 333 |       "dev": true
 334 |     },
 335 |     "argparse": {
 336 |       "version": "1.0.10",
 337 |       "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz",
 338 |       "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==",
 339 |       "dev": true,
 340 |       "requires": {
 341 |         "sprintf-js": "~1.0.2"
 342 |       }
 343 |     },
 344 |     "array-flatten": {
 345 |       "version": "1.1.1",
 346 |       "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
 347 |       "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=",
 348 |       "dev": true
 349 |     },
 350 |     "assertion-error": {
 351 |       "version": "1.1.0",
 352 |       "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz",
 353 |       "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==",
 354 |       "dev": true
 355 |     },
 356 |     "asyncify-wasm": {
 357 |       "version": "1.1.2",
 358 |       "resolved": "https://registry.npmjs.org/asyncify-wasm/-/asyncify-wasm-1.1.2.tgz",
 359 |       "integrity": "sha512-KGrv8/ZcxRgnQRTL3IYidpA7cFMtVozIRopd2lw/LgCSUWVtxDg93QQ7ulll9kGpq2Vf/RevGU503dRYXxqF+Q=="
 360 |     },
 361 |     "asynckit": {
 362 |       "version": "0.4.0",
 363 |       "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
 364 |       "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=",
 365 |       "dev": true
 366 |     },
 367 |     "balanced-match": {
 368 |       "version": "1.0.0",
 369 |       "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
 370 |       "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=",
 371 |       "dev": true
 372 |     },
 373 |     "binary-extensions": {
 374 |       "version": "2.1.0",
 375 |       "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz",
 376 |       "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==",
 377 |       "dev": true
 378 |     },
 379 |     "binaryen": {
 380 |       "version": "98.0.0",
 381 |       "resolved": "https://registry.npmjs.org/binaryen/-/binaryen-98.0.0.tgz",
 382 |       "integrity": "sha512-8wIRIp+uBCL1XpJA9F+zL9TyEsPOIEIaru2KqEPls+nBHx/y64L1isybaOQ+myzn6QgNpSEYMaSH10gaK4ddJg==",
 383 |       "dev": true
 384 |     },
 385 |     "body-parser": {
 386 |       "version": "1.19.0",
 387 |       "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
 388 |       "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
 389 |       "dev": true,
 390 |       "requires": {
 391 |         "bytes": "3.1.0",
 392 |         "content-type": "~1.0.4",
 393 |         "debug": "2.6.9",
 394 |         "depd": "~1.1.2",
 395 |         "http-errors": "1.7.2",
 396 |         "iconv-lite": "0.4.24",
 397 |         "on-finished": "~2.3.0",
 398 |         "qs": "6.7.0",
 399 |         "raw-body": "2.4.0",
 400 |         "type-is": "~1.6.17"
 401 |       }
 402 |     },
 403 |     "brace-expansion": {
 404 |       "version": "1.1.11",
 405 |       "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
 406 |       "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
 407 |       "dev": true,
 408 |       "requires": {
 409 |         "balanced-match": "^1.0.0",
 410 |         "concat-map": "0.0.1"
 411 |       }
 412 |     },
 413 |     "braces": {
 414 |       "version": "3.0.2",
 415 |       "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
 416 |       "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
 417 |       "dev": true,
 418 |       "requires": {
 419 |         "fill-range": "^7.0.1"
 420 |       }
 421 |     },
 422 |     "browser-stdout": {
 423 |       "version": "1.3.1",
 424 |       "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz",
 425 |       "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==",
 426 |       "dev": true
 427 |     },
 428 |     "browserslist": {
 429 |       "version": "4.16.6",
 430 |       "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.16.6.tgz",
 431 |       "integrity": "sha512-Wspk/PqO+4W9qp5iUTJsa1B/QrYn1keNCcEP5OvP7WBwT4KaDly0uONYmC6Xa3Z5IqnUgS0KcgLYu1l74x0ZXQ==",
 432 |       "dev": true,
 433 |       "requires": {
 434 |         "caniuse-lite": "^1.0.30001219",
 435 |         "colorette": "^1.2.2",
 436 |         "electron-to-chromium": "^1.3.723",
 437 |         "escalade": "^3.1.1",
 438 |         "node-releases": "^1.1.71"
 439 |       }
 440 |     },
 441 |     "buffer-from": {
 442 |       "version": "1.1.1",
 443 |       "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz",
 444 |       "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==",
 445 |       "dev": true
 446 |     },
 447 |     "bytes": {
 448 |       "version": "3.1.0",
 449 |       "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
 450 |       "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==",
 451 |       "dev": true
 452 |     },
 453 |     "camelcase": {
 454 |       "version": "5.3.1",
 455 |       "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz",
 456 |       "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==",
 457 |       "dev": true
 458 |     },
 459 |     "caniuse-lite": {
 460 |       "version": "1.0.30001236",
 461 |       "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001236.tgz",
 462 |       "integrity": "sha512-o0PRQSrSCGJKCPZcgMzl5fUaj5xHe8qA2m4QRvnyY4e1lITqoNkr7q/Oh1NcpGSy0Th97UZ35yoKcINPoq7YOQ==",
 463 |       "dev": true
 464 |     },
 465 |     "chai": {
 466 |       "version": "4.2.0",
 467 |       "resolved": "https://registry.npmjs.org/chai/-/chai-4.2.0.tgz",
 468 |       "integrity": "sha512-XQU3bhBukrOsQCuwZndwGcCVQHyZi53fQ6Ys1Fym7E4olpIqqZZhhoFJoaKVvV17lWQoXYwgWN2nF5crA8J2jw==",
 469 |       "dev": true,
 470 |       "requires": {
 471 |         "assertion-error": "^1.1.0",
 472 |         "check-error": "^1.0.2",
 473 |         "deep-eql": "^3.0.1",
 474 |         "get-func-name": "^2.0.0",
 475 |         "pathval": "^1.1.0",
 476 |         "type-detect": "^4.0.5"
 477 |       }
 478 |     },
 479 |     "chalk": {
 480 |       "version": "4.1.1",
 481 |       "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz",
 482 |       "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==",
 483 |       "dev": true,
 484 |       "requires": {
 485 |         "ansi-styles": "^4.1.0",
 486 |         "supports-color": "^7.1.0"
 487 |       },
 488 |       "dependencies": {
 489 |         "ansi-styles": {
 490 |           "version": "4.3.0",
 491 |           "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
 492 |           "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
 493 |           "dev": true,
 494 |           "requires": {
 495 |             "color-convert": "^2.0.1"
 496 |           }
 497 |         },
 498 |         "color-convert": {
 499 |           "version": "2.0.1",
 500 |           "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
 501 |           "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
 502 |           "dev": true,
 503 |           "requires": {
 504 |             "color-name": "~1.1.4"
 505 |           }
 506 |         },
 507 |         "color-name": {
 508 |           "version": "1.1.4",
 509 |           "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
 510 |           "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
 511 |           "dev": true
 512 |         }
 513 |       }
 514 |     },
 515 |     "check-error": {
 516 |       "version": "1.0.2",
 517 |       "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz",
 518 |       "integrity": "sha1-V00xLt2Iu13YkS6Sht1sCu1KrII=",
 519 |       "dev": true
 520 |     },
 521 |     "chokidar": {
 522 |       "version": "3.4.3",
 523 |       "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.3.tgz",
 524 |       "integrity": "sha512-DtM3g7juCXQxFVSNPNByEC2+NImtBuxQQvWlHunpJIS5Ocr0lG306cC7FCi7cEA0fzmybPUIl4txBIobk1gGOQ==",
 525 |       "dev": true,
 526 |       "requires": {
 527 |         "anymatch": "~3.1.1",
 528 |         "braces": "~3.0.2",
 529 |         "fsevents": "~2.1.2",
 530 |         "glob-parent": "~5.1.0",
 531 |         "is-binary-path": "~2.1.0",
 532 |         "is-glob": "~4.0.1",
 533 |         "normalize-path": "~3.0.0",
 534 |         "readdirp": "~3.5.0"
 535 |       }
 536 |     },
 537 |     "chrome-trace-event": {
 538 |       "version": "1.0.3",
 539 |       "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.3.tgz",
 540 |       "integrity": "sha512-p3KULyQg4S7NIHixdwbGX+nFHkoBiA4YQmyWtjb8XngSKV124nJmRysgAeujbUVb15vh+RvFUfCPqU7rXk+hZg==",
 541 |       "dev": true
 542 |     },
 543 |     "cliui": {
 544 |       "version": "5.0.0",
 545 |       "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz",
 546 |       "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==",
 547 |       "dev": true,
 548 |       "requires": {
 549 |         "string-width": "^3.1.0",
 550 |         "strip-ansi": "^5.2.0",
 551 |         "wrap-ansi": "^5.1.0"
 552 |       },
 553 |       "dependencies": {
 554 |         "ansi-regex": {
 555 |           "version": "4.1.0",
 556 |           "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
 557 |           "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
 558 |           "dev": true
 559 |         },
 560 |         "string-width": {
 561 |           "version": "3.1.0",
 562 |           "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
 563 |           "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
 564 |           "dev": true,
 565 |           "requires": {
 566 |             "emoji-regex": "^7.0.1",
 567 |             "is-fullwidth-code-point": "^2.0.0",
 568 |             "strip-ansi": "^5.1.0"
 569 |           }
 570 |         },
 571 |         "strip-ansi": {
 572 |           "version": "5.2.0",
 573 |           "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
 574 |           "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
 575 |           "dev": true,
 576 |           "requires": {
 577 |             "ansi-regex": "^4.1.0"
 578 |           }
 579 |         }
 580 |       }
 581 |     },
 582 |     "clone-deep": {
 583 |       "version": "4.0.1",
 584 |       "resolved": "https://registry.npmjs.org/clone-deep/-/clone-deep-4.0.1.tgz",
 585 |       "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==",
 586 |       "dev": true,
 587 |       "requires": {
 588 |         "is-plain-object": "^2.0.4",
 589 |         "kind-of": "^6.0.2",
 590 |         "shallow-clone": "^3.0.0"
 591 |       }
 592 |     },
 593 |     "color-convert": {
 594 |       "version": "1.9.3",
 595 |       "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
 596 |       "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
 597 |       "dev": true,
 598 |       "requires": {
 599 |         "color-name": "1.1.3"
 600 |       }
 601 |     },
 602 |     "color-name": {
 603 |       "version": "1.1.3",
 604 |       "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
 605 |       "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=",
 606 |       "dev": true
 607 |     },
 608 |     "colorette": {
 609 |       "version": "1.2.2",
 610 |       "resolved": "https://registry.npmjs.org/colorette/-/colorette-1.2.2.tgz",
 611 |       "integrity": "sha512-MKGMzyfeuutC/ZJ1cba9NqcNpfeqMUcYmyF1ZFY6/Cn7CNSAKx6a+s48sqLqyAiZuaP2TcqMhoo+dlwFnVxT9w==",
 612 |       "dev": true
 613 |     },
 614 |     "combined-stream": {
 615 |       "version": "1.0.8",
 616 |       "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
 617 |       "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
 618 |       "dev": true,
 619 |       "requires": {
 620 |         "delayed-stream": "~1.0.0"
 621 |       }
 622 |     },
 623 |     "commander": {
 624 |       "version": "2.20.3",
 625 |       "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz",
 626 |       "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==",
 627 |       "dev": true
 628 |     },
 629 |     "concat-map": {
 630 |       "version": "0.0.1",
 631 |       "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
 632 |       "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
 633 |       "dev": true
 634 |     },
 635 |     "content-disposition": {
 636 |       "version": "0.5.3",
 637 |       "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
 638 |       "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
 639 |       "dev": true,
 640 |       "requires": {
 641 |         "safe-buffer": "5.1.2"
 642 |       }
 643 |     },
 644 |     "content-type": {
 645 |       "version": "1.0.4",
 646 |       "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
 647 |       "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==",
 648 |       "dev": true
 649 |     },
 650 |     "cookie": {
 651 |       "version": "0.4.0",
 652 |       "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
 653 |       "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==",
 654 |       "dev": true
 655 |     },
 656 |     "cookie-signature": {
 657 |       "version": "1.0.6",
 658 |       "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
 659 |       "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=",
 660 |       "dev": true
 661 |     },
 662 |     "create-require": {
 663 |       "version": "1.1.1",
 664 |       "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz",
 665 |       "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==",
 666 |       "dev": true
 667 |     },
 668 |     "cross-spawn": {
 669 |       "version": "7.0.3",
 670 |       "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
 671 |       "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==",
 672 |       "dev": true,
 673 |       "requires": {
 674 |         "path-key": "^3.1.0",
 675 |         "shebang-command": "^2.0.0",
 676 |         "which": "^2.0.1"
 677 |       }
 678 |     },
 679 |     "debug": {
 680 |       "version": "2.6.9",
 681 |       "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
 682 |       "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
 683 |       "dev": true,
 684 |       "requires": {
 685 |         "ms": "2.0.0"
 686 |       }
 687 |     },
 688 |     "decamelize": {
 689 |       "version": "1.2.0",
 690 |       "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz",
 691 |       "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=",
 692 |       "dev": true
 693 |     },
 694 |     "deep-eql": {
 695 |       "version": "3.0.1",
 696 |       "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-3.0.1.tgz",
 697 |       "integrity": "sha512-+QeIQyN5ZuO+3Uk5DYh6/1eKO0m0YmJFGNmFHGACpf1ClL1nmlV/p4gNgbl2pJGxgXb4faqo6UE+M5ACEMyVcw==",
 698 |       "dev": true,
 699 |       "requires": {
 700 |         "type-detect": "^4.0.0"
 701 |       }
 702 |     },
 703 |     "delayed-stream": {
 704 |       "version": "1.0.0",
 705 |       "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
 706 |       "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=",
 707 |       "dev": true
 708 |     },
 709 |     "depd": {
 710 |       "version": "1.1.2",
 711 |       "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
 712 |       "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=",
 713 |       "dev": true
 714 |     },
 715 |     "destroy": {
 716 |       "version": "1.0.4",
 717 |       "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
 718 |       "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=",
 719 |       "dev": true
 720 |     },
 721 |     "diff": {
 722 |       "version": "4.0.2",
 723 |       "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz",
 724 |       "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==",
 725 |       "dev": true
 726 |     },
 727 |     "ee-first": {
 728 |       "version": "1.1.1",
 729 |       "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
 730 |       "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=",
 731 |       "dev": true
 732 |     },
 733 |     "electron-to-chromium": {
 734 |       "version": "1.3.752",
 735 |       "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.752.tgz",
 736 |       "integrity": "sha512-2Tg+7jSl3oPxgsBsWKh5H83QazTkmWG/cnNwJplmyZc7KcN61+I10oUgaXSVk/NwfvN3BdkKDR4FYuRBQQ2v0A==",
 737 |       "dev": true
 738 |     },
 739 |     "emoji-regex": {
 740 |       "version": "7.0.3",
 741 |       "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz",
 742 |       "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==",
 743 |       "dev": true
 744 |     },
 745 |     "encodeurl": {
 746 |       "version": "1.0.2",
 747 |       "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
 748 |       "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=",
 749 |       "dev": true
 750 |     },
 751 |     "enhanced-resolve": {
 752 |       "version": "5.8.2",
 753 |       "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.8.2.tgz",
 754 |       "integrity": "sha512-F27oB3WuHDzvR2DOGNTaYy0D5o0cnrv8TeI482VM4kYgQd/FT9lUQwuNsJ0oOHtBUq7eiW5ytqzp7nBFknL+GA==",
 755 |       "dev": true,
 756 |       "requires": {
 757 |         "graceful-fs": "^4.2.4",
 758 |         "tapable": "^2.2.0"
 759 |       }
 760 |     },
 761 |     "envinfo": {
 762 |       "version": "7.8.1",
 763 |       "resolved": "https://registry.npmjs.org/envinfo/-/envinfo-7.8.1.tgz",
 764 |       "integrity": "sha512-/o+BXHmB7ocbHEAs6F2EnG0ogybVVUdkRunTT2glZU9XAaGmhqskrvKwqXuDfNjEO0LZKWdejEEpnq8aM0tOaw==",
 765 |       "dev": true
 766 |     },
 767 |     "es-module-lexer": {
 768 |       "version": "0.4.1",
 769 |       "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-0.4.1.tgz",
 770 |       "integrity": "sha512-ooYciCUtfw6/d2w56UVeqHPcoCFAiJdz5XOkYpv/Txl1HMUozpXjz/2RIQgqwKdXNDPSF1W7mJCFse3G+HDyAA==",
 771 |       "dev": true
 772 |     },
 773 |     "escalade": {
 774 |       "version": "3.1.1",
 775 |       "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz",
 776 |       "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==",
 777 |       "dev": true
 778 |     },
 779 |     "escape-html": {
 780 |       "version": "1.0.3",
 781 |       "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
 782 |       "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=",
 783 |       "dev": true
 784 |     },
 785 |     "eslint-scope": {
 786 |       "version": "5.1.1",
 787 |       "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz",
 788 |       "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==",
 789 |       "dev": true,
 790 |       "requires": {
 791 |         "esrecurse": "^4.3.0",
 792 |         "estraverse": "^4.1.1"
 793 |       }
 794 |     },
 795 |     "esprima": {
 796 |       "version": "4.0.1",
 797 |       "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz",
 798 |       "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==",
 799 |       "dev": true
 800 |     },
 801 |     "esrecurse": {
 802 |       "version": "4.3.0",
 803 |       "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz",
 804 |       "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==",
 805 |       "dev": true,
 806 |       "requires": {
 807 |         "estraverse": "^5.2.0"
 808 |       },
 809 |       "dependencies": {
 810 |         "estraverse": {
 811 |           "version": "5.2.0",
 812 |           "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz",
 813 |           "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==",
 814 |           "dev": true
 815 |         }
 816 |       }
 817 |     },
 818 |     "estraverse": {
 819 |       "version": "4.3.0",
 820 |       "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz",
 821 |       "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==",
 822 |       "dev": true
 823 |     },
 824 |     "etag": {
 825 |       "version": "1.8.1",
 826 |       "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
 827 |       "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=",
 828 |       "dev": true
 829 |     },
 830 |     "events": {
 831 |       "version": "3.3.0",
 832 |       "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz",
 833 |       "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==",
 834 |       "dev": true
 835 |     },
 836 |     "execa": {
 837 |       "version": "5.1.1",
 838 |       "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz",
 839 |       "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==",
 840 |       "dev": true,
 841 |       "requires": {
 842 |         "cross-spawn": "^7.0.3",
 843 |         "get-stream": "^6.0.0",
 844 |         "human-signals": "^2.1.0",
 845 |         "is-stream": "^2.0.0",
 846 |         "merge-stream": "^2.0.0",
 847 |         "npm-run-path": "^4.0.1",
 848 |         "onetime": "^5.1.2",
 849 |         "signal-exit": "^3.0.3",
 850 |         "strip-final-newline": "^2.0.0"
 851 |       }
 852 |     },
 853 |     "express": {
 854 |       "version": "4.17.1",
 855 |       "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
 856 |       "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
 857 |       "dev": true,
 858 |       "requires": {
 859 |         "accepts": "~1.3.7",
 860 |         "array-flatten": "1.1.1",
 861 |         "body-parser": "1.19.0",
 862 |         "content-disposition": "0.5.3",
 863 |         "content-type": "~1.0.4",
 864 |         "cookie": "0.4.0",
 865 |         "cookie-signature": "1.0.6",
 866 |         "debug": "2.6.9",
 867 |         "depd": "~1.1.2",
 868 |         "encodeurl": "~1.0.2",
 869 |         "escape-html": "~1.0.3",
 870 |         "etag": "~1.8.1",
 871 |         "finalhandler": "~1.1.2",
 872 |         "fresh": "0.5.2",
 873 |         "merge-descriptors": "1.0.1",
 874 |         "methods": "~1.1.2",
 875 |         "on-finished": "~2.3.0",
 876 |         "parseurl": "~1.3.3",
 877 |         "path-to-regexp": "0.1.7",
 878 |         "proxy-addr": "~2.0.5",
 879 |         "qs": "6.7.0",
 880 |         "range-parser": "~1.2.1",
 881 |         "safe-buffer": "5.1.2",
 882 |         "send": "0.17.1",
 883 |         "serve-static": "1.14.1",
 884 |         "setprototypeof": "1.1.1",
 885 |         "statuses": "~1.5.0",
 886 |         "type-is": "~1.6.18",
 887 |         "utils-merge": "1.0.1",
 888 |         "vary": "~1.1.2"
 889 |       }
 890 |     },
 891 |     "fast-deep-equal": {
 892 |       "version": "3.1.3",
 893 |       "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
 894 |       "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==",
 895 |       "dev": true
 896 |     },
 897 |     "fast-json-stable-stringify": {
 898 |       "version": "2.1.0",
 899 |       "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
 900 |       "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==",
 901 |       "dev": true
 902 |     },
 903 |     "fastest-levenshtein": {
 904 |       "version": "1.0.12",
 905 |       "resolved": "https://registry.npmjs.org/fastest-levenshtein/-/fastest-levenshtein-1.0.12.tgz",
 906 |       "integrity": "sha512-On2N+BpYJ15xIC974QNVuYGMOlEVt4s0EOI3wwMqOmK1fdDY+FN/zltPV8vosq4ad4c/gJ1KHScUn/6AWIgiow==",
 907 |       "dev": true
 908 |     },
 909 |     "fill-range": {
 910 |       "version": "7.0.1",
 911 |       "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
 912 |       "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
 913 |       "dev": true,
 914 |       "requires": {
 915 |         "to-regex-range": "^5.0.1"
 916 |       }
 917 |     },
 918 |     "finalhandler": {
 919 |       "version": "1.1.2",
 920 |       "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
 921 |       "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
 922 |       "dev": true,
 923 |       "requires": {
 924 |         "debug": "2.6.9",
 925 |         "encodeurl": "~1.0.2",
 926 |         "escape-html": "~1.0.3",
 927 |         "on-finished": "~2.3.0",
 928 |         "parseurl": "~1.3.3",
 929 |         "statuses": "~1.5.0",
 930 |         "unpipe": "~1.0.0"
 931 |       }
 932 |     },
 933 |     "find-up": {
 934 |       "version": "3.0.0",
 935 |       "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz",
 936 |       "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==",
 937 |       "dev": true,
 938 |       "requires": {
 939 |         "locate-path": "^3.0.0"
 940 |       }
 941 |     },
 942 |     "flat": {
 943 |       "version": "5.0.2",
 944 |       "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz",
 945 |       "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==",
 946 |       "dev": true
 947 |     },
 948 |     "form-data": {
 949 |       "version": "3.0.0",
 950 |       "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.0.tgz",
 951 |       "integrity": "sha512-CKMFDglpbMi6PyN+brwB9Q/GOw0eAnsrEZDgcsH5Krhz5Od/haKHAX0NmQfha2zPPz0JpWzA7GJHGSnvCRLWsg==",
 952 |       "dev": true,
 953 |       "requires": {
 954 |         "asynckit": "^0.4.0",
 955 |         "combined-stream": "^1.0.8",
 956 |         "mime-types": "^2.1.12"
 957 |       }
 958 |     },
 959 |     "forwarded": {
 960 |       "version": "0.1.2",
 961 |       "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
 962 |       "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=",
 963 |       "dev": true
 964 |     },
 965 |     "fresh": {
 966 |       "version": "0.5.2",
 967 |       "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
 968 |       "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=",
 969 |       "dev": true
 970 |     },
 971 |     "fs.realpath": {
 972 |       "version": "1.0.0",
 973 |       "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
 974 |       "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=",
 975 |       "dev": true
 976 |     },
 977 |     "fsevents": {
 978 |       "version": "2.1.3",
 979 |       "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz",
 980 |       "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==",
 981 |       "dev": true,
 982 |       "optional": true
 983 |     },
 984 |     "function-bind": {
 985 |       "version": "1.1.1",
 986 |       "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
 987 |       "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==",
 988 |       "dev": true
 989 |     },
 990 |     "get-caller-file": {
 991 |       "version": "2.0.5",
 992 |       "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz",
 993 |       "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==",
 994 |       "dev": true
 995 |     },
 996 |     "get-func-name": {
 997 |       "version": "2.0.0",
 998 |       "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.0.tgz",
 999 |       "integrity": "sha1-6td0q+5y4gQJQzoGY2YCPdaIekE=",
1000 |       "dev": true
1001 |     },
1002 |     "get-stream": {
1003 |       "version": "6.0.1",
1004 |       "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz",
1005 |       "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==",
1006 |       "dev": true
1007 |     },
1008 |     "glob-parent": {
1009 |       "version": "5.1.2",
1010 |       "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
1011 |       "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
1012 |       "dev": true,
1013 |       "requires": {
1014 |         "is-glob": "^4.0.1"
1015 |       }
1016 |     },
1017 |     "glob-to-regexp": {
1018 |       "version": "0.4.1",
1019 |       "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz",
1020 |       "integrity": "sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw==",
1021 |       "dev": true
1022 |     },
1023 |     "graceful-fs": {
1024 |       "version": "4.2.4",
1025 |       "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz",
1026 |       "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==",
1027 |       "dev": true
1028 |     },
1029 |     "growl": {
1030 |       "version": "1.10.5",
1031 |       "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz",
1032 |       "integrity": "sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==",
1033 |       "dev": true
1034 |     },
1035 |     "has": {
1036 |       "version": "1.0.3",
1037 |       "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz",
1038 |       "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==",
1039 |       "dev": true,
1040 |       "requires": {
1041 |         "function-bind": "^1.1.1"
1042 |       }
1043 |     },
1044 |     "he": {
1045 |       "version": "1.2.0",
1046 |       "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz",
1047 |       "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==",
1048 |       "dev": true
1049 |     },
1050 |     "http-errors": {
1051 |       "version": "1.7.2",
1052 |       "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
1053 |       "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
1054 |       "dev": true,
1055 |       "requires": {
1056 |         "depd": "~1.1.2",
1057 |         "inherits": "2.0.3",
1058 |         "setprototypeof": "1.1.1",
1059 |         "statuses": ">= 1.5.0 < 2",
1060 |         "toidentifier": "1.0.0"
1061 |       }
1062 |     },
1063 |     "human-signals": {
1064 |       "version": "2.1.0",
1065 |       "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz",
1066 |       "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==",
1067 |       "dev": true
1068 |     },
1069 |     "iconv-lite": {
1070 |       "version": "0.4.24",
1071 |       "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
1072 |       "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
1073 |       "dev": true,
1074 |       "requires": {
1075 |         "safer-buffer": ">= 2.1.2 < 3"
1076 |       }
1077 |     },
1078 |     "import-local": {
1079 |       "version": "3.0.2",
1080 |       "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.0.2.tgz",
1081 |       "integrity": "sha512-vjL3+w0oulAVZ0hBHnxa/Nm5TAurf9YLQJDhqRZyqb+VKGOB6LU8t9H1Nr5CIo16vh9XfJTOoHwU0B71S557gA==",
1082 |       "dev": true,
1083 |       "requires": {
1084 |         "pkg-dir": "^4.2.0",
1085 |         "resolve-cwd": "^3.0.0"
1086 |       }
1087 |     },
1088 |     "inflight": {
1089 |       "version": "1.0.6",
1090 |       "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
1091 |       "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
1092 |       "dev": true,
1093 |       "requires": {
1094 |         "once": "^1.3.0",
1095 |         "wrappy": "1"
1096 |       }
1097 |     },
1098 |     "inherits": {
1099 |       "version": "2.0.3",
1100 |       "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
1101 |       "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=",
1102 |       "dev": true
1103 |     },
1104 |     "interpret": {
1105 |       "version": "2.2.0",
1106 |       "resolved": "https://registry.npmjs.org/interpret/-/interpret-2.2.0.tgz",
1107 |       "integrity": "sha512-Ju0Bz/cEia55xDwUWEa8+olFpCiQoypjnQySseKtmjNrnps3P+xfpUmGr90T7yjlVJmOtybRvPXhKMbHr+fWnw==",
1108 |       "dev": true
1109 |     },
1110 |     "ipaddr.js": {
1111 |       "version": "1.9.1",
1112 |       "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
1113 |       "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==",
1114 |       "dev": true
1115 |     },
1116 |     "is-binary-path": {
1117 |       "version": "2.1.0",
1118 |       "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
1119 |       "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
1120 |       "dev": true,
1121 |       "requires": {
1122 |         "binary-extensions": "^2.0.0"
1123 |       }
1124 |     },
1125 |     "is-core-module": {
1126 |       "version": "2.4.0",
1127 |       "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.4.0.tgz",
1128 |       "integrity": "sha512-6A2fkfq1rfeQZjxrZJGerpLCTHRNEBiSgnu0+obeJpEPZRUooHgsizvzv0ZjJwOz3iWIHdJtVWJ/tmPr3D21/A==",
1129 |       "dev": true,
1130 |       "requires": {
1131 |         "has": "^1.0.3"
1132 |       }
1133 |     },
1134 |     "is-extglob": {
1135 |       "version": "2.1.1",
1136 |       "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
1137 |       "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=",
1138 |       "dev": true
1139 |     },
1140 |     "is-fullwidth-code-point": {
1141 |       "version": "2.0.0",
1142 |       "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
1143 |       "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=",
1144 |       "dev": true
1145 |     },
1146 |     "is-glob": {
1147 |       "version": "4.0.1",
1148 |       "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz",
1149 |       "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==",
1150 |       "dev": true,
1151 |       "requires": {
1152 |         "is-extglob": "^2.1.1"
1153 |       }
1154 |     },
1155 |     "is-number": {
1156 |       "version": "7.0.0",
1157 |       "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
1158 |       "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
1159 |       "dev": true
1160 |     },
1161 |     "is-plain-obj": {
1162 |       "version": "2.1.0",
1163 |       "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-2.1.0.tgz",
1164 |       "integrity": "sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==",
1165 |       "dev": true
1166 |     },
1167 |     "is-plain-object": {
1168 |       "version": "2.0.4",
1169 |       "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz",
1170 |       "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==",
1171 |       "dev": true,
1172 |       "requires": {
1173 |         "isobject": "^3.0.1"
1174 |       }
1175 |     },
1176 |     "is-stream": {
1177 |       "version": "2.0.0",
1178 |       "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz",
1179 |       "integrity": "sha512-XCoy+WlUr7d1+Z8GgSuXmpuUFC9fOhRXglJMx+dwLKTkL44Cjd4W1Z5P+BQZpr+cR93aGP4S/s7Ftw6Nd/kiEw==",
1180 |       "dev": true
1181 |     },
1182 |     "isexe": {
1183 |       "version": "2.0.0",
1184 |       "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
1185 |       "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=",
1186 |       "dev": true
1187 |     },
1188 |     "isobject": {
1189 |       "version": "3.0.1",
1190 |       "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz",
1191 |       "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=",
1192 |       "dev": true
1193 |     },
1194 |     "jest-worker": {
1195 |       "version": "27.0.2",
1196 |       "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.0.2.tgz",
1197 |       "integrity": "sha512-EoBdilOTTyOgmHXtw/cPc+ZrCA0KJMrkXzkrPGNwLmnvvlN1nj7MPrxpT7m+otSv2e1TLaVffzDnE/LB14zJMg==",
1198 |       "dev": true,
1199 |       "requires": {
1200 |         "@types/node": "*",
1201 |         "merge-stream": "^2.0.0",
1202 |         "supports-color": "^8.0.0"
1203 |       },
1204 |       "dependencies": {
1205 |         "has-flag": {
1206 |           "version": "4.0.0",
1207 |           "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
1208 |           "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
1209 |           "dev": true
1210 |         },
1211 |         "supports-color": {
1212 |           "version": "8.1.1",
1213 |           "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz",
1214 |           "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==",
1215 |           "dev": true,
1216 |           "requires": {
1217 |             "has-flag": "^4.0.0"
1218 |           }
1219 |         }
1220 |       }
1221 |     },
1222 |     "js-yaml": {
1223 |       "version": "3.14.0",
1224 |       "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.0.tgz",
1225 |       "integrity": "sha512-/4IbIeHcD9VMHFqDR/gQ7EdZdLimOvW2DdcxFjdyyZ9NsbS+ccrXqVWDtab/lRl5AlUqmpBx8EhPaWR+OtY17A==",
1226 |       "dev": true,
1227 |       "requires": {
1228 |         "argparse": "^1.0.7",
1229 |         "esprima": "^4.0.0"
1230 |       }
1231 |     },
1232 |     "json-parse-better-errors": {
1233 |       "version": "1.0.2",
1234 |       "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz",
1235 |       "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==",
1236 |       "dev": true
1237 |     },
1238 |     "json-schema-traverse": {
1239 |       "version": "0.4.1",
1240 |       "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
1241 |       "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
1242 |       "dev": true
1243 |     },
1244 |     "kind-of": {
1245 |       "version": "6.0.3",
1246 |       "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz",
1247 |       "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==",
1248 |       "dev": true
1249 |     },
1250 |     "loader-runner": {
1251 |       "version": "4.2.0",
1252 |       "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-4.2.0.tgz",
1253 |       "integrity": "sha512-92+huvxMvYlMzMt0iIOukcwYBFpkYJdpl2xsZ7LrlayO7E8SOv+JJUEK17B/dJIHAOLMfh2dZZ/Y18WgmGtYNw==",
1254 |       "dev": true
1255 |     },
1256 |     "locate-path": {
1257 |       "version": "3.0.0",
1258 |       "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz",
1259 |       "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==",
1260 |       "dev": true,
1261 |       "requires": {
1262 |         "p-locate": "^3.0.0",
1263 |         "path-exists": "^3.0.0"
1264 |       }
1265 |     },
1266 |     "log-symbols": {
1267 |       "version": "4.0.0",
1268 |       "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.0.0.tgz",
1269 |       "integrity": "sha512-FN8JBzLx6CzeMrB0tg6pqlGU1wCrXW+ZXGH481kfsBqer0hToTIiHdjH4Mq8xJUbvATujKCvaREGWpGUionraA==",
1270 |       "dev": true,
1271 |       "requires": {
1272 |         "chalk": "^4.0.0"
1273 |       },
1274 |       "dependencies": {
1275 |         "ansi-styles": {
1276 |           "version": "4.3.0",
1277 |           "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
1278 |           "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
1279 |           "dev": true,
1280 |           "requires": {
1281 |             "color-convert": "^2.0.1"
1282 |           }
1283 |         },
1284 |         "chalk": {
1285 |           "version": "4.1.0",
1286 |           "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz",
1287 |           "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==",
1288 |           "dev": true,
1289 |           "requires": {
1290 |             "ansi-styles": "^4.1.0",
1291 |             "supports-color": "^7.1.0"
1292 |           }
1293 |         },
1294 |         "color-convert": {
1295 |           "version": "2.0.1",
1296 |           "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
1297 |           "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
1298 |           "dev": true,
1299 |           "requires": {
1300 |             "color-name": "~1.1.4"
1301 |           }
1302 |         },
1303 |         "color-name": {
1304 |           "version": "1.1.4",
1305 |           "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
1306 |           "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
1307 |           "dev": true
1308 |         }
1309 |       }
1310 |     },
1311 |     "lru-cache": {
1312 |       "version": "6.0.0",
1313 |       "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz",
1314 |       "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==",
1315 |       "dev": true,
1316 |       "requires": {
1317 |         "yallist": "^4.0.0"
1318 |       }
1319 |     },
1320 |     "make-error": {
1321 |       "version": "1.3.6",
1322 |       "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz",
1323 |       "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==",
1324 |       "dev": true
1325 |     },
1326 |     "media-typer": {
1327 |       "version": "0.3.0",
1328 |       "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
1329 |       "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=",
1330 |       "dev": true
1331 |     },
1332 |     "merge-descriptors": {
1333 |       "version": "1.0.1",
1334 |       "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
1335 |       "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=",
1336 |       "dev": true
1337 |     },
1338 |     "merge-stream": {
1339 |       "version": "2.0.0",
1340 |       "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz",
1341 |       "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==",
1342 |       "dev": true
1343 |     },
1344 |     "methods": {
1345 |       "version": "1.1.2",
1346 |       "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
1347 |       "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=",
1348 |       "dev": true
1349 |     },
1350 |     "micromatch": {
1351 |       "version": "4.0.4",
1352 |       "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.4.tgz",
1353 |       "integrity": "sha512-pRmzw/XUcwXGpD9aI9q/0XOwLNygjETJ8y0ao0wdqprrzDa4YnxLcz7fQRZr8voh8V10kGhABbNcHVk5wHgWwg==",
1354 |       "dev": true,
1355 |       "requires": {
1356 |         "braces": "^3.0.1",
1357 |         "picomatch": "^2.2.3"
1358 |       },
1359 |       "dependencies": {
1360 |         "picomatch": {
1361 |           "version": "2.3.0",
1362 |           "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz",
1363 |           "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==",
1364 |           "dev": true
1365 |         }
1366 |       }
1367 |     },
1368 |     "mime": {
1369 |       "version": "1.6.0",
1370 |       "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
1371 |       "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==",
1372 |       "dev": true
1373 |     },
1374 |     "mime-db": {
1375 |       "version": "1.44.0",
1376 |       "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz",
1377 |       "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==",
1378 |       "dev": true
1379 |     },
1380 |     "mime-types": {
1381 |       "version": "2.1.27",
1382 |       "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz",
1383 |       "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==",
1384 |       "dev": true,
1385 |       "requires": {
1386 |         "mime-db": "1.44.0"
1387 |       }
1388 |     },
1389 |     "mimic-fn": {
1390 |       "version": "2.1.0",
1391 |       "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz",
1392 |       "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==",
1393 |       "dev": true
1394 |     },
1395 |     "minimatch": {
1396 |       "version": "3.0.4",
1397 |       "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
1398 |       "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
1399 |       "dev": true,
1400 |       "requires": {
1401 |         "brace-expansion": "^1.1.7"
1402 |       }
1403 |     },
1404 |     "minipass": {
1405 |       "version": "3.1.3",
1406 |       "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.3.tgz",
1407 |       "integrity": "sha512-Mgd2GdMVzY+x3IJ+oHnVM+KG3lA5c8tnabyJKmHSaG2kAGpudxuOf8ToDkhumF7UzME7DecbQE9uOZhNm7PuJg==",
1408 |       "dev": true,
1409 |       "requires": {
1410 |         "yallist": "^4.0.0"
1411 |       },
1412 |       "dependencies": {
1413 |         "yallist": {
1414 |           "version": "4.0.0",
1415 |           "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
1416 |           "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==",
1417 |           "dev": true
1418 |         }
1419 |       }
1420 |     },
1421 |     "mocha": {
1422 |       "version": "8.2.1",
1423 |       "resolved": "https://registry.npmjs.org/mocha/-/mocha-8.2.1.tgz",
1424 |       "integrity": "sha512-cuLBVfyFfFqbNR0uUKbDGXKGk+UDFe6aR4os78XIrMQpZl/nv7JYHcvP5MFIAb374b2zFXsdgEGwmzMtP0Xg8w==",
1425 |       "dev": true,
1426 |       "requires": {
1427 |         "@ungap/promise-all-settled": "1.1.2",
1428 |         "ansi-colors": "4.1.1",
1429 |         "browser-stdout": "1.3.1",
1430 |         "chokidar": "3.4.3",
1431 |         "debug": "4.2.0",
1432 |         "diff": "4.0.2",
1433 |         "escape-string-regexp": "4.0.0",
1434 |         "find-up": "5.0.0",
1435 |         "glob": "7.1.6",
1436 |         "growl": "1.10.5",
1437 |         "he": "1.2.0",
1438 |         "js-yaml": "3.14.0",
1439 |         "log-symbols": "4.0.0",
1440 |         "minimatch": "3.0.4",
1441 |         "ms": "2.1.2",
1442 |         "nanoid": "3.1.12",
1443 |         "serialize-javascript": "5.0.1",
1444 |         "strip-json-comments": "3.1.1",
1445 |         "supports-color": "7.2.0",
1446 |         "which": "2.0.2",
1447 |         "wide-align": "1.1.3",
1448 |         "workerpool": "6.0.2",
1449 |         "yargs": "13.3.2",
1450 |         "yargs-parser": "13.1.2",
1451 |         "yargs-unparser": "2.0.0"
1452 |       },
1453 |       "dependencies": {
1454 |         "debug": {
1455 |           "version": "4.2.0",
1456 |           "resolved": "https://registry.npmjs.org/debug/-/debug-4.2.0.tgz",
1457 |           "integrity": "sha512-IX2ncY78vDTjZMFUdmsvIRFY2Cf4FnD0wRs+nQwJU8Lu99/tPFdb0VybiiMTPe3I6rQmwsqQqRBvxU+bZ/I8sg==",
1458 |           "dev": true,
1459 |           "requires": {
1460 |             "ms": "2.1.2"
1461 |           }
1462 |         },
1463 |         "escape-string-regexp": {
1464 |           "version": "4.0.0",
1465 |           "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz",
1466 |           "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==",
1467 |           "dev": true
1468 |         },
1469 |         "find-up": {
1470 |           "version": "5.0.0",
1471 |           "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
1472 |           "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
1473 |           "dev": true,
1474 |           "requires": {
1475 |             "locate-path": "^6.0.0",
1476 |             "path-exists": "^4.0.0"
1477 |           }
1478 |         },
1479 |         "glob": {
1480 |           "version": "7.1.6",
1481 |           "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz",
1482 |           "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==",
1483 |           "dev": true,
1484 |           "requires": {
1485 |             "fs.realpath": "^1.0.0",
1486 |             "inflight": "^1.0.4",
1487 |             "inherits": "2",
1488 |             "minimatch": "^3.0.4",
1489 |             "once": "^1.3.0",
1490 |             "path-is-absolute": "^1.0.0"
1491 |           }
1492 |         },
1493 |         "locate-path": {
1494 |           "version": "6.0.0",
1495 |           "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
1496 |           "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
1497 |           "dev": true,
1498 |           "requires": {
1499 |             "p-locate": "^5.0.0"
1500 |           }
1501 |         },
1502 |         "ms": {
1503 |           "version": "2.1.2",
1504 |           "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
1505 |           "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
1506 |           "dev": true
1507 |         },
1508 |         "p-limit": {
1509 |           "version": "3.1.0",
1510 |           "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
1511 |           "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
1512 |           "dev": true,
1513 |           "requires": {
1514 |             "yocto-queue": "^0.1.0"
1515 |           }
1516 |         },
1517 |         "p-locate": {
1518 |           "version": "5.0.0",
1519 |           "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
1520 |           "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
1521 |           "dev": true,
1522 |           "requires": {
1523 |             "p-limit": "^3.0.2"
1524 |           }
1525 |         },
1526 |         "path-exists": {
1527 |           "version": "4.0.0",
1528 |           "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
1529 |           "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
1530 |           "dev": true
1531 |         },
1532 |         "serialize-javascript": {
1533 |           "version": "5.0.1",
1534 |           "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-5.0.1.tgz",
1535 |           "integrity": "sha512-SaaNal9imEO737H2c05Og0/8LUXG7EnsZyMa8MzkmuHoELfT6txuj0cMqRj6zfPKnmQ1yasR4PCJc8x+M4JSPA==",
1536 |           "dev": true,
1537 |           "requires": {
1538 |             "randombytes": "^2.1.0"
1539 |           }
1540 |         },
1541 |         "which": {
1542 |           "version": "2.0.2",
1543 |           "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
1544 |           "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
1545 |           "dev": true,
1546 |           "requires": {
1547 |             "isexe": "^2.0.0"
1548 |           }
1549 |         }
1550 |       }
1551 |     },
1552 |     "ms": {
1553 |       "version": "2.0.0",
1554 |       "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1555 |       "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=",
1556 |       "dev": true
1557 |     },
1558 |     "nanoid": {
1559 |       "version": "3.1.12",
1560 |       "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.1.12.tgz",
1561 |       "integrity": "sha512-1qstj9z5+x491jfiC4Nelk+f8XBad7LN20PmyWINJEMRSf3wcAjAWysw1qaA8z6NSKe2sjq1hRSDpBH5paCb6A==",
1562 |       "dev": true
1563 |     },
1564 |     "negotiator": {
1565 |       "version": "0.6.2",
1566 |       "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
1567 |       "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==",
1568 |       "dev": true
1569 |     },
1570 |     "neo-async": {
1571 |       "version": "2.6.2",
1572 |       "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz",
1573 |       "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==",
1574 |       "dev": true
1575 |     },
1576 |     "node-releases": {
1577 |       "version": "1.1.73",
1578 |       "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.73.tgz",
1579 |       "integrity": "sha512-uW7fodD6pyW2FZNZnp/Z3hvWKeEW1Y8R1+1CnErE8cXFXzl5blBOoVB41CvMer6P6Q0S5FXDwcHgFd1Wj0U9zg==",
1580 |       "dev": true
1581 |     },
1582 |     "normalize-path": {
1583 |       "version": "3.0.0",
1584 |       "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
1585 |       "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
1586 |       "dev": true
1587 |     },
1588 |     "npm-run-path": {
1589 |       "version": "4.0.1",
1590 |       "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz",
1591 |       "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==",
1592 |       "dev": true,
1593 |       "requires": {
1594 |         "path-key": "^3.0.0"
1595 |       }
1596 |     },
1597 |     "on-finished": {
1598 |       "version": "2.3.0",
1599 |       "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
1600 |       "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
1601 |       "dev": true,
1602 |       "requires": {
1603 |         "ee-first": "1.1.1"
1604 |       }
1605 |     },
1606 |     "once": {
1607 |       "version": "1.4.0",
1608 |       "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1609 |       "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
1610 |       "dev": true,
1611 |       "requires": {
1612 |         "wrappy": "1"
1613 |       }
1614 |     },
1615 |     "onetime": {
1616 |       "version": "5.1.2",
1617 |       "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz",
1618 |       "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==",
1619 |       "dev": true,
1620 |       "requires": {
1621 |         "mimic-fn": "^2.1.0"
1622 |       }
1623 |     },
1624 |     "p-limit": {
1625 |       "version": "2.3.0",
1626 |       "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz",
1627 |       "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==",
1628 |       "dev": true,
1629 |       "requires": {
1630 |         "p-try": "^2.0.0"
1631 |       }
1632 |     },
1633 |     "p-locate": {
1634 |       "version": "3.0.0",
1635 |       "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz",
1636 |       "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==",
1637 |       "dev": true,
1638 |       "requires": {
1639 |         "p-limit": "^2.0.0"
1640 |       }
1641 |     },
1642 |     "p-try": {
1643 |       "version": "2.2.0",
1644 |       "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz",
1645 |       "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==",
1646 |       "dev": true
1647 |     },
1648 |     "parseurl": {
1649 |       "version": "1.3.3",
1650 |       "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
1651 |       "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==",
1652 |       "dev": true
1653 |     },
1654 |     "path-exists": {
1655 |       "version": "3.0.0",
1656 |       "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz",
1657 |       "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=",
1658 |       "dev": true
1659 |     },
1660 |     "path-is-absolute": {
1661 |       "version": "1.0.1",
1662 |       "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
1663 |       "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=",
1664 |       "dev": true
1665 |     },
1666 |     "path-key": {
1667 |       "version": "3.1.1",
1668 |       "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
1669 |       "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==",
1670 |       "dev": true
1671 |     },
1672 |     "path-parse": {
1673 |       "version": "1.0.7",
1674 |       "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz",
1675 |       "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==",
1676 |       "dev": true
1677 |     },
1678 |     "path-to-regexp": {
1679 |       "version": "0.1.7",
1680 |       "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
1681 |       "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=",
1682 |       "dev": true
1683 |     },
1684 |     "pathval": {
1685 |       "version": "1.1.0",
1686 |       "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.0.tgz",
1687 |       "integrity": "sha1-uULm1L3mUwBe9rcTYd74cn0GReA=",
1688 |       "dev": true
1689 |     },
1690 |     "picomatch": {
1691 |       "version": "2.2.2",
1692 |       "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz",
1693 |       "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==",
1694 |       "dev": true
1695 |     },
1696 |     "pkg-dir": {
1697 |       "version": "4.2.0",
1698 |       "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz",
1699 |       "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==",
1700 |       "dev": true,
1701 |       "requires": {
1702 |         "find-up": "^4.0.0"
1703 |       },
1704 |       "dependencies": {
1705 |         "find-up": {
1706 |           "version": "4.1.0",
1707 |           "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz",
1708 |           "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==",
1709 |           "dev": true,
1710 |           "requires": {
1711 |             "locate-path": "^5.0.0",
1712 |             "path-exists": "^4.0.0"
1713 |           }
1714 |         },
1715 |         "locate-path": {
1716 |           "version": "5.0.0",
1717 |           "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz",
1718 |           "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==",
1719 |           "dev": true,
1720 |           "requires": {
1721 |             "p-locate": "^4.1.0"
1722 |           }
1723 |         },
1724 |         "p-locate": {
1725 |           "version": "4.1.0",
1726 |           "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz",
1727 |           "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==",
1728 |           "dev": true,
1729 |           "requires": {
1730 |             "p-limit": "^2.2.0"
1731 |           }
1732 |         },
1733 |         "path-exists": {
1734 |           "version": "4.0.0",
1735 |           "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
1736 |           "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
1737 |           "dev": true
1738 |         }
1739 |       }
1740 |     },
1741 |     "proxy-addr": {
1742 |       "version": "2.0.6",
1743 |       "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz",
1744 |       "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==",
1745 |       "dev": true,
1746 |       "requires": {
1747 |         "forwarded": "~0.1.2",
1748 |         "ipaddr.js": "1.9.1"
1749 |       }
1750 |     },
1751 |     "punycode": {
1752 |       "version": "2.1.1",
1753 |       "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz",
1754 |       "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==",
1755 |       "dev": true
1756 |     },
1757 |     "qs": {
1758 |       "version": "6.7.0",
1759 |       "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
1760 |       "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==",
1761 |       "dev": true
1762 |     },
1763 |     "randombytes": {
1764 |       "version": "2.1.0",
1765 |       "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz",
1766 |       "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==",
1767 |       "dev": true,
1768 |       "requires": {
1769 |         "safe-buffer": "^5.1.0"
1770 |       }
1771 |     },
1772 |     "range-parser": {
1773 |       "version": "1.2.1",
1774 |       "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
1775 |       "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
1776 |       "dev": true
1777 |     },
1778 |     "raw-body": {
1779 |       "version": "2.4.0",
1780 |       "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
1781 |       "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
1782 |       "dev": true,
1783 |       "requires": {
1784 |         "bytes": "3.1.0",
1785 |         "http-errors": "1.7.2",
1786 |         "iconv-lite": "0.4.24",
1787 |         "unpipe": "1.0.0"
1788 |       }
1789 |     },
1790 |     "readdirp": {
1791 |       "version": "3.5.0",
1792 |       "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.5.0.tgz",
1793 |       "integrity": "sha512-cMhu7c/8rdhkHXWsY+osBhfSy0JikwpHK/5+imo+LpeasTF8ouErHrlYkwT0++njiyuDvc7OFY5T3ukvZ8qmFQ==",
1794 |       "dev": true,
1795 |       "requires": {
1796 |         "picomatch": "^2.2.1"
1797 |       }
1798 |     },
1799 |     "rechoir": {
1800 |       "version": "0.7.0",
1801 |       "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.7.0.tgz",
1802 |       "integrity": "sha512-ADsDEH2bvbjltXEP+hTIAmeFekTFK0V2BTxMkok6qILyAJEXV0AFfoWcAq4yfll5VdIMd/RVXq0lR+wQi5ZU3Q==",
1803 |       "dev": true,
1804 |       "requires": {
1805 |         "resolve": "^1.9.0"
1806 |       }
1807 |     },
1808 |     "require-directory": {
1809 |       "version": "2.1.1",
1810 |       "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
1811 |       "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=",
1812 |       "dev": true
1813 |     },
1814 |     "require-main-filename": {
1815 |       "version": "2.0.0",
1816 |       "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz",
1817 |       "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==",
1818 |       "dev": true
1819 |     },
1820 |     "resolve": {
1821 |       "version": "1.20.0",
1822 |       "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.20.0.tgz",
1823 |       "integrity": "sha512-wENBPt4ySzg4ybFQW2TT1zMQucPK95HSh/nq2CFTZVOGut2+pQvSsgtda4d26YrYcr067wjbmzOG8byDPBX63A==",
1824 |       "dev": true,
1825 |       "requires": {
1826 |         "is-core-module": "^2.2.0",
1827 |         "path-parse": "^1.0.6"
1828 |       }
1829 |     },
1830 |     "resolve-cwd": {
1831 |       "version": "3.0.0",
1832 |       "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz",
1833 |       "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==",
1834 |       "dev": true,
1835 |       "requires": {
1836 |         "resolve-from": "^5.0.0"
1837 |       }
1838 |     },
1839 |     "resolve-from": {
1840 |       "version": "5.0.0",
1841 |       "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz",
1842 |       "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==",
1843 |       "dev": true
1844 |     },
1845 |     "safe-buffer": {
1846 |       "version": "5.1.2",
1847 |       "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
1848 |       "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==",
1849 |       "dev": true
1850 |     },
1851 |     "safer-buffer": {
1852 |       "version": "2.1.2",
1853 |       "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
1854 |       "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
1855 |       "dev": true
1856 |     },
1857 |     "schema-utils": {
1858 |       "version": "3.0.0",
1859 |       "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz",
1860 |       "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==",
1861 |       "dev": true,
1862 |       "requires": {
1863 |         "@types/json-schema": "^7.0.6",
1864 |         "ajv": "^6.12.5",
1865 |         "ajv-keywords": "^3.5.2"
1866 |       }
1867 |     },
1868 |     "semver": {
1869 |       "version": "7.3.5",
1870 |       "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz",
1871 |       "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==",
1872 |       "dev": true,
1873 |       "requires": {
1874 |         "lru-cache": "^6.0.0"
1875 |       }
1876 |     },
1877 |     "send": {
1878 |       "version": "0.17.1",
1879 |       "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
1880 |       "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
1881 |       "dev": true,
1882 |       "requires": {
1883 |         "debug": "2.6.9",
1884 |         "depd": "~1.1.2",
1885 |         "destroy": "~1.0.4",
1886 |         "encodeurl": "~1.0.2",
1887 |         "escape-html": "~1.0.3",
1888 |         "etag": "~1.8.1",
1889 |         "fresh": "0.5.2",
1890 |         "http-errors": "~1.7.2",
1891 |         "mime": "1.6.0",
1892 |         "ms": "2.1.1",
1893 |         "on-finished": "~2.3.0",
1894 |         "range-parser": "~1.2.1",
1895 |         "statuses": "~1.5.0"
1896 |       },
1897 |       "dependencies": {
1898 |         "ms": {
1899 |           "version": "2.1.1",
1900 |           "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
1901 |           "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==",
1902 |           "dev": true
1903 |         }
1904 |       }
1905 |     },
1906 |     "serialize-javascript": {
1907 |       "version": "5.0.1",
1908 |       "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-5.0.1.tgz",
1909 |       "integrity": "sha512-SaaNal9imEO737H2c05Og0/8LUXG7EnsZyMa8MzkmuHoELfT6txuj0cMqRj6zfPKnmQ1yasR4PCJc8x+M4JSPA==",
1910 |       "dev": true,
1911 |       "requires": {
1912 |         "randombytes": "^2.1.0"
1913 |       }
1914 |     },
1915 |     "serve-static": {
1916 |       "version": "1.14.1",
1917 |       "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
1918 |       "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
1919 |       "dev": true,
1920 |       "requires": {
1921 |         "encodeurl": "~1.0.2",
1922 |         "escape-html": "~1.0.3",
1923 |         "parseurl": "~1.3.3",
1924 |         "send": "0.17.1"
1925 |       }
1926 |     },
1927 |     "set-blocking": {
1928 |       "version": "2.0.0",
1929 |       "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz",
1930 |       "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=",
1931 |       "dev": true
1932 |     },
1933 |     "setprototypeof": {
1934 |       "version": "1.1.1",
1935 |       "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
1936 |       "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==",
1937 |       "dev": true
1938 |     },
1939 |     "shallow-clone": {
1940 |       "version": "3.0.1",
1941 |       "resolved": "https://registry.npmjs.org/shallow-clone/-/shallow-clone-3.0.1.tgz",
1942 |       "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==",
1943 |       "dev": true,
1944 |       "requires": {
1945 |         "kind-of": "^6.0.2"
1946 |       }
1947 |     },
1948 |     "shebang-command": {
1949 |       "version": "2.0.0",
1950 |       "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
1951 |       "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
1952 |       "dev": true,
1953 |       "requires": {
1954 |         "shebang-regex": "^3.0.0"
1955 |       }
1956 |     },
1957 |     "shebang-regex": {
1958 |       "version": "3.0.0",
1959 |       "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
1960 |       "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==",
1961 |       "dev": true
1962 |     },
1963 |     "signal-exit": {
1964 |       "version": "3.0.3",
1965 |       "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz",
1966 |       "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==",
1967 |       "dev": true
1968 |     },
1969 |     "source-list-map": {
1970 |       "version": "2.0.1",
1971 |       "resolved": "https://registry.npmjs.org/source-list-map/-/source-list-map-2.0.1.tgz",
1972 |       "integrity": "sha512-qnQ7gVMxGNxsiL4lEuJwe/To8UnK7fAnmbGEEH8RpLouuKbeEm0lhbQVFIrNSuB+G7tVrAlVsZgETT5nljf+Iw==",
1973 |       "dev": true
1974 |     },
1975 |     "source-map": {
1976 |       "version": "0.6.1",
1977 |       "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
1978 |       "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==",
1979 |       "dev": true
1980 |     },
1981 |     "source-map-support": {
1982 |       "version": "0.5.19",
1983 |       "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.19.tgz",
1984 |       "integrity": "sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw==",
1985 |       "dev": true,
1986 |       "requires": {
1987 |         "buffer-from": "^1.0.0",
1988 |         "source-map": "^0.6.0"
1989 |       }
1990 |     },
1991 |     "sprintf-js": {
1992 |       "version": "1.0.3",
1993 |       "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz",
1994 |       "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=",
1995 |       "dev": true
1996 |     },
1997 |     "ssri": {
1998 |       "version": "8.0.1",
1999 |       "resolved": "https://registry.npmjs.org/ssri/-/ssri-8.0.1.tgz",
2000 |       "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==",
2001 |       "dev": true,
2002 |       "requires": {
2003 |         "minipass": "^3.1.1"
2004 |       }
2005 |     },
2006 |     "statuses": {
2007 |       "version": "1.5.0",
2008 |       "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
2009 |       "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=",
2010 |       "dev": true
2011 |     },
2012 |     "string-width": {
2013 |       "version": "2.1.1",
2014 |       "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz",
2015 |       "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==",
2016 |       "dev": true,
2017 |       "requires": {
2018 |         "is-fullwidth-code-point": "^2.0.0",
2019 |         "strip-ansi": "^4.0.0"
2020 |       }
2021 |     },
2022 |     "strip-ansi": {
2023 |       "version": "4.0.0",
2024 |       "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz",
2025 |       "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=",
2026 |       "dev": true,
2027 |       "requires": {
2028 |         "ansi-regex": "^3.0.0"
2029 |       }
2030 |     },
2031 |     "strip-final-newline": {
2032 |       "version": "2.0.0",
2033 |       "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz",
2034 |       "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==",
2035 |       "dev": true
2036 |     },
2037 |     "strip-json-comments": {
2038 |       "version": "3.1.1",
2039 |       "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz",
2040 |       "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==",
2041 |       "dev": true
2042 |     },
2043 |     "supports-color": {
2044 |       "version": "7.2.0",
2045 |       "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
2046 |       "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
2047 |       "dev": true,
2048 |       "requires": {
2049 |         "has-flag": "^4.0.0"
2050 |       },
2051 |       "dependencies": {
2052 |         "has-flag": {
2053 |           "version": "4.0.0",
2054 |           "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
2055 |           "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
2056 |           "dev": true
2057 |         }
2058 |       }
2059 |     },
2060 |     "tapable": {
2061 |       "version": "2.2.0",
2062 |       "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.2.0.tgz",
2063 |       "integrity": "sha512-FBk4IesMV1rBxX2tfiK8RAmogtWn53puLOQlvO8XuwlgxcYbP4mVPS9Ph4aeamSyyVjOl24aYWAuc8U5kCVwMw==",
2064 |       "dev": true
2065 |     },
2066 |     "terser": {
2067 |       "version": "5.7.0",
2068 |       "resolved": "https://registry.npmjs.org/terser/-/terser-5.7.0.tgz",
2069 |       "integrity": "sha512-HP5/9hp2UaZt5fYkuhNBR8YyRcT8juw8+uFbAme53iN9hblvKnLUTKkmwJG6ocWpIKf8UK4DoeWG4ty0J6S6/g==",
2070 |       "dev": true,
2071 |       "requires": {
2072 |         "commander": "^2.20.0",
2073 |         "source-map": "~0.7.2",
2074 |         "source-map-support": "~0.5.19"
2075 |       },
2076 |       "dependencies": {
2077 |         "source-map": {
2078 |           "version": "0.7.3",
2079 |           "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.3.tgz",
2080 |           "integrity": "sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ==",
2081 |           "dev": true
2082 |         }
2083 |       }
2084 |     },
2085 |     "terser-webpack-plugin": {
2086 |       "version": "5.1.3",
2087 |       "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-5.1.3.tgz",
2088 |       "integrity": "sha512-cxGbMqr6+A2hrIB5ehFIF+F/iST5ZOxvOmy9zih9ySbP1C2oEWQSOUS+2SNBTjzx5xLKO4xnod9eywdfq1Nb9A==",
2089 |       "dev": true,
2090 |       "requires": {
2091 |         "jest-worker": "^27.0.2",
2092 |         "p-limit": "^3.1.0",
2093 |         "schema-utils": "^3.0.0",
2094 |         "serialize-javascript": "^5.0.1",
2095 |         "source-map": "^0.6.1",
2096 |         "terser": "^5.7.0"
2097 |       },
2098 |       "dependencies": {
2099 |         "p-limit": {
2100 |           "version": "3.1.0",
2101 |           "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
2102 |           "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
2103 |           "dev": true,
2104 |           "requires": {
2105 |             "yocto-queue": "^0.1.0"
2106 |           }
2107 |         }
2108 |       }
2109 |     },
2110 |     "to-regex-range": {
2111 |       "version": "5.0.1",
2112 |       "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
2113 |       "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
2114 |       "dev": true,
2115 |       "requires": {
2116 |         "is-number": "^7.0.0"
2117 |       }
2118 |     },
2119 |     "toidentifier": {
2120 |       "version": "1.0.0",
2121 |       "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
2122 |       "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==",
2123 |       "dev": true
2124 |     },
2125 |     "ts-loader": {
2126 |       "version": "9.2.3",
2127 |       "resolved": "https://registry.npmjs.org/ts-loader/-/ts-loader-9.2.3.tgz",
2128 |       "integrity": "sha512-sEyWiU3JMHBL55CIeC4iqJQadI0U70A5af0kvgbNLHVNz2ACztQg0j/9x10bjjIht8WfFYLKfn4L6tkZ+pu+8Q==",
2129 |       "dev": true,
2130 |       "requires": {
2131 |         "chalk": "^4.1.0",
2132 |         "enhanced-resolve": "^5.0.0",
2133 |         "micromatch": "^4.0.0",
2134 |         "semver": "^7.3.4"
2135 |       }
2136 |     },
2137 |     "ts-node": {
2138 |       "version": "9.1.1",
2139 |       "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-9.1.1.tgz",
2140 |       "integrity": "sha512-hPlt7ZACERQGf03M253ytLY3dHbGNGrAq9qIHWUY9XHYl1z7wYngSr3OQ5xmui8o2AaxsONxIzjafLUiWBo1Fg==",
2141 |       "dev": true,
2142 |       "requires": {
2143 |         "arg": "^4.1.0",
2144 |         "create-require": "^1.1.0",
2145 |         "diff": "^4.0.1",
2146 |         "make-error": "^1.1.1",
2147 |         "source-map-support": "^0.5.17",
2148 |         "yn": "3.1.1"
2149 |       }
2150 |     },
2151 |     "type-detect": {
2152 |       "version": "4.0.8",
2153 |       "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz",
2154 |       "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==",
2155 |       "dev": true
2156 |     },
2157 |     "type-is": {
2158 |       "version": "1.6.18",
2159 |       "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
2160 |       "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
2161 |       "dev": true,
2162 |       "requires": {
2163 |         "media-typer": "0.3.0",
2164 |         "mime-types": "~2.1.24"
2165 |       }
2166 |     },
2167 |     "typescript": {
2168 |       "version": "3.9.7",
2169 |       "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.7.tgz",
2170 |       "integrity": "sha512-BLbiRkiBzAwsjut4x/dsibSTB6yWpwT5qWmC2OfuCg3GgVQCSgMs4vEctYPhsaGtd0AeuuHMkjZ2h2WG8MSzRw==",
2171 |       "dev": true
2172 |     },
2173 |     "unpipe": {
2174 |       "version": "1.0.0",
2175 |       "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
2176 |       "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=",
2177 |       "dev": true
2178 |     },
2179 |     "uri-js": {
2180 |       "version": "4.4.1",
2181 |       "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz",
2182 |       "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==",
2183 |       "dev": true,
2184 |       "requires": {
2185 |         "punycode": "^2.1.0"
2186 |       }
2187 |     },
2188 |     "utils-merge": {
2189 |       "version": "1.0.1",
2190 |       "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
2191 |       "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=",
2192 |       "dev": true
2193 |     },
2194 |     "v8-compile-cache": {
2195 |       "version": "2.3.0",
2196 |       "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz",
2197 |       "integrity": "sha512-l8lCEmLcLYZh4nbunNZvQCJc5pv7+RCwa8q/LdUx8u7lsWvPDKmpodJAJNwkAhJC//dFY48KuIEmjtd4RViDrA==",
2198 |       "dev": true
2199 |     },
2200 |     "vary": {
2201 |       "version": "1.1.2",
2202 |       "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
2203 |       "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=",
2204 |       "dev": true
2205 |     },
2206 |     "watchpack": {
2207 |       "version": "2.2.0",
2208 |       "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-2.2.0.tgz",
2209 |       "integrity": "sha512-up4YAn/XHgZHIxFBVCdlMiWDj6WaLKpwVeGQk2I5thdYxF/KmF0aaz6TfJZ/hfl1h/XlcDr7k1KH7ThDagpFaA==",
2210 |       "dev": true,
2211 |       "requires": {
2212 |         "glob-to-regexp": "^0.4.1",
2213 |         "graceful-fs": "^4.1.2"
2214 |       }
2215 |     },
2216 |     "webpack": {
2217 |       "version": "5.38.1",
2218 |       "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.38.1.tgz",
2219 |       "integrity": "sha512-OqRmYD1OJbHZph6RUMD93GcCZy4Z4wC0ele4FXyYF0J6AxO1vOSuIlU1hkS/lDlR9CDYBz64MZRmdbdnFFoT2g==",
2220 |       "dev": true,
2221 |       "requires": {
2222 |         "@types/eslint-scope": "^3.7.0",
2223 |         "@types/estree": "^0.0.47",
2224 |         "@webassemblyjs/ast": "1.11.0",
2225 |         "@webassemblyjs/wasm-edit": "1.11.0",
2226 |         "@webassemblyjs/wasm-parser": "1.11.0",
2227 |         "acorn": "^8.2.1",
2228 |         "browserslist": "^4.14.5",
2229 |         "chrome-trace-event": "^1.0.2",
2230 |         "enhanced-resolve": "^5.8.0",
2231 |         "es-module-lexer": "^0.4.0",
2232 |         "eslint-scope": "5.1.1",
2233 |         "events": "^3.2.0",
2234 |         "glob-to-regexp": "^0.4.1",
2235 |         "graceful-fs": "^4.2.4",
2236 |         "json-parse-better-errors": "^1.0.2",
2237 |         "loader-runner": "^4.2.0",
2238 |         "mime-types": "^2.1.27",
2239 |         "neo-async": "^2.6.2",
2240 |         "schema-utils": "^3.0.0",
2241 |         "tapable": "^2.1.1",
2242 |         "terser-webpack-plugin": "^5.1.1",
2243 |         "watchpack": "^2.2.0",
2244 |         "webpack-sources": "^2.3.0"
2245 |       },
2246 |       "dependencies": {
2247 |         "enhanced-resolve": {
2248 |           "version": "5.8.2",
2249 |           "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.8.2.tgz",
2250 |           "integrity": "sha512-F27oB3WuHDzvR2DOGNTaYy0D5o0cnrv8TeI482VM4kYgQd/FT9lUQwuNsJ0oOHtBUq7eiW5ytqzp7nBFknL+GA==",
2251 |           "dev": true,
2252 |           "requires": {
2253 |             "graceful-fs": "^4.2.4",
2254 |             "tapable": "^2.2.0"
2255 |           }
2256 |         },
2257 |         "tapable": {
2258 |           "version": "2.2.0",
2259 |           "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.2.0.tgz",
2260 |           "integrity": "sha512-FBk4IesMV1rBxX2tfiK8RAmogtWn53puLOQlvO8XuwlgxcYbP4mVPS9Ph4aeamSyyVjOl24aYWAuc8U5kCVwMw==",
2261 |           "dev": true
2262 |         }
2263 |       }
2264 |     },
2265 |     "webpack-cli": {
2266 |       "version": "4.7.2",
2267 |       "resolved": "https://registry.npmjs.org/webpack-cli/-/webpack-cli-4.7.2.tgz",
2268 |       "integrity": "sha512-mEoLmnmOIZQNiRl0ebnjzQ74Hk0iKS5SiEEnpq3dRezoyR3yPaeQZCMCe+db4524pj1Pd5ghZXjT41KLzIhSLw==",
2269 |       "dev": true,
2270 |       "requires": {
2271 |         "@discoveryjs/json-ext": "^0.5.0",
2272 |         "@webpack-cli/configtest": "^1.0.4",
2273 |         "@webpack-cli/info": "^1.3.0",
2274 |         "@webpack-cli/serve": "^1.5.1",
2275 |         "colorette": "^1.2.1",
2276 |         "commander": "^7.0.0",
2277 |         "execa": "^5.0.0",
2278 |         "fastest-levenshtein": "^1.0.12",
2279 |         "import-local": "^3.0.2",
2280 |         "interpret": "^2.2.0",
2281 |         "rechoir": "^0.7.0",
2282 |         "v8-compile-cache": "^2.2.0",
2283 |         "webpack-merge": "^5.7.3"
2284 |       },
2285 |       "dependencies": {
2286 |         "commander": {
2287 |           "version": "7.2.0",
2288 |           "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz",
2289 |           "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==",
2290 |           "dev": true
2291 |         }
2292 |       }
2293 |     },
2294 |     "webpack-merge": {
2295 |       "version": "5.8.0",
2296 |       "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-5.8.0.tgz",
2297 |       "integrity": "sha512-/SaI7xY0831XwP6kzuwhKWVKDP9t1QY1h65lAFLbZqMPIuYcD9QAW4u9STIbU9kaJbPBB/geU/gLr1wDjOhQ+Q==",
2298 |       "dev": true,
2299 |       "requires": {
2300 |         "clone-deep": "^4.0.1",
2301 |         "wildcard": "^2.0.0"
2302 |       }
2303 |     },
2304 |     "webpack-sources": {
2305 |       "version": "2.3.0",
2306 |       "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-2.3.0.tgz",
2307 |       "integrity": "sha512-WyOdtwSvOML1kbgtXbTDnEW0jkJ7hZr/bDByIwszhWd/4XX1A3XMkrbFMsuH4+/MfLlZCUzlAdg4r7jaGKEIgQ==",
2308 |       "dev": true,
2309 |       "requires": {
2310 |         "source-list-map": "^2.0.1",
2311 |         "source-map": "^0.6.1"
2312 |       }
2313 |     },
2314 |     "which": {
2315 |       "version": "2.0.2",
2316 |       "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
2317 |       "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
2318 |       "dev": true,
2319 |       "requires": {
2320 |         "isexe": "^2.0.0"
2321 |       }
2322 |     },
2323 |     "which-module": {
2324 |       "version": "2.0.0",
2325 |       "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz",
2326 |       "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=",
2327 |       "dev": true
2328 |     },
2329 |     "wide-align": {
2330 |       "version": "1.1.3",
2331 |       "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz",
2332 |       "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==",
2333 |       "dev": true,
2334 |       "requires": {
2335 |         "string-width": "^1.0.2 || 2"
2336 |       }
2337 |     },
2338 |     "wildcard": {
2339 |       "version": "2.0.0",
2340 |       "resolved": "https://registry.npmjs.org/wildcard/-/wildcard-2.0.0.tgz",
2341 |       "integrity": "sha512-JcKqAHLPxcdb9KM49dufGXn2x3ssnfjbcaQdLlfZsL9rH9wgDQjUtDxbo8NE0F6SFvydeu1VhZe7hZuHsB2/pw==",
2342 |       "dev": true
2343 |     },
2344 |     "workerpool": {
2345 |       "version": "6.0.2",
2346 |       "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.0.2.tgz",
2347 |       "integrity": "sha512-DSNyvOpFKrNusaaUwk+ej6cBj1bmhLcBfj80elGk+ZIo5JSkq+unB1dLKEOcNfJDZgjGICfhQ0Q5TbP0PvF4+Q==",
2348 |       "dev": true
2349 |     },
2350 |     "wrap-ansi": {
2351 |       "version": "5.1.0",
2352 |       "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz",
2353 |       "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==",
2354 |       "dev": true,
2355 |       "requires": {
2356 |         "ansi-styles": "^3.2.0",
2357 |         "string-width": "^3.0.0",
2358 |         "strip-ansi": "^5.0.0"
2359 |       },
2360 |       "dependencies": {
2361 |         "ansi-regex": {
2362 |           "version": "4.1.0",
2363 |           "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
2364 |           "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
2365 |           "dev": true
2366 |         },
2367 |         "string-width": {
2368 |           "version": "3.1.0",
2369 |           "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
2370 |           "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
2371 |           "dev": true,
2372 |           "requires": {
2373 |             "emoji-regex": "^7.0.1",
2374 |             "is-fullwidth-code-point": "^2.0.0",
2375 |             "strip-ansi": "^5.1.0"
2376 |           }
2377 |         },
2378 |         "strip-ansi": {
2379 |           "version": "5.2.0",
2380 |           "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
2381 |           "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
2382 |           "dev": true,
2383 |           "requires": {
2384 |             "ansi-regex": "^4.1.0"
2385 |           }
2386 |         }
2387 |       }
2388 |     },
2389 |     "wrappy": {
2390 |       "version": "1.0.2",
2391 |       "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
2392 |       "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=",
2393 |       "dev": true
2394 |     },
2395 |     "y18n": {
2396 |       "version": "4.0.3",
2397 |       "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.3.tgz",
2398 |       "integrity": "sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ==",
2399 |       "dev": true
2400 |     },
2401 |     "yallist": {
2402 |       "version": "4.0.0",
2403 |       "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
2404 |       "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==",
2405 |       "dev": true
2406 |     },
2407 |     "yargs": {
2408 |       "version": "13.3.2",
2409 |       "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz",
2410 |       "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==",
2411 |       "dev": true,
2412 |       "requires": {
2413 |         "cliui": "^5.0.0",
2414 |         "find-up": "^3.0.0",
2415 |         "get-caller-file": "^2.0.1",
2416 |         "require-directory": "^2.1.1",
2417 |         "require-main-filename": "^2.0.0",
2418 |         "set-blocking": "^2.0.0",
2419 |         "string-width": "^3.0.0",
2420 |         "which-module": "^2.0.0",
2421 |         "y18n": "^4.0.0",
2422 |         "yargs-parser": "^13.1.2"
2423 |       },
2424 |       "dependencies": {
2425 |         "ansi-regex": {
2426 |           "version": "4.1.0",
2427 |           "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
2428 |           "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
2429 |           "dev": true
2430 |         },
2431 |         "string-width": {
2432 |           "version": "3.1.0",
2433 |           "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
2434 |           "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
2435 |           "dev": true,
2436 |           "requires": {
2437 |             "emoji-regex": "^7.0.1",
2438 |             "is-fullwidth-code-point": "^2.0.0",
2439 |             "strip-ansi": "^5.1.0"
2440 |           }
2441 |         },
2442 |         "strip-ansi": {
2443 |           "version": "5.2.0",
2444 |           "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
2445 |           "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
2446 |           "dev": true,
2447 |           "requires": {
2448 |             "ansi-regex": "^4.1.0"
2449 |           }
2450 |         }
2451 |       }
2452 |     },
2453 |     "yargs-parser": {
2454 |       "version": "13.1.2",
2455 |       "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz",
2456 |       "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==",
2457 |       "dev": true,
2458 |       "requires": {
2459 |         "camelcase": "^5.0.0",
2460 |         "decamelize": "^1.2.0"
2461 |       }
2462 |     },
2463 |     "yargs-unparser": {
2464 |       "version": "2.0.0",
2465 |       "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-2.0.0.tgz",
2466 |       "integrity": "sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==",
2467 |       "dev": true,
2468 |       "requires": {
2469 |         "camelcase": "^6.0.0",
2470 |         "decamelize": "^4.0.0",
2471 |         "flat": "^5.0.2",
2472 |         "is-plain-obj": "^2.1.0"
2473 |       },
2474 |       "dependencies": {
2475 |         "camelcase": {
2476 |           "version": "6.2.0",
2477 |           "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.2.0.tgz",
2478 |           "integrity": "sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg==",
2479 |           "dev": true
2480 |         },
2481 |         "decamelize": {
2482 |           "version": "4.0.0",
2483 |           "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-4.0.0.tgz",
2484 |           "integrity": "sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==",
2485 |           "dev": true
2486 |         }
2487 |       }
2488 |     },
2489 |     "yn": {
2490 |       "version": "3.1.1",
2491 |       "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz",
2492 |       "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==",
2493 |       "dev": true
2494 |     },
2495 |     "yocto-queue": {
2496 |       "version": "0.1.0",
2497 |       "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
2498 |       "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==",
2499 |       "dev": true
2500 |     }
2501 |   }
2502 | }
2503 | 
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
 1 | {
 2 |   "name": "@deislabs/wasm-linker-js",
 3 |   "version": "0.2.1",
 4 |   "description": "A simple WebAssembly linker",
 5 |   "keywords": [
 6 |     "wasm",
 7 |     "webassembly"
 8 |   ],
 9 |   "author": {
10 |     "name": "The DeisLabs team at Microsoft"
11 |   },
12 |   "homepage": "https://github.com/deislabs/wasm-linker-js",
13 |   "repository": {
14 |     "type": "git",
15 |     "url": "https://github.com/deislabs/wasm-linker-js.git"
16 |   },
17 |   "bugs": {
18 |     "url": "https://github.com/deislabs/wasm-linker-js/issues"
19 |   },
20 |   "main": "dist/src/index.js",
21 |   "types": "dist/src/index.d.ts",
22 |   "directories": {
23 |     "lib": "lib"
24 |   },
25 |   "files": [
26 |     "dist/src/*",
27 |     "dist/wasm-linker.js"
28 |   ],
29 |   "dependencies": {
30 |     "asyncify-wasm": "^1.1.2"
31 |   },
32 |   "devDependencies": {
33 |     "@types/chai": "^4.0.1",
34 |     "@types/mocha": "^8.2.0",
35 |     "@types/node": "^14.14.14",
36 |     "@types/node-fetch": "^2.5.7",
37 |     "@types/uuid": "^8.3.0",
38 |     "binaryen": "^98.0.0",
39 |     "chai": "^4.1.0",
40 |     "express": "4.17.1",
41 |     "mocha": "^8.2.1",
42 |     "ssri": "^8.0.1",
43 |     "ts-loader": "^9.2.3",
44 |     "ts-node": "^9.1.1",
45 |     "typescript": "^3.8.3",
46 |     "webpack": "^5.38.1",
47 |     "webpack-cli": "^4.7.2"
48 |   },
49 |   "scripts": {
50 |     "build": "tsc && webpack",
51 |     "test": "mocha --timeout 90000 --require ts-node/register --project tsconfig.json --recursive ./tests/**/*.ts",
52 |     "examples": "npm run build && cd examples && node node-example.js && node --experimental-wasi-unstable-preview1 --experimental-wasm-bigint wasi-example.js"
53 |   },
54 |   "license": "MIT"
55 | }
56 | 
--------------------------------------------------------------------------------
/readme.md:
--------------------------------------------------------------------------------
  1 | # wasm-linker-js
  2 | 
  3 | ### A simple WebAssembly Linker in JavaScript
  4 | 
  5 | ![actions badge][actions-badge] [![NPM version][npm-image]][npm]
  6 | 
  7 | This is _an experimental_ JavaScript library that helps instantiating
  8 | WebAssembly modules with imports by providing functionality to link JavaScript
  9 | objects (functions, memories, globals) as imports, as well as automatically
 10 | perform name based resolution for linking entire modules.
 11 | 
 12 | The API loosely follows the [Wasmtime][wasmtime] linker, (see the [linker
 13 | documentation][wasmtime-linker]), and it exposes asynchronous import
 14 | functionality enabled by [Binaryen][binaryen] and [Asyncify][asyncify].
 15 | 
 16 | ### Using the Linker
 17 | 
 18 | > For more examples of using the Linker in both TypeScript and JavaScript, check
 19 | > the [linker tests][linker-tests] and the [Node.js examples][node-examples].
 20 | 
 21 | First, add the package to your project:
 22 | 
 23 | ```plaintext
 24 | $ npm install @deislabs/wasm-linker-js
 25 | ```
 26 | 
 27 | > Note that in order to run the examples shown here, `binaryen` is also required
 28 | > (`npm install binaryen`), in order to show the text format of the WebAssembly
 29 | > modules. In real world scenarios that is not necessary, and the modules can be
 30 | > compiled from their binary representation without additional dependencies.
 31 | 
 32 | #### Defining a single import
 33 | 
 34 | Assuming we are trying to instantiate the module represented in its text format
 35 | below (transformed to its binary representation using [Binaryen][binaryen]), we
 36 | can satisfy its import using the `define` method available on the linker:
 37 | 
 38 | ```js
 39 | const { Linker } = require("@deislabs/wasm-linker-js");
 40 | const { parseText } = require("binaryen");
 41 | 
 42 | const usingAdd = `
 43 | (module
 44 |     (import "calculator" "add" (func $calc_add (param i32 i32) (result i32)))
 45 |   
 46 |     (memory 1 1)
 47 |     (export "memory" (memory 0))
 48 |     (export "add" (func $add))
 49 | 
 50 |     (func $add (param i32) (param i32) (result i32)
 51 |         (return
 52 |             (call $calc_add
 53 |                 (local.get 0)
 54 |                 (local.get 1)
 55 |             )
 56 |         )
 57 |     )
 58 | )
 59 | `;
 60 | 
 61 | (async () => {
 62 |   var linker = new Linker();
 63 | 
 64 |   // The "usingAdd" module imports calculator.add.
 65 |   // We define it,  provide a JS implementation, then
 66 |   // instantiate it.
 67 |   linker.define("calculator", "add", (a, b) => a + b);
 68 |   var calc = await linker.instantiate(parseText(usingAdd).emitBinary());
 69 | 
 70 |   var result = calc.instance.exports.add(1, 2);
 71 |   console.log(result);
 72 | })();
 73 | ```
 74 | 
 75 | #### Linking an entire module
 76 | 
 77 | If we have a compiled module that exports items (defined below in its text
 78 | format and contained in the `add` constant) that our initial module needs to
 79 | import, we can add it to the linker, then continue instantiating our module
 80 | (defined above in its text format and contained in the `usingAdd` constant):
 81 | 
 82 | ```js
 83 | const { Linker } = require("@deislabs/wasm-linker-js");
 84 | const { parseText } = require("binaryen");
 85 | 
 86 | const add = `
 87 | (module
 88 |   (memory 1 1)
 89 |   (export "memory" (memory 0))
 90 |   (export "add" (func $add))
 91 |   
 92 |   (func $add (param i32) (param i32) (result i32)
 93 |       (return
 94 |           (i32.add
 95 |               (local.get 0)
 96 |               (local.get 1)
 97 |           )
 98 |       )
 99 |   )
100 | )
101 | `;
102 | 
103 | (async () => {
104 |   var linker = new Linker();
105 | 
106 |   // The "usingAdd" module above imports calculator.add.
107 |   // We link a module that exports the functionality
108 |   // required, then instantiate the module that uses it.
109 |   await linker.module(
110 |     "calculator",
111 |     new WebAssembly.Module(parseText(add).emitBinary())
112 |   );
113 |   var calc = await linker.instantiate(parseText(usingAdd).emitBinary());
114 |   var result = calc.instance.exports.add(1, 2);
115 |   console.log(result);
116 | })();
117 | ```
118 | 
119 | #### Defining asynchronous imports
120 | 
121 | The current WebAssembly MVP does not have a way of waiting for the execution of
122 | asynchronous imports (see [this issue][async-wasm-issue]). To enable this
123 | functionality, [Binaryen][binaryen] has a pass that [transforms a Wasm module
124 | and allows it to pause and resume by unwiding and rewinding the call
125 | stack][asyncify-blog]. When enabled, this library can use the [JavaScript
126 | wrapper of Asyncify][asyncify] and define asynchronous import functions for
127 | WebAssembly modules (note that the Asyncify pass must have been applied to the
128 | module before instantiating using the linker):
129 | 
130 | ```js
131 | const { Linker } = require("@deislabs/wasm-linker-js");
132 | const { parseText } = require("binaryen");
133 | 
134 | (async () => {
135 |   var useAsyncify = true;
136 |   var linker = new Linker(useAsyncify);
137 | 
138 |   // Notice how we define an asynchronous import, which
139 |   // will wait for 1.5s before returning the result.
140 |   var sleep = function (ms) {
141 |     return new Promise((resolve) => {
142 |       setTimeout(resolve, ms);
143 |     });
144 |   };
145 |   linker.define("calculator", "add", async (a, b) => {
146 |     await sleep(1500);
147 |     return a + b;
148 |   });
149 | 
150 |   let bytes = parseText(usingAdd);
151 | 
152 |   // we perform the asyncify compiler pass from Binaryen
153 |   bytes.runPasses(["asyncify"]);
154 |   var calc = await linker.instantiate(bytes.emitBinary());
155 | 
156 |   var result = await calc.instance.exports.add(1, 2);
157 |   console.log(result);
158 | })();
159 | ```
160 | 
161 | #### Using the streaming APIs in the browser
162 | 
163 | For browsers that support the WebAssembly streaming APIs, the linker exposes two
164 | methods that can be used to efficiently instantiate modules from a streamed
165 | source: `moduleStreaming`, which instantiates a module and adds its exports to
166 | the linker's cache, and `instantiateStreaming`, which instantiates a module and
167 | returns its the `WebAssemblyInstantiatedSource`:
168 | 
169 | ```js
170 | (async () => {
171 |   var linker = new Linker();
172 |   await linker.moduleStreaming("calculator", fetch("calculator.wasm"));
173 |   var mod = await linker.instantiateStreaming(fetch("using_calculator.wasm"));
174 |   console.log(mod.instance.exports.multiply(3, 4));
175 | }
176 | ```
177 | 
178 | See [the documentation for the browser streaming APIs][mdn-streaming] for more
179 | information about instantiating modules from streamed sources.
180 | 
181 | The linker also allows adding an already instantiated module, through the
182 | `instance` method, and aliasing a module under a new name, through the `alias`
183 | method. Most public methods defined on the Linker have a correspondent in the
184 | [Wasmtime][wasmtime] Linker, and we try to keep the APIs similar.
185 | 
186 | ### Implementation notes and known issues
187 | 
188 | - When defining multiple import items with the same name, the last one takes
189 |   precedence (the existing items are replaced). This behavior could change in
190 |   the future to add a configurable property defining whether import shadowing
191 |   should be allowed.
192 | - When instantiating a linker with Asyncify enabled, _all_ modules linked and
193 |   instantiated with the linker will be instantiated using Asyncify's JavaScript
194 |   wrapper. This behavior could change in the future to allow a per-instance (and
195 |   by extension per module linked) setting for Asyncify. (this can be avoided
196 |   through instantiating a module separately and adding it to the linker using
197 |   the `instance` method).
198 | - There is a [browser example in the `examples/` directory][browser-demo] in
199 |   this repository. While functional, the implementation is far from ideal - the
200 |   WebPack configuration for generating a browser-compatible library is not
201 |   optimal (this should be changed to use ECMAScript modules).
202 | - **This library is experimental, and the API is not stable**. We welcome
203 |   feedback on both the public API and the implementation of this library.
204 | 
205 | ### Contributing
206 | 
207 | This project welcomes contributions through the GitHub pull request process.
208 | Prerequisites to building the project:
209 | 
210 | - Node.js
211 | - `npm`
212 | 
213 | To iterate on the project locally:
214 | 
215 | ```plaintext
216 | $ npm run build
217 | $ npm test
218 | $ npm run examples
219 | ```
220 | 
221 | ### Code of Conduct
222 | 
223 | This project has adopted the
224 | [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/).
225 | 
226 | For more information see the
227 | [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or
228 | contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any
229 | additional questions or comments.
230 | 
231 | [wasmtime]: https://github.com/bytecodealliance/wasmtime
232 | [wasmtime-linker]: https://docs.rs/wasmtime/0.21.0/wasmtime/
233 | [binaryen]: https://github.com/WebAssembly/binaryen
234 | [asyncify]: https://github.com/GoogleChromeLabs/asyncify
235 | [async-wasm-issue]: https://github.com/WebAssembly/design/issues/720
236 | [asyncify-blog]: https://kripken.github.io/blog/wasm/2019/07/16/asyncify.html
237 | [browser-demo]: examples/index.html
238 | [node-examples]: examples/node-example.js
239 | [linker-tests]: tests/linker.ts
240 | [mdn-streaming]:
241 |   https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WebAssembly/instantiateStreaming
242 | [npm-image]: https://badge.fury.io/js/%40deislabs%2Fwasm-linker-js.svg
243 | [npm]: https://www.npmjs.com/package/@deislabs/wasm-linker-js
244 | [actions-badge]:
245 |   https://github.com/deislabs/wasm-linker-js/workflows/Build%20and%20Test/badge.svg
246 | 
--------------------------------------------------------------------------------
/serve.js:
--------------------------------------------------------------------------------
 1 | // A simple web server that serves Wasm modules
 2 | // using the appropriate MIME type.
 3 | //
 4 | // Used to run the browser example, which fetches
 5 | // WebAssembly modules and instantiates them.
 6 | 
 7 | var express = require("express");
 8 | var app = express();
 9 | express.static.mime.types["wasm"] = "application/wasm";
10 | app.use(express.static(__dirname + "/"));
11 | app.listen(8080);
12 | 
--------------------------------------------------------------------------------
/src/index.ts:
--------------------------------------------------------------------------------
1 | export { Linker } from "./linker";
2 | export { Store } from "./store";
3 | 
--------------------------------------------------------------------------------
/src/linker.ts:
--------------------------------------------------------------------------------
  1 | import * as asyncify from "asyncify-wasm";
  2 | import { Store } from "./store";
  3 | 
  4 | /**
  5 |  * A JavaScript object that can be used to link WebAssembly instances.
  6 |  *
  7 |  * This is a helper class to be used when instantiating WebAssembly
  8 |  * modules with imports. It can be used to define individual imports
  9 |  * (like functions, or memories), or link entire modules when
 10 |  * instantiating modules that depend on other modules.
 11 |  */
 12 | export class Linker {
 13 |   public store: Store;
 14 |   public useAsyncify: boolean;
 15 | 
 16 |   constructor(useAsyncify: boolean = false, store: Store = new Store()) {
 17 |     this.useAsyncify = useAsyncify;
 18 |     this.store = store;
 19 |   }
 20 | 
 21 |   /**
 22 |    * Adds a new WebAssembly.ImportValue to the imports cache.
 23 |    *
 24 |    * Example:
 25 |    * ```js
 26 | const { Linker } = require("@deislabs/wasm-linker-js");
 27 | const { parseText } = require("binaryen");
 28 | const assert = require("assert");
 29 | 
 30 | (async () => {
 31 |   const usingAdd = `
 32 |     (module
 33 |         (import "calculator" "add" (func $calc_add (param i32 i32) (result i32))
 34 |         (export "add" (func $add))
 35 |     
 36 |         (func $add (param i32) (param i32) (result i32)
 37 |             (return
 38 |                 (call $calc_add
 39 |                     (local.get 0)
 40 |                     (local.get 1)
 41 |                 )
 42 |             )
 43 |         )
 44 |     )
 45 |     `;
 46 | 
 47 |   var linker = new Linker();
 48 | 
 49 |   // the "usingAdd" module above imports calculator.add
 50 |   // we define it and provide a JS implementation, then
 51 |   // instantiate it.
 52 |   linker.define("calculator", "add", (a, b) => a + b);
 53 |   var calc = await linker.instantiate(
 54 |     parseText(usingAdd).emitBinary());
 55 |   assert.equal(await calc.instance.exports.add(1, 2), 3)
 56 |    * ```
 57 |    */
 58 |   define(module: string, name: string, item: WebAssembly.ImportValue): void {
 59 |     return this.store.addImport(module, name, item);
 60 |   }
 61 | 
 62 |   /**
 63 |    * Instantiate and add a module to the linker's instance cache.
 64 |    * 
 65 |    * Example:
 66 |    * ```js
 67 | const { Linker } = require("@deislabs/wasm-linker-js");
 68 | const { parseText } = require("binaryen");
 69 | const assert = require("assert");
 70 | 
 71 | (async () => {
 72 |   const usingAdd = `
 73 | (module
 74 |     (import "calculator" "add" (func $calc_add (param i32 i32) (result i32)))
 75 |     (export "add" (func $add))
 76 | 
 77 |     (func $add (param i32) (param i32) (result i32)
 78 |         (return
 79 |             (call $calc_add
 80 |                 (local.get 0)
 81 |                 (local.get 1)
 82 |             )
 83 |         )
 84 |     )
 85 | )
 86 | `;
 87 | 
 88 |   const add = `
 89 | (module
 90 |   (memory 1 1)
 91 |   (export "memory" (memory 0))
 92 |   (export "add" (func $add))
 93 | 
 94 |   (func $add (param i32) (param i32) (result i32)
 95 |       (return
 96 |           (i32.add
 97 |               (local.get 0)
 98 |               (local.get 1)
 99 |           )
100 |       )
101 |   )
102 | )
103 | `;
104 | 
105 |   var linker = new Linker();
106 | 
107 |   // the "usingAdd" module above imports calculator.add
108 |   // we link a module that we know exports the functionality
109 |   // required, then instantiate the module that uses it.
110 | 
111 |   await linker.module(
112 |     "calculator",
113 |     new WebAssembly.Module(parseText(add).emitBinary())
114 |   );
115 |   var calc = await linker.instantiate(parseText(add).emitBinary());
116 |   assert.equal(await calc.instance.exports.add(1, 41), 42);
117 | })();
118 |    *```
119 |    */
120 |   async module(name: string, module: WebAssembly.Module): Promise {
121 |     return this.store.addInstance(name, await this.instantiate(module));
122 |   }
123 | 
124 |   /**
125 |    * Instantiate a module using `instantiateStreaming` and add its exports 
126 |    * to the linker's cache.
127 |    * Not available in Safari and Node.js.
128 |    * See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WebAssembly/instantiateStreaming
129 |    * Example:
130 |    * ```js
131 |         var linker = new Linker();
132 |         await linker.moduleStreaming("calculator", fetch("calculator.wasm"));
133 |         var mod = await linker.instantiateStreaming(
134 |           fetch("using_calculator.wasm")
135 |         );
136 |         console.log(mod.instance.exports.multiply(3, 4));
137 | ```
138 |    */
139 |   async moduleStreaming(name: string, source: Response): Promise {
140 |     return this.store.addInstance(
141 |       name,
142 |       await (await this.instantiateStreaming(source)).instance
143 |     );
144 |   }
145 | 
146 |   /**
147 |    * Add an instance to the linker's instance cache.
148 |    * Example:
149 |    * ```js
150 | const { Linker } = require("@deislabs/wasm-linker-js");
151 | const { parseText } = require("binaryen");
152 | const assert = require("assert");
153 | 
154 | (async () => {
155 |   const usingAdd = `
156 | (module
157 |     (import "calculator" "add" (func $calc_add (param i32 i32) (result i32)))
158 |     (export "add" (func $add))
159 | 
160 |     (func $add (param i32) (param i32) (result i32)
161 |         (return
162 |             (call $calc_add
163 |                 (local.get 0)
164 |                 (local.get 1)
165 |             )
166 |         )
167 |     )
168 | )
169 | `;
170 | 
171 |   const add = `
172 | (module
173 |   (memory 1 1)
174 |   (export "memory" (memory 0))
175 |   (export "add" (func $add))
176 | 
177 |   (func $add (param i32) (param i32) (result i32)
178 |       (return
179 |           (i32.add
180 |               (local.get 0)
181 |               (local.get 1)
182 |           )
183 |       )
184 |    )
185 | )
186 | `;
187 | 
188 |     var linker = new Linker();
189 |     var depsInstance = await WebAssembly.instantiate(
190 |       new WebAssembly.Module(parseText(add).emitBinary())
191 |     );
192 |     linker.instance("calculator", depsInstance);
193 |     var calc = await linker.instantiate(parseText(usingAdd).emitBinary());
194 | 
195 |     assert.equal(await calc.instance.exports.add(1, 41), 42);
196 | })();
197 |    *```
198 |    */
199 |   instance(name: string, instance: WebAssembly.Instance): void {
200 |     return this.store.addInstance(name, instance);
201 |   }
202 | 
203 |   /**
204 |    * Alias all exports from one module as another.
205 |    * Example:
206 |    * ```js
207 | const { Linker } = require("@deislabs/wasm-linker-js");
208 | const { parseText } = require("binaryen");
209 | const assert = require("assert");
210 | 
211 | (async () => {
212 |   const add = `
213 | (module
214 |   (memory 1 1)
215 |   (export "memory" (memory 0))
216 |   (export "add" (func $add))
217 |   
218 |   (func $add (param i32) (param i32) (result i32)
219 |       (return
220 |           (i32.add
221 |               (local.get 0)
222 |               (local.get 1)
223 |           )
224 |       )
225 |   )
226 | )
227 | `;
228 | 
229 |   const usingAddWithAlias = `
230 | (module
231 |     (import "calculator1" "add" (func $calc_add1 (param i32 i32) (result i32)))
232 |     (import "calculator2" "add" (func $calc_add2 (param i32 i32) (result i32)))
233 | 
234 |     (memory 1 1)
235 |     (export "memory" (memory 0))
236 | 
237 |     (export "add1" (func $add1))
238 |     (export "add2" (func $add2))
239 | 
240 |     (func $add1 (param i32) (param i32) (result i32)
241 |         (return
242 |             (call $calc_add1
243 |                 (local.get 0)
244 |                 (local.get 1)
245 |             )
246 |         )
247 |     )
248 | 
249 |     (func $add2 (param i32) (param i32) (result i32)
250 |         (return
251 |             (call $calc_add2
252 |                 (local.get 0)
253 |                 (local.get 1)
254 |             )
255 |         )
256 |     )
257 | )
258 | `;
259 | 
260 |     var linker = new Linker();
261 |     // we first define the module that implements `add` as `calculator1`
262 |     await linker.module(
263 |       "calculator1",
264 |       new WebAssembly.Module(parseText(add).emitBinary())
265 |     );
266 |     // we alias `calculator1` as `calculator2`
267 |     linker.alias("calculator1", "calculator2");
268 |     var calc = await linker.instantiate(
269 |       parseText(usingAddWithAlias).emitBinary()
270 |     );
271 | 
272 |     assert.equal(await calc.instance.exports.add1(1, 99), 100);
273 |     assert.equal(await calc.instance.exports.add2(2, 99), 101);
274 | })();
275 |    *```
276 |    */
277 |   alias(module: string, asModule: string): void {
278 |     return this.store.aliasModule(module, asModule);
279 |   }
280 | 
281 |   /**
282 |    * Instantiate a WebAssembly module.
283 |    *
284 |    * This function iterates through the module's imports and performs
285 |    * name based resolution in trying to satisfy all module imports.
286 |    * First, it searches
287 |    *
288 |    * This function iterates through the module's imports and does a
289 |    * name based import resolution based on the exported items of the
290 |    * modules in the store's instance cache, and passes the store's
291 |    * imports cache when actually instantiates the module.
292 |    */
293 |   async instantiate(module: WebAssembly.Module): Promise {
294 |     return this.useAsyncify
295 |       ? await asyncify.instantiate(module, this.store.imports)
296 |       : await WebAssembly.instantiate(module, this.store.imports);
297 |   }
298 | 
299 |   /**
300 |    * Compile and instantiate a WebAssembly module directly from a streamed 
301 |    * underlying source.
302 |    * Not available in Safari and Node.js.
303 |    * See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WebAssembly/instantiateStreaming
304 |    * 
305 |    * Example:
306 |    * ```js
307 |         var linker = new Linker();
308 |         var mod = await linker.instantiateStreaming(
309 |           fetch("calculator.wasm")
310 |         );
311 |         console.log(mod.instance.exports.multiply(3, 4));
312 | ```
313 |    */
314 |   async instantiateStreaming(
315 |     source: Response
316 |   ): Promise {
317 |     return this.useAsyncify
318 |       ? await asyncify.instantiateStreaming(source, this.store.imports)
319 |       : await WebAssembly.instantiateStreaming(source, this.store.imports);
320 |   }
321 | 
322 |   /**
323 |    * Add an imports object to the linker.
324 |    * Example:
325 |    * ```js
326 | const { Linker } = require("@deislabs/wasm-linker-js");
327 | const { parseText } = require("binaryen");
328 | const assert = require("assert");
329 | 
330 | (async () => {
331 |   const usingAdd = `
332 |     (module
333 |         (import "calculator" "add" (func $calc_add (param i32 i32) (result i32))
334 |         (export "add" (func $add))
335 |     
336 |         (func $add (param i32) (param i32) (result i32)
337 |             (return
338 |                 (call $calc_add
339 |                     (local.get 0)
340 |                     (local.get 1)
341 |                 )
342 |             )
343 |         )
344 |     )
345 |     `;
346 | 
347 |   var linker = new Linker();
348 | 
349 |   var importObject = {
350 |     calculator: {
351 |       add: (a: number, b: number) => a + b,
352 |     },
353 |   };
354 | 
355 |   linker.imports(importObject);
356 | 
357 |   var calc = await linker.instantiate(
358 |     parseText(usingAdd).emitBinary());
359 |   assert.equal(await calc.instance.exports.add(1, 2), 3)
360 |    * ```
361 |    */
362 |   imports(importObject: any): void {
363 |     // This ensures that import objects defined using the
364 |     // current convention can be directly reused when
365 |     // using this linker.
366 |     //
367 |     // The signature is not ideal, and in the future we
368 |     // should explore making `importObject` a typed
369 |     // object.
370 |     return this.store.addImportObject(importObject);
371 |   }
372 | 
373 |   // This is currently unused.
374 |   // If we were to use this type of resolution, we would always
375 |   // have to make an explicit choice between first adding imports
376 |   // defined by the user versus those coming from modules (imports
377 |   // not manually defined by a user, but automatically resolved by name).
378 |   //
379 |   // See the implementation of Store.addInstance, but in short,
380 |   // in the case of identical import names, the one defined last
381 |   // will be used, regardless of whether it was explicitly
382 |   // defined using `defined` or with `module`.
383 |   //
384 |   // Leaving this function here for completeness.
385 |   // @ts-ignore
386 |   private resolveImports(module: WebAssembly.Module): void {
387 |     let imports = WebAssembly.Module.imports(module);
388 |     imports.forEach((im) => {
389 |       if (
390 |         this.store.instances[im.module] !== undefined &&
391 |         this.store.instances[im.module].exports[im.name] !== undefined
392 |       ) {
393 |         this.define(
394 |           im.module,
395 |           im.name,
396 |           this.store.instances[im.module].exports[im.name]
397 |         );
398 |       }
399 |     });
400 |   }
401 | }
402 | 
--------------------------------------------------------------------------------
/src/store.ts:
--------------------------------------------------------------------------------
 1 | export class Store {
 2 |   public instances: Record;
 3 |   public imports: Record>;
 4 | 
 5 |   constructor() {
 6 |     this.imports = {};
 7 |     this.instances = {};
 8 |   }
 9 | 
10 |   /**
11 |    * Add an import to the imports cache.
12 |    *
13 |    * Identical import names will always be replaced
14 |    * by the last one defined.
15 |    */
16 |   public addImport(
17 |     module: string,
18 |     name: string,
19 |     item: WebAssembly.ImportValue
20 |   ): void {
21 |     // If this is the first import defined for the given module,
22 |     // create the entry for the module in the imports cache.
23 |     if (this.imports[module] === undefined) {
24 |       this.imports[module] = {};
25 |     }
26 | 
27 |     this.imports[module][name] = item;
28 |   }
29 | 
30 |   /**
31 |    * Add an imports object to the imports cache.
32 |    */
33 |   public addImportObject(importObject: any): void {
34 |     for (const [mod, im] of Object.entries(importObject)) {
35 |       this.imports[mod] = im as Record;
36 |     }
37 |   }
38 | 
39 |   /**
40 |    * Add an instance to the store's instance cache, and
41 |    * all its exports to the imports cache.
42 |    *
43 |    * After each call to the `addInstance` function, the imports
44 |    * cache will contain all exports from the instance, with
45 |    * identical import names replaced by the latest definition
46 |    */
47 |   public addInstance(name: string, instance: WebAssembly.Instance): void {
48 |     this.instances[name] = instance;
49 | 
50 |     for (const [exp, value] of Object.entries(instance.exports)) {
51 |       this.addImport(name, exp, value);
52 |     }
53 |   }
54 | 
55 |   public aliasModule(module: string, asModule: string): void {
56 |     for (const [name, value] of Object.entries(this.imports[module])) {
57 |       this.addImport(asModule, name, value);
58 |     }
59 |   }
60 | }
61 | 
--------------------------------------------------------------------------------
/tests/linker.ts:
--------------------------------------------------------------------------------
  1 | import "mocha";
  2 | import { assert } from "chai";
  3 | import * as wat from "./wat";
  4 | import { Linker } from "../src/linker";
  5 | 
  6 | describe("linker tests", async () => {
  7 |   it("correctly define a single method and instantiate", async () => {
  8 |     let linker = new Linker();
  9 |     linker.define("calculator", "add", (a: number, b: number) => a + b);
 10 |     let instance = await linker.instantiate(wat.moduleFromText(wat.usingAdd));
 11 |     assert.equal(await (instance.exports["add"] as Function)(1, 2), 3);
 12 |   });
 13 | 
 14 |   it("correctly define an entire module and instantiate", async () => {
 15 |     let linker = new Linker();
 16 |     await linker.module("calculator", wat.moduleFromText(wat.add));
 17 |     let instance = await linker.instantiate(wat.moduleFromText(wat.usingAdd));
 18 |     assert.equal(await (instance.exports["add"] as Function)(1, 2), 3);
 19 |   });
 20 | 
 21 |   it("ensure the last import is chosen between identical imports", async () => {
 22 |     let linker = new Linker();
 23 |     linker.define("calculator", "add", (a: number, b: number) => 42);
 24 | 
 25 |     await linker.module("calculator", wat.moduleFromText(wat.add));
 26 |     let instance = await linker.instantiate(wat.moduleFromText(wat.usingAdd));
 27 |     assert.equal(await (instance.exports["add"] as Function)(1, 2), 3);
 28 |   });
 29 | 
 30 |   it("ensure the last import is chosen between identical imports with changed order", async () => {
 31 |     let linker = new Linker();
 32 |     await linker.module("calculator", wat.moduleFromText(wat.add));
 33 |     linker.define("calculator", "add", (a: number, b: number) => 42);
 34 |     let instance = await linker.instantiate(wat.moduleFromText(wat.usingAdd));
 35 |     assert.equal(await (instance.exports["add"] as Function)(1, 2), 42);
 36 |   });
 37 | 
 38 |   it("use last item when multiple identical items are defined", async () => {
 39 |     let linker = new Linker();
 40 | 
 41 |     linker.define("calculator", "add", (a: number, b: number) => 41);
 42 |     linker.define("calculator", "add", (a: number, b: number) => 42);
 43 |     await linker.module("calculator", wat.moduleFromText(wat.add));
 44 |     linker.define("calculator", "add", (a: number, b: number) => 43);
 45 |     let instance = await linker.instantiate(wat.moduleFromText(wat.usingAdd));
 46 |     assert.equal(await (instance.exports["add"] as Function)(1, 2), 43);
 47 |   });
 48 | 
 49 |   it("define an async import", async () => {
 50 |     let useAsyncify = true;
 51 |     let linker = new Linker(useAsyncify);
 52 | 
 53 |     linker.define("calculator", "add", async () => {
 54 |       await sleep(1500);
 55 |       return 42;
 56 |     });
 57 | 
 58 |     let instance = await linker.instantiate(
 59 |       wat.asyncModuleFromText(wat.usingAdd)
 60 |     );
 61 |     assert.equal(await (instance.exports["add"] as Function)(1, 2), 42);
 62 |   });
 63 | 
 64 |   it("add an existing instance to the linker", async () => {
 65 |     let linker = new Linker();
 66 | 
 67 |     let calc = await WebAssembly.instantiate(wat.moduleFromText(wat.add), {});
 68 |     linker.instance("calculator", calc);
 69 |     let instance = await linker.instantiate(wat.moduleFromText(wat.usingAdd));
 70 |     assert.equal(await (instance.exports["add"] as Function)(1, 2), 3);
 71 |   });
 72 | 
 73 |   it("alias modules", async () => {
 74 |     let linker = new Linker();
 75 |     await linker.module("calculator1", wat.moduleFromText(wat.add));
 76 |     linker.alias("calculator1", "calculator2");
 77 |     let instance = await linker.instantiate(
 78 |       wat.asyncModuleFromText(wat.usingAddWithAlias)
 79 |     );
 80 |     assert.equal(await (instance.exports["add"] as Function)(1, 2), 3);
 81 |   });
 82 | 
 83 |   it("add import object to the linker", async () => {
 84 |     let linker = new Linker();
 85 |     var importObject = {
 86 |       calculator: {
 87 |         add: (a: number, b: number) => a + b,
 88 |       },
 89 |     };
 90 | 
 91 |     linker.imports(importObject);
 92 |     let instance = await linker.instantiate(wat.moduleFromText(wat.usingAdd));
 93 |     assert.equal(await (instance.exports["add"] as Function)(1, 2), 3);
 94 |   });
 95 | 
 96 |   it("add import object with multiple imports to the linker", async () => {
 97 |     let linker = new Linker();
 98 |     var importObject = {
 99 |       calculator1: {
100 |         add: (a: number, b: number) => a + b,
101 |       },
102 |       calculator2: {
103 |         add: (a: number, b: number) => a + b,
104 |       },
105 |     };
106 | 
107 |     linker.imports(importObject);
108 |     let instance = await linker.instantiate(
109 |       wat.moduleFromText(wat.usingAddWithAlias)
110 |     );
111 |     assert.equal(await (instance.exports["add"] as Function)(1, 2), 3);
112 |   });
113 | });
114 | 
115 | async function sleep(ms: number) {
116 |   return new Promise((resolve) => {
117 |     setTimeout(resolve, ms);
118 |   });
119 | }
120 | 
--------------------------------------------------------------------------------
/tests/store.ts:
--------------------------------------------------------------------------------
 1 | import "mocha";
 2 | import { assert } from "chai";
 3 | import { Store } from "../src/store";
 4 | import * as wat from "./wat";
 5 | 
 6 | describe("store tests", async () => {
 7 |   it("a new store contains zero imports and instances in cache", () => {
 8 |     let store = new Store();
 9 |     assert.equal(Object.keys(store.imports).length, 0);
10 |     assert.equal(Object.keys(store.instances).length, 0);
11 |   });
12 | 
13 |   it("the length of the import cache for a new store with one import defined is one", () => {
14 |     let store = new Store();
15 |     store.addImport("module1", "function1", () => {});
16 |     assert.equal(Object.keys(store.imports).length, 1);
17 |     assert.equal(Object.keys(store.imports["module1"]).length, 1);
18 |     assert.equal(Object.keys(store.instances).length, 0);
19 |   });
20 | 
21 |   it("a function import is properly added in the import cache", () => {
22 |     let store = new Store();
23 |     let fn = (a: number, b: number) => a + b;
24 |     store.addImport("module1", "function1", fn);
25 |     assert.equal(store.imports["module1"]["function1"], fn);
26 |   });
27 | 
28 |   it("a memory import is properly added in the import cache", () => {
29 |     let store = new Store();
30 |     let memory = new WebAssembly.Memory({ initial: 10, maximum: 100 });
31 |     store.addImport("some-module-import", "memory", memory);
32 |     assert.equal(store.imports["some-module-import"]["memory"], memory);
33 |   });
34 | 
35 |   it("the length of the instance cache for a new store with one instance defined with one export is one", async () => {
36 |     let store = new Store();
37 |     let module = wat.moduleFromText(wat.add);
38 |     store.addInstance("calculator", await WebAssembly.instantiate(module, {}));
39 | 
40 |     assert.equal(Object.keys(store.imports).length, 1);
41 |     assert.equal(Object.keys(store.instances).length, 1);
42 |   });
43 | 
44 |   it("a module alias creates entry for the alias", () => {
45 |     let store = new Store();
46 |     let fn1 = (a: number, b: number) => a + b;
47 |     let fn2 = (a: number, b: number) => a * b;
48 |     store.addImport("calculator1", "function1", fn1);
49 |     store.addImport("calculator1", "function2", fn2);
50 | 
51 |     store.aliasModule("calculator1", "calculator2");
52 |     assert.equal(Object.keys(store.imports["calculator2"]).length, 2);
53 |     assert.equal(store.imports["calculator2"]["function1"], fn1);
54 |     assert.equal(store.imports["calculator2"]["function2"], fn2);
55 |   });
56 | 
57 |   it("an import object is correctly added to the store", () => {
58 |     let store = new Store();
59 | 
60 |     let name = "someModule";
61 |     let fnName = "imported_func";
62 |     let fn = () => {};
63 | 
64 |     let importObject = {
65 |       [name]: {
66 |         [fnName]: fn,
67 |       },
68 |     };
69 | 
70 |     store.addImportObject(importObject);
71 | 
72 |     assert.equal(Object.keys(store.imports).length, 1);
73 |     assert.equal(Object.keys(store.imports["someModule"]).length, 1);
74 |     assert.equal(store.imports[name][fnName], fn);
75 |   });
76 | });
77 | 
--------------------------------------------------------------------------------
/tests/wat.ts:
--------------------------------------------------------------------------------
 1 | import { parseText } from "binaryen";
 2 | 
 3 | export function moduleFromText(text: string): WebAssembly.Module {
 4 |   return new WebAssembly.Module(parseText(text).emitBinary());
 5 | }
 6 | 
 7 | export function asyncModuleFromText(text: string): WebAssembly.Module {
 8 |   let bytes = parseText(text);
 9 |   bytes.runPasses(["asyncify"]);
10 | 
11 |   return new WebAssembly.Module(bytes.emitBinary());
12 | }
13 | 
14 | export const add = `
15 | (module
16 |     (memory 1 1)
17 |     (export "memory" (memory 0))
18 |     (export "add" (func $add))
19 |     
20 |     (func $add (param i32) (param i32) (result i32)
21 |         (return
22 |             (i32.add
23 |                 (local.get 0)
24 |                 (local.get 1)
25 |             )
26 |         )
27 |     )
28 | )
29 | `;
30 | 
31 | export const usingAdd = `
32 | (module
33 |     (import "calculator" "add" (func $calc_add (param i32 i32) (result i32)))
34 |     
35 |     (memory 1 1)
36 |     (export "memory" (memory 0))
37 |     (export "add" (func $add))
38 | 
39 |     (func $add (param i32) (param i32) (result i32)
40 |         (return
41 |             (call $calc_add
42 |                 (local.get 0)
43 |                 (local.get 1)
44 |             )
45 |         )
46 |     )
47 | )
48 | `;
49 | 
50 | export const usingAddWithAlias = `
51 | (module
52 |     (import "calculator1" "add" (func $calc_add (param i32 i32) (result i32)))
53 |     (import "calculator2" "add" (func $calc_add2 (param i32 i32) (result i32)))
54 | 
55 |     (memory 1 1)
56 |     (export "memory" (memory 0))
57 |     (export "add" (func $add))
58 | 
59 |     (func $add (param i32) (param i32) (result i32)
60 |         (return
61 |             (call $calc_add
62 |                 (local.get 0)
63 |                 (local.get 1)
64 |             )
65 |         )
66 |     )
67 | )
68 | `;
69 | 
--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
 1 | {
 2 |   "compilerOptions": {
 3 |     "target": "es6",
 4 |     "lib": ["es2015", "dom"],
 5 |     "module": "CommonJS",
 6 |     "declaration": true,
 7 |     "outDir": "dist",
 8 |     "strict": true,
 9 |     "sourceMap": true,
10 |     "baseUrl": ".",
11 |     "paths": {
12 |       "*": ["src/@types/*"]
13 |     }
14 |   },
15 |   "include": ["src/**/*", "tests/**/*"]
16 | }
17 | 
--------------------------------------------------------------------------------
/webpack.config.js:
--------------------------------------------------------------------------------
 1 | path = require("path");
 2 | module.exports = {
 3 |   entry: "./src/index.ts",
 4 |   devtool: "inline-source-map",
 5 |   module: {
 6 |     rules: [
 7 |       {
 8 |         test: /\.tsx?$/,
 9 |         use: "ts-loader",
10 |         exclude: /node_modules/,
11 |       },
12 |     ],
13 |   },
14 |   resolve: {
15 |     modules: [path.resolve(__dirname, "./src"), "node_modules"],
16 |     extensions: [".ts", ".js"],
17 |   },
18 |   output: {
19 |     filename: "wasm-linker.js",
20 |     path: path.resolve(__dirname, "dist"),
21 |     libraryTarget: "umd",
22 |   },
23 | };
24 | 
--------------------------------------------------------------------------------