├── .gitignore ├── LICENSE ├── README.md ├── frontend ├── .gitignore ├── README.md ├── package-lock.json ├── package.json ├── public │ ├── favicon.png │ ├── global.css │ ├── images │ │ └── smiley.png │ └── index.html ├── rollup.config.js └── src │ ├── App.svelte │ ├── Chat.svelte │ ├── main.js │ ├── string-channel.js │ ├── string-channel.ts │ └── worker.js └── server ├── index.js ├── package-lock.json ├── package.json ├── string-channel.js └── string-channel.ts /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | lerna-debug.log* 8 | 9 | # Diagnostic reports (https://nodejs.org/api/report.html) 10 | report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json 11 | 12 | # Runtime data 13 | pids 14 | *.pid 15 | *.seed 16 | *.pid.lock 17 | 18 | # Directory for instrumented libs generated by jscoverage/JSCover 19 | lib-cov 20 | 21 | # Coverage directory used by tools like istanbul 22 | coverage 23 | *.lcov 24 | 25 | # nyc test coverage 26 | .nyc_output 27 | 28 | # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) 29 | .grunt 30 | 31 | # Bower dependency directory (https://bower.io/) 32 | bower_components 33 | 34 | # node-waf configuration 35 | .lock-wscript 36 | 37 | # Compiled binary addons (https://nodejs.org/api/addons.html) 38 | build/Release 39 | 40 | # Dependency directories 41 | node_modules/ 42 | jspm_packages/ 43 | 44 | # TypeScript v1 declaration files 45 | typings/ 46 | 47 | # TypeScript cache 48 | *.tsbuildinfo 49 | 50 | # Optional npm cache directory 51 | .npm 52 | 53 | # Optional eslint cache 54 | .eslintcache 55 | 56 | # Microbundle cache 57 | .rpt2_cache/ 58 | .rts2_cache_cjs/ 59 | .rts2_cache_es/ 60 | .rts2_cache_umd/ 61 | 62 | # Optional REPL history 63 | .node_repl_history 64 | 65 | # Output of 'npm pack' 66 | *.tgz 67 | 68 | # Yarn Integrity file 69 | .yarn-integrity 70 | 71 | # dotenv environment variables file 72 | .env 73 | .env.test 74 | 75 | # parcel-bundler cache (https://parceljs.org/) 76 | .cache 77 | 78 | # Next.js build output 79 | .next 80 | 81 | # Nuxt.js build / generate output 82 | .nuxt 83 | dist 84 | 85 | # Gatsby files 86 | .cache/ 87 | # Comment in the public line in if your project uses Gatsby and *not* Next.js 88 | # https://nextjs.org/blog/next-9-1#public-directory-support 89 | # public 90 | 91 | # vuepress build output 92 | .vuepress/dist 93 | 94 | # Serverless directories 95 | .serverless/ 96 | 97 | # FuseBox cache 98 | .fusebox/ 99 | 100 | # DynamoDB Local files 101 | .dynamodb/ 102 | 103 | # TernJS port file 104 | .tern-port 105 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Apache License 2 | Version 2.0, January 2004 3 | http://www.apache.org/licenses/ 4 | 5 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 6 | 7 | 1. Definitions. 8 | 9 | "License" shall mean the terms and conditions for use, reproduction, 10 | and distribution as defined by Sections 1 through 9 of this document. 11 | 12 | "Licensor" shall mean the copyright owner or entity authorized by 13 | the copyright owner that is granting the License. 14 | 15 | "Legal Entity" shall mean the union of the acting entity and all 16 | other entities that control, are controlled by, or are under common 17 | control with that entity. For the purposes of this definition, 18 | "control" means (i) the power, direct or indirect, to cause the 19 | direction or management of such entity, whether by contract or 20 | otherwise, or (ii) ownership of fifty percent (50%) or more of the 21 | outstanding shares, or (iii) beneficial ownership of such entity. 22 | 23 | "You" (or "Your") shall mean an individual or Legal Entity 24 | exercising permissions granted by this License. 25 | 26 | "Source" form shall mean the preferred form for making modifications, 27 | including but not limited to software source code, documentation 28 | source, and configuration files. 29 | 30 | "Object" form shall mean any form resulting from mechanical 31 | transformation or translation of a Source form, including but 32 | not limited to compiled object code, generated documentation, 33 | and conversions to other media types. 34 | 35 | "Work" shall mean the work of authorship, whether in Source or 36 | Object form, made available under the License, as indicated by a 37 | copyright notice that is included in or attached to the work 38 | (an example is provided in the Appendix below). 39 | 40 | "Derivative Works" shall mean any work, whether in Source or Object 41 | form, that is based on (or derived from) the Work and for which the 42 | editorial revisions, annotations, elaborations, or other modifications 43 | represent, as a whole, an original work of authorship. For the purposes 44 | of this License, Derivative Works shall not include works that remain 45 | separable from, or merely link (or bind by name) to the interfaces of, 46 | the Work and Derivative Works thereof. 47 | 48 | "Contribution" shall mean any work of authorship, including 49 | the original version of the Work and any modifications or additions 50 | to that Work or Derivative Works thereof, that is intentionally 51 | submitted to Licensor for inclusion in the Work by the copyright owner 52 | or by an individual or Legal Entity authorized to submit on behalf of 53 | the copyright owner. For the purposes of this definition, "submitted" 54 | means any form of electronic, verbal, or written communication sent 55 | to the Licensor or its representatives, including but not limited to 56 | communication on electronic mailing lists, source code control systems, 57 | and issue tracking systems that are managed by, or on behalf of, the 58 | Licensor for the purpose of discussing and improving the Work, but 59 | excluding communication that is conspicuously marked or otherwise 60 | designated in writing by the copyright owner as "Not a Contribution." 61 | 62 | "Contributor" shall mean Licensor and any individual or Legal Entity 63 | on behalf of whom a Contribution has been received by Licensor and 64 | subsequently incorporated within the Work. 65 | 66 | 2. Grant of Copyright License. Subject to the terms and conditions of 67 | this License, each Contributor hereby grants to You a perpetual, 68 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 69 | copyright license to reproduce, prepare Derivative Works of, 70 | publicly display, publicly perform, sublicense, and distribute the 71 | Work and such Derivative Works in Source or Object form. 72 | 73 | 3. Grant of Patent License. Subject to the terms and conditions of 74 | this License, each Contributor hereby grants to You a perpetual, 75 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 76 | (except as stated in this section) patent license to make, have made, 77 | use, offer to sell, sell, import, and otherwise transfer the Work, 78 | where such license applies only to those patent claims licensable 79 | by such Contributor that are necessarily infringed by their 80 | Contribution(s) alone or by combination of their Contribution(s) 81 | with the Work to which such Contribution(s) was submitted. If You 82 | institute patent litigation against any entity (including a 83 | cross-claim or counterclaim in a lawsuit) alleging that the Work 84 | or a Contribution incorporated within the Work constitutes direct 85 | or contributory patent infringement, then any patent licenses 86 | granted to You under this License for that Work shall terminate 87 | as of the date such litigation is filed. 88 | 89 | 4. Redistribution. You may reproduce and distribute copies of the 90 | Work or Derivative Works thereof in any medium, with or without 91 | modifications, and in Source or Object form, provided that You 92 | meet the following conditions: 93 | 94 | (a) You must give any other recipients of the Work or 95 | Derivative Works a copy of this License; and 96 | 97 | (b) You must cause any modified files to carry prominent notices 98 | stating that You changed the files; and 99 | 100 | (c) You must retain, in the Source form of any Derivative Works 101 | that You distribute, all copyright, patent, trademark, and 102 | attribution notices from the Source form of the Work, 103 | excluding those notices that do not pertain to any part of 104 | the Derivative Works; and 105 | 106 | (d) If the Work includes a "NOTICE" text file as part of its 107 | distribution, then any Derivative Works that You distribute must 108 | include a readable copy of the attribution notices contained 109 | within such NOTICE file, excluding those notices that do not 110 | pertain to any part of the Derivative Works, in at least one 111 | of the following places: within a NOTICE text file distributed 112 | as part of the Derivative Works; within the Source form or 113 | documentation, if provided along with the Derivative Works; or, 114 | within a display generated by the Derivative Works, if and 115 | wherever such third-party notices normally appear. The contents 116 | of the NOTICE file are for informational purposes only and 117 | do not modify the License. You may add Your own attribution 118 | notices within Derivative Works that You distribute, alongside 119 | or as an addendum to the NOTICE text from the Work, provided 120 | that such additional attribution notices cannot be construed 121 | as modifying the License. 122 | 123 | You may add Your own copyright statement to Your modifications and 124 | may provide additional or different license terms and conditions 125 | for use, reproduction, or distribution of Your modifications, or 126 | for any such Derivative Works as a whole, provided Your use, 127 | reproduction, and distribution of the Work otherwise complies with 128 | the conditions stated in this License. 129 | 130 | 5. Submission of Contributions. Unless You explicitly state otherwise, 131 | any Contribution intentionally submitted for inclusion in the Work 132 | by You to the Licensor shall be under the terms and conditions of 133 | this License, without any additional terms or conditions. 134 | Notwithstanding the above, nothing herein shall supersede or modify 135 | the terms of any separate license agreement you may have executed 136 | with Licensor regarding such Contributions. 137 | 138 | 6. Trademarks. This License does not grant permission to use the trade 139 | names, trademarks, service marks, or product names of the Licensor, 140 | except as required for reasonable and customary use in describing the 141 | origin of the Work and reproducing the content of the NOTICE file. 142 | 143 | 7. Disclaimer of Warranty. Unless required by applicable law or 144 | agreed to in writing, Licensor provides the Work (and each 145 | Contributor provides its Contributions) on an "AS IS" BASIS, 146 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 147 | implied, including, without limitation, any warranties or conditions 148 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A 149 | PARTICULAR PURPOSE. You are solely responsible for determining the 150 | appropriateness of using or redistributing the Work and assume any 151 | risks associated with Your exercise of permissions under this License. 152 | 153 | 8. Limitation of Liability. In no event and under no legal theory, 154 | whether in tort (including negligence), contract, or otherwise, 155 | unless required by applicable law (such as deliberate and grossly 156 | negligent acts) or agreed to in writing, shall any Contributor be 157 | liable to You for damages, including any direct, indirect, special, 158 | incidental, or consequential damages of any character arising as a 159 | result of this License or out of the use or inability to use the 160 | Work (including but not limited to damages for loss of goodwill, 161 | work stoppage, computer failure or malfunction, or any and all 162 | other commercial damages or losses), even if such Contributor 163 | has been advised of the possibility of such damages. 164 | 165 | 9. Accepting Warranty or Additional Liability. While redistributing 166 | the Work or Derivative Works thereof, You may choose to offer, 167 | and charge a fee for, acceptance of support, warranty, indemnity, 168 | or other liability obligations and/or rights consistent with this 169 | License. However, in accepting such obligations, You may act only 170 | on Your own behalf and on Your sole responsibility, not on behalf 171 | of any other Contributor, and only if You agree to indemnify, 172 | defend, and hold each Contributor harmless for any liability 173 | incurred by, or claims asserted against, such Contributor by reason 174 | of your accepting any such warranty or additional liability. 175 | 176 | END OF TERMS AND CONDITIONS 177 | 178 | APPENDIX: How to apply the Apache License to your work. 179 | 180 | To apply the Apache License to your work, attach the following 181 | boilerplate notice, with the fields enclosed by brackets "[]" 182 | replaced with your own identifying information. (Don't include 183 | the brackets!) The text should be enclosed in the appropriate 184 | comment syntax for the file format. We also recommend that a 185 | file or class name and description of purpose be included on the 186 | same "printed page" as the copyright notice for easier 187 | identification within third-party archives. 188 | 189 | Copyright [yyyy] [name of copyright owner] 190 | 191 | Licensed under the Apache License, Version 2.0 (the "License"); 192 | you may not use this file except in compliance with the License. 193 | You may obtain a copy of the License at 194 | 195 | http://www.apache.org/licenses/LICENSE-2.0 196 | 197 | Unless required by applicable law or agreed to in writing, software 198 | distributed under the License is distributed on an "AS IS" BASIS, 199 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 200 | See the License for the specific language governing permissions and 201 | limitations under the License. 202 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # comlink-websocket-chat-experiment 2 | do not run in production 3 | -------------------------------------------------------------------------------- /frontend/.gitignore: -------------------------------------------------------------------------------- 1 | /node_modules/ 2 | /public/build/ 3 | 4 | .DS_Store 5 | -------------------------------------------------------------------------------- /frontend/README.md: -------------------------------------------------------------------------------- 1 | *Looking for a shareable component template? Go here --> [sveltejs/component-template](https://github.com/sveltejs/component-template)* 2 | 3 | --- 4 | 5 | # svelte app 6 | 7 | This is a project template for [Svelte](https://svelte.dev) apps. It lives at https://github.com/sveltejs/template. 8 | 9 | To create a new project based on this template using [degit](https://github.com/Rich-Harris/degit): 10 | 11 | ```bash 12 | npx degit sveltejs/template svelte-app 13 | cd svelte-app 14 | ``` 15 | 16 | *Note that you will need to have [Node.js](https://nodejs.org) installed.* 17 | 18 | 19 | ## Get started 20 | 21 | Install the dependencies... 22 | 23 | ```bash 24 | cd svelte-app 25 | npm install 26 | ``` 27 | 28 | ...then start [Rollup](https://rollupjs.org): 29 | 30 | ```bash 31 | npm run dev 32 | ``` 33 | 34 | Navigate to [localhost:5000](http://localhost:5000). You should see your app running. Edit a component file in `src`, save it, and reload the page to see your changes. 35 | 36 | By default, the server will only respond to requests from localhost. To allow connections from other computers, edit the `sirv` commands in package.json to include the option `--host 0.0.0.0`. 37 | 38 | 39 | ## Building and running in production mode 40 | 41 | To create an optimised version of the app: 42 | 43 | ```bash 44 | npm run build 45 | ``` 46 | 47 | You can run the newly built app with `npm run start`. This uses [sirv](https://github.com/lukeed/sirv), which is included in your package.json's `dependencies` so that the app will work when you deploy to platforms like [Heroku](https://heroku.com). 48 | 49 | 50 | ## Single-page app mode 51 | 52 | By default, sirv will only respond to requests that match files in `public`. This is to maximise compatibility with static fileservers, allowing you to deploy your app anywhere. 53 | 54 | If you're building a single-page app (SPA) with multiple routes, sirv needs to be able to respond to requests for *any* path. You can make it so by editing the `"start"` command in package.json: 55 | 56 | ```js 57 | "start": "sirv public --single" 58 | ``` 59 | 60 | 61 | ## Deploying to the web 62 | 63 | ### With [now](https://zeit.co/now) 64 | 65 | Install `now` if you haven't already: 66 | 67 | ```bash 68 | npm install -g now 69 | ``` 70 | 71 | Then, from within your project folder: 72 | 73 | ```bash 74 | cd public 75 | now deploy --name my-project 76 | ``` 77 | 78 | As an alternative, use the [Now desktop client](https://zeit.co/download) and simply drag the unzipped project folder to the taskbar icon. 79 | 80 | ### With [surge](https://surge.sh/) 81 | 82 | Install `surge` if you haven't already: 83 | 84 | ```bash 85 | npm install -g surge 86 | ``` 87 | 88 | Then, from within your project folder: 89 | 90 | ```bash 91 | npm run build 92 | surge public my-project.surge.sh 93 | ``` 94 | -------------------------------------------------------------------------------- /frontend/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "svelte-app", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@babel/code-frame": { 8 | "version": "7.8.3", 9 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.8.3.tgz", 10 | "integrity": "sha512-a9gxpmdXtZEInkCSHUJDLHZVBgb1QS0jhss4cPP93EW7s+uC5bikET2twEF3KV+7rDblJcmNvTR7VJejqd2C2g==", 11 | "dev": true, 12 | "requires": { 13 | "@babel/highlight": "^7.8.3" 14 | } 15 | }, 16 | "@babel/highlight": { 17 | "version": "7.8.3", 18 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.8.3.tgz", 19 | "integrity": "sha512-PX4y5xQUvy0fnEVHrYOarRPXVWafSjTW9T0Hab8gVIawpl2Sj0ORyrygANq+KjcNlSSTw0YCLSNA8OyZ1I4yEg==", 20 | "dev": true, 21 | "requires": { 22 | "chalk": "^2.0.0", 23 | "esutils": "^2.0.2", 24 | "js-tokens": "^4.0.0" 25 | } 26 | }, 27 | "@polka/url": { 28 | "version": "0.5.0", 29 | "resolved": "https://registry.npmjs.org/@polka/url/-/url-0.5.0.tgz", 30 | "integrity": "sha512-oZLYFEAzUKyi3SKnXvj32ZCEGH6RDnao7COuCVhDydMS9NrCSVXhM79VaKyP5+Zc33m0QXEd2DN3UkU7OsHcfw==" 31 | }, 32 | "@rollup/plugin-commonjs": { 33 | "version": "11.0.2", 34 | "resolved": "https://registry.npmjs.org/@rollup/plugin-commonjs/-/plugin-commonjs-11.0.2.tgz", 35 | "integrity": "sha512-MPYGZr0qdbV5zZj8/2AuomVpnRVXRU5XKXb3HVniwRoRCreGlf5kOE081isNWeiLIi6IYkwTX9zE0/c7V8g81g==", 36 | "dev": true, 37 | "requires": { 38 | "@rollup/pluginutils": "^3.0.0", 39 | "estree-walker": "^1.0.1", 40 | "is-reference": "^1.1.2", 41 | "magic-string": "^0.25.2", 42 | "resolve": "^1.11.0" 43 | } 44 | }, 45 | "@rollup/plugin-node-resolve": { 46 | "version": "7.1.1", 47 | "resolved": "https://registry.npmjs.org/@rollup/plugin-node-resolve/-/plugin-node-resolve-7.1.1.tgz", 48 | "integrity": "sha512-14ddhD7TnemeHE97a4rLOhobfYvUVcaYuqTnL8Ti7Jxi9V9Jr5LY7Gko4HZ5k4h4vqQM0gBQt6tsp9xXW94WPA==", 49 | "dev": true, 50 | "requires": { 51 | "@rollup/pluginutils": "^3.0.6", 52 | "@types/resolve": "0.0.8", 53 | "builtin-modules": "^3.1.0", 54 | "is-module": "^1.0.0", 55 | "resolve": "^1.14.2" 56 | } 57 | }, 58 | "@rollup/pluginutils": { 59 | "version": "3.0.8", 60 | "resolved": "https://registry.npmjs.org/@rollup/pluginutils/-/pluginutils-3.0.8.tgz", 61 | "integrity": "sha512-rYGeAc4sxcZ+kPG/Tw4/fwJODC3IXHYDH4qusdN/b6aLw5LPUbzpecYbEJh4sVQGPFJxd2dBU4kc1H3oy9/bnw==", 62 | "dev": true, 63 | "requires": { 64 | "estree-walker": "^1.0.1" 65 | } 66 | }, 67 | "@types/estree": { 68 | "version": "0.0.39", 69 | "resolved": "https://registry.npmjs.org/@types/estree/-/estree-0.0.39.tgz", 70 | "integrity": "sha512-EYNwp3bU+98cpU4lAWYYL7Zz+2gryWH1qbdDTidVd6hkiR6weksdbMadyXKXNPEkQFhXM+hVO9ZygomHXp+AIw==", 71 | "dev": true 72 | }, 73 | "@types/node": { 74 | "version": "13.7.0", 75 | "resolved": "https://registry.npmjs.org/@types/node/-/node-13.7.0.tgz", 76 | "integrity": "sha512-GnZbirvmqZUzMgkFn70c74OQpTTUcCzlhQliTzYjQMqg+hVKcDnxdL19Ne3UdYzdMA/+W3eb646FWn/ZaT1NfQ==", 77 | "dev": true 78 | }, 79 | "@types/resolve": { 80 | "version": "0.0.8", 81 | "resolved": "https://registry.npmjs.org/@types/resolve/-/resolve-0.0.8.tgz", 82 | "integrity": "sha512-auApPaJf3NPfe18hSoJkp8EbZzer2ISk7o8mCC3M9he/a04+gbMF97NkpD2S8riMGvm4BMRI59/SZQSaLTKpsQ==", 83 | "dev": true, 84 | "requires": { 85 | "@types/node": "*" 86 | } 87 | }, 88 | "acorn": { 89 | "version": "7.1.0", 90 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.1.0.tgz", 91 | "integrity": "sha512-kL5CuoXA/dgxlBbVrflsflzQ3PAas7RYZB52NOm/6839iVYJgKMJ3cQJD+t2i5+qFa8h3MDpEOJiS64E8JLnSQ==", 92 | "dev": true 93 | }, 94 | "ansi-styles": { 95 | "version": "3.2.1", 96 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 97 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 98 | "dev": true, 99 | "requires": { 100 | "color-convert": "^1.9.0" 101 | } 102 | }, 103 | "anymatch": { 104 | "version": "2.0.0", 105 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", 106 | "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", 107 | "dev": true, 108 | "requires": { 109 | "micromatch": "^3.1.4", 110 | "normalize-path": "^2.1.1" 111 | }, 112 | "dependencies": { 113 | "normalize-path": { 114 | "version": "2.1.1", 115 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", 116 | "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", 117 | "dev": true, 118 | "requires": { 119 | "remove-trailing-separator": "^1.0.1" 120 | } 121 | } 122 | } 123 | }, 124 | "arr-diff": { 125 | "version": "4.0.0", 126 | "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", 127 | "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", 128 | "dev": true 129 | }, 130 | "arr-flatten": { 131 | "version": "1.1.0", 132 | "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", 133 | "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==", 134 | "dev": true 135 | }, 136 | "arr-union": { 137 | "version": "3.1.0", 138 | "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", 139 | "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=", 140 | "dev": true 141 | }, 142 | "array-unique": { 143 | "version": "0.3.2", 144 | "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", 145 | "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", 146 | "dev": true 147 | }, 148 | "assign-symbols": { 149 | "version": "1.0.0", 150 | "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", 151 | "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=", 152 | "dev": true 153 | }, 154 | "async-each": { 155 | "version": "1.0.3", 156 | "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.3.tgz", 157 | "integrity": "sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==", 158 | "dev": true 159 | }, 160 | "async-limiter": { 161 | "version": "1.0.1", 162 | "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz", 163 | "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==", 164 | "dev": true 165 | }, 166 | "atob": { 167 | "version": "2.1.2", 168 | "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", 169 | "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", 170 | "dev": true 171 | }, 172 | "base": { 173 | "version": "0.11.2", 174 | "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz", 175 | "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", 176 | "dev": true, 177 | "requires": { 178 | "cache-base": "^1.0.1", 179 | "class-utils": "^0.3.5", 180 | "component-emitter": "^1.2.1", 181 | "define-property": "^1.0.0", 182 | "isobject": "^3.0.1", 183 | "mixin-deep": "^1.2.0", 184 | "pascalcase": "^0.1.1" 185 | }, 186 | "dependencies": { 187 | "define-property": { 188 | "version": "1.0.0", 189 | "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", 190 | "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", 191 | "dev": true, 192 | "requires": { 193 | "is-descriptor": "^1.0.0" 194 | } 195 | }, 196 | "is-accessor-descriptor": { 197 | "version": "1.0.0", 198 | "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", 199 | "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", 200 | "dev": true, 201 | "requires": { 202 | "kind-of": "^6.0.0" 203 | } 204 | }, 205 | "is-data-descriptor": { 206 | "version": "1.0.0", 207 | "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", 208 | "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", 209 | "dev": true, 210 | "requires": { 211 | "kind-of": "^6.0.0" 212 | } 213 | }, 214 | "is-descriptor": { 215 | "version": "1.0.2", 216 | "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", 217 | "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", 218 | "dev": true, 219 | "requires": { 220 | "is-accessor-descriptor": "^1.0.0", 221 | "is-data-descriptor": "^1.0.0", 222 | "kind-of": "^6.0.2" 223 | } 224 | } 225 | } 226 | }, 227 | "binary-extensions": { 228 | "version": "1.13.1", 229 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", 230 | "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", 231 | "dev": true 232 | }, 233 | "bindings": { 234 | "version": "1.5.0", 235 | "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", 236 | "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", 237 | "dev": true, 238 | "optional": true, 239 | "requires": { 240 | "file-uri-to-path": "1.0.0" 241 | } 242 | }, 243 | "braces": { 244 | "version": "2.3.2", 245 | "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", 246 | "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", 247 | "dev": true, 248 | "requires": { 249 | "arr-flatten": "^1.1.0", 250 | "array-unique": "^0.3.2", 251 | "extend-shallow": "^2.0.1", 252 | "fill-range": "^4.0.0", 253 | "isobject": "^3.0.1", 254 | "repeat-element": "^1.1.2", 255 | "snapdragon": "^0.8.1", 256 | "snapdragon-node": "^2.0.1", 257 | "split-string": "^3.0.2", 258 | "to-regex": "^3.0.1" 259 | }, 260 | "dependencies": { 261 | "extend-shallow": { 262 | "version": "2.0.1", 263 | "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", 264 | "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", 265 | "dev": true, 266 | "requires": { 267 | "is-extendable": "^0.1.0" 268 | } 269 | } 270 | } 271 | }, 272 | "buffer-from": { 273 | "version": "1.1.1", 274 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", 275 | "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==", 276 | "dev": true 277 | }, 278 | "builtin-modules": { 279 | "version": "3.1.0", 280 | "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-3.1.0.tgz", 281 | "integrity": "sha512-k0KL0aWZuBt2lrxrcASWDfwOLMnodeQjodT/1SxEQAXsHANgo6ZC/VEaSEHCXt7aSTZ4/4H5LKa+tBXmW7Vtvw==", 282 | "dev": true 283 | }, 284 | "cache-base": { 285 | "version": "1.0.1", 286 | "resolved": "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz", 287 | "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==", 288 | "dev": true, 289 | "requires": { 290 | "collection-visit": "^1.0.0", 291 | "component-emitter": "^1.2.1", 292 | "get-value": "^2.0.6", 293 | "has-value": "^1.0.0", 294 | "isobject": "^3.0.1", 295 | "set-value": "^2.0.0", 296 | "to-object-path": "^0.3.0", 297 | "union-value": "^1.0.0", 298 | "unset-value": "^1.0.0" 299 | } 300 | }, 301 | "chalk": { 302 | "version": "2.4.2", 303 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 304 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 305 | "dev": true, 306 | "requires": { 307 | "ansi-styles": "^3.2.1", 308 | "escape-string-regexp": "^1.0.5", 309 | "supports-color": "^5.3.0" 310 | } 311 | }, 312 | "chokidar": { 313 | "version": "2.1.8", 314 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", 315 | "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", 316 | "dev": true, 317 | "requires": { 318 | "anymatch": "^2.0.0", 319 | "async-each": "^1.0.1", 320 | "braces": "^2.3.2", 321 | "fsevents": "^1.2.7", 322 | "glob-parent": "^3.1.0", 323 | "inherits": "^2.0.3", 324 | "is-binary-path": "^1.0.0", 325 | "is-glob": "^4.0.0", 326 | "normalize-path": "^3.0.0", 327 | "path-is-absolute": "^1.0.0", 328 | "readdirp": "^2.2.1", 329 | "upath": "^1.1.1" 330 | } 331 | }, 332 | "class-utils": { 333 | "version": "0.3.6", 334 | "resolved": "https://registry.npmjs.org/class-utils/-/class-utils-0.3.6.tgz", 335 | "integrity": "sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==", 336 | "dev": true, 337 | "requires": { 338 | "arr-union": "^3.1.0", 339 | "define-property": "^0.2.5", 340 | "isobject": "^3.0.0", 341 | "static-extend": "^0.1.1" 342 | }, 343 | "dependencies": { 344 | "define-property": { 345 | "version": "0.2.5", 346 | "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", 347 | "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", 348 | "dev": true, 349 | "requires": { 350 | "is-descriptor": "^0.1.0" 351 | } 352 | } 353 | } 354 | }, 355 | "collection-visit": { 356 | "version": "1.0.0", 357 | "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz", 358 | "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=", 359 | "dev": true, 360 | "requires": { 361 | "map-visit": "^1.0.0", 362 | "object-visit": "^1.0.0" 363 | } 364 | }, 365 | "color-convert": { 366 | "version": "1.9.3", 367 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 368 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 369 | "dev": true, 370 | "requires": { 371 | "color-name": "1.1.3" 372 | } 373 | }, 374 | "color-name": { 375 | "version": "1.1.3", 376 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 377 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 378 | "dev": true 379 | }, 380 | "comlink": { 381 | "version": "4.2.0", 382 | "resolved": "https://registry.npmjs.org/comlink/-/comlink-4.2.0.tgz", 383 | "integrity": "sha512-33hF3yYzZicIWLREuvluTGSZkbAXYwRmrA9IYQo0/P+0O45qgvzR0FHX4tRZztObFMTs/RhnN0G0UBNDAZSiCg==" 384 | }, 385 | "commander": { 386 | "version": "2.20.3", 387 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", 388 | "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", 389 | "dev": true 390 | }, 391 | "component-emitter": { 392 | "version": "1.3.0", 393 | "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", 394 | "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==", 395 | "dev": true 396 | }, 397 | "console-clear": { 398 | "version": "1.1.1", 399 | "resolved": "https://registry.npmjs.org/console-clear/-/console-clear-1.1.1.tgz", 400 | "integrity": "sha512-pMD+MVR538ipqkG5JXeOEbKWS5um1H4LUUccUQG68qpeqBYbzYy79Gh55jkd2TtPdRfUaLWdv6LPP//5Zt0aPQ==" 401 | }, 402 | "copy-descriptor": { 403 | "version": "0.1.1", 404 | "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", 405 | "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", 406 | "dev": true 407 | }, 408 | "core-util-is": { 409 | "version": "1.0.2", 410 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 411 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", 412 | "dev": true 413 | }, 414 | "debug": { 415 | "version": "2.6.9", 416 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 417 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 418 | "dev": true, 419 | "requires": { 420 | "ms": "2.0.0" 421 | } 422 | }, 423 | "decode-uri-component": { 424 | "version": "0.2.0", 425 | "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", 426 | "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", 427 | "dev": true 428 | }, 429 | "define-property": { 430 | "version": "2.0.2", 431 | "resolved": "https://registry.npmjs.org/define-property/-/define-property-2.0.2.tgz", 432 | "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", 433 | "dev": true, 434 | "requires": { 435 | "is-descriptor": "^1.0.2", 436 | "isobject": "^3.0.1" 437 | }, 438 | "dependencies": { 439 | "is-accessor-descriptor": { 440 | "version": "1.0.0", 441 | "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", 442 | "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", 443 | "dev": true, 444 | "requires": { 445 | "kind-of": "^6.0.0" 446 | } 447 | }, 448 | "is-data-descriptor": { 449 | "version": "1.0.0", 450 | "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", 451 | "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", 452 | "dev": true, 453 | "requires": { 454 | "kind-of": "^6.0.0" 455 | } 456 | }, 457 | "is-descriptor": { 458 | "version": "1.0.2", 459 | "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", 460 | "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", 461 | "dev": true, 462 | "requires": { 463 | "is-accessor-descriptor": "^1.0.0", 464 | "is-data-descriptor": "^1.0.0", 465 | "kind-of": "^6.0.2" 466 | } 467 | } 468 | } 469 | }, 470 | "escape-string-regexp": { 471 | "version": "1.0.5", 472 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 473 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 474 | "dev": true 475 | }, 476 | "estree-walker": { 477 | "version": "1.0.1", 478 | "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-1.0.1.tgz", 479 | "integrity": "sha512-1fMXF3YP4pZZVozF8j/ZLfvnR8NSIljt56UhbZ5PeeDmmGHpgpdwQt7ITlGvYaQukCvuBRMLEiKiYC+oeIg4cg==", 480 | "dev": true 481 | }, 482 | "esutils": { 483 | "version": "2.0.3", 484 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", 485 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", 486 | "dev": true 487 | }, 488 | "expand-brackets": { 489 | "version": "2.1.4", 490 | "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", 491 | "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", 492 | "dev": true, 493 | "requires": { 494 | "debug": "^2.3.3", 495 | "define-property": "^0.2.5", 496 | "extend-shallow": "^2.0.1", 497 | "posix-character-classes": "^0.1.0", 498 | "regex-not": "^1.0.0", 499 | "snapdragon": "^0.8.1", 500 | "to-regex": "^3.0.1" 501 | }, 502 | "dependencies": { 503 | "define-property": { 504 | "version": "0.2.5", 505 | "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", 506 | "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", 507 | "dev": true, 508 | "requires": { 509 | "is-descriptor": "^0.1.0" 510 | } 511 | }, 512 | "extend-shallow": { 513 | "version": "2.0.1", 514 | "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", 515 | "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", 516 | "dev": true, 517 | "requires": { 518 | "is-extendable": "^0.1.0" 519 | } 520 | } 521 | } 522 | }, 523 | "extend-shallow": { 524 | "version": "3.0.2", 525 | "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", 526 | "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", 527 | "dev": true, 528 | "requires": { 529 | "assign-symbols": "^1.0.0", 530 | "is-extendable": "^1.0.1" 531 | }, 532 | "dependencies": { 533 | "is-extendable": { 534 | "version": "1.0.1", 535 | "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", 536 | "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", 537 | "dev": true, 538 | "requires": { 539 | "is-plain-object": "^2.0.4" 540 | } 541 | } 542 | } 543 | }, 544 | "extglob": { 545 | "version": "2.0.4", 546 | "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", 547 | "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", 548 | "dev": true, 549 | "requires": { 550 | "array-unique": "^0.3.2", 551 | "define-property": "^1.0.0", 552 | "expand-brackets": "^2.1.4", 553 | "extend-shallow": "^2.0.1", 554 | "fragment-cache": "^0.2.1", 555 | "regex-not": "^1.0.0", 556 | "snapdragon": "^0.8.1", 557 | "to-regex": "^3.0.1" 558 | }, 559 | "dependencies": { 560 | "define-property": { 561 | "version": "1.0.0", 562 | "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", 563 | "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", 564 | "dev": true, 565 | "requires": { 566 | "is-descriptor": "^1.0.0" 567 | } 568 | }, 569 | "extend-shallow": { 570 | "version": "2.0.1", 571 | "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", 572 | "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", 573 | "dev": true, 574 | "requires": { 575 | "is-extendable": "^0.1.0" 576 | } 577 | }, 578 | "is-accessor-descriptor": { 579 | "version": "1.0.0", 580 | "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", 581 | "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", 582 | "dev": true, 583 | "requires": { 584 | "kind-of": "^6.0.0" 585 | } 586 | }, 587 | "is-data-descriptor": { 588 | "version": "1.0.0", 589 | "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", 590 | "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", 591 | "dev": true, 592 | "requires": { 593 | "kind-of": "^6.0.0" 594 | } 595 | }, 596 | "is-descriptor": { 597 | "version": "1.0.2", 598 | "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", 599 | "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", 600 | "dev": true, 601 | "requires": { 602 | "is-accessor-descriptor": "^1.0.0", 603 | "is-data-descriptor": "^1.0.0", 604 | "kind-of": "^6.0.2" 605 | } 606 | } 607 | } 608 | }, 609 | "file-uri-to-path": { 610 | "version": "1.0.0", 611 | "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", 612 | "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", 613 | "dev": true, 614 | "optional": true 615 | }, 616 | "fill-range": { 617 | "version": "4.0.0", 618 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", 619 | "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", 620 | "dev": true, 621 | "requires": { 622 | "extend-shallow": "^2.0.1", 623 | "is-number": "^3.0.0", 624 | "repeat-string": "^1.6.1", 625 | "to-regex-range": "^2.1.0" 626 | }, 627 | "dependencies": { 628 | "extend-shallow": { 629 | "version": "2.0.1", 630 | "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", 631 | "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", 632 | "dev": true, 633 | "requires": { 634 | "is-extendable": "^0.1.0" 635 | } 636 | } 637 | } 638 | }, 639 | "for-in": { 640 | "version": "1.0.2", 641 | "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", 642 | "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", 643 | "dev": true 644 | }, 645 | "fragment-cache": { 646 | "version": "0.2.1", 647 | "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", 648 | "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=", 649 | "dev": true, 650 | "requires": { 651 | "map-cache": "^0.2.2" 652 | } 653 | }, 654 | "fsevents": { 655 | "version": "1.2.11", 656 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.11.tgz", 657 | "integrity": "sha512-+ux3lx6peh0BpvY0JebGyZoiR4D+oYzdPZMKJwkZ+sFkNJzpL7tXc/wehS49gUAxg3tmMHPHZkA8JU2rhhgDHw==", 658 | "dev": true, 659 | "optional": true, 660 | "requires": { 661 | "bindings": "^1.5.0", 662 | "nan": "^2.12.1", 663 | "node-pre-gyp": "*" 664 | }, 665 | "dependencies": { 666 | "abbrev": { 667 | "version": "1.1.1", 668 | "bundled": true, 669 | "dev": true, 670 | "optional": true 671 | }, 672 | "ansi-regex": { 673 | "version": "2.1.1", 674 | "bundled": true, 675 | "dev": true, 676 | "optional": true 677 | }, 678 | "aproba": { 679 | "version": "1.2.0", 680 | "bundled": true, 681 | "dev": true, 682 | "optional": true 683 | }, 684 | "are-we-there-yet": { 685 | "version": "1.1.5", 686 | "bundled": true, 687 | "dev": true, 688 | "optional": true, 689 | "requires": { 690 | "delegates": "^1.0.0", 691 | "readable-stream": "^2.0.6" 692 | } 693 | }, 694 | "balanced-match": { 695 | "version": "1.0.0", 696 | "bundled": true, 697 | "dev": true, 698 | "optional": true 699 | }, 700 | "brace-expansion": { 701 | "version": "1.1.11", 702 | "bundled": true, 703 | "dev": true, 704 | "optional": true, 705 | "requires": { 706 | "balanced-match": "^1.0.0", 707 | "concat-map": "0.0.1" 708 | } 709 | }, 710 | "chownr": { 711 | "version": "1.1.3", 712 | "bundled": true, 713 | "dev": true, 714 | "optional": true 715 | }, 716 | "code-point-at": { 717 | "version": "1.1.0", 718 | "bundled": true, 719 | "dev": true, 720 | "optional": true 721 | }, 722 | "concat-map": { 723 | "version": "0.0.1", 724 | "bundled": true, 725 | "dev": true, 726 | "optional": true 727 | }, 728 | "console-control-strings": { 729 | "version": "1.1.0", 730 | "bundled": true, 731 | "dev": true, 732 | "optional": true 733 | }, 734 | "core-util-is": { 735 | "version": "1.0.2", 736 | "bundled": true, 737 | "dev": true, 738 | "optional": true 739 | }, 740 | "debug": { 741 | "version": "3.2.6", 742 | "bundled": true, 743 | "dev": true, 744 | "optional": true, 745 | "requires": { 746 | "ms": "^2.1.1" 747 | } 748 | }, 749 | "deep-extend": { 750 | "version": "0.6.0", 751 | "bundled": true, 752 | "dev": true, 753 | "optional": true 754 | }, 755 | "delegates": { 756 | "version": "1.0.0", 757 | "bundled": true, 758 | "dev": true, 759 | "optional": true 760 | }, 761 | "detect-libc": { 762 | "version": "1.0.3", 763 | "bundled": true, 764 | "dev": true, 765 | "optional": true 766 | }, 767 | "fs-minipass": { 768 | "version": "1.2.7", 769 | "bundled": true, 770 | "dev": true, 771 | "optional": true, 772 | "requires": { 773 | "minipass": "^2.6.0" 774 | } 775 | }, 776 | "fs.realpath": { 777 | "version": "1.0.0", 778 | "bundled": true, 779 | "dev": true, 780 | "optional": true 781 | }, 782 | "gauge": { 783 | "version": "2.7.4", 784 | "bundled": true, 785 | "dev": true, 786 | "optional": true, 787 | "requires": { 788 | "aproba": "^1.0.3", 789 | "console-control-strings": "^1.0.0", 790 | "has-unicode": "^2.0.0", 791 | "object-assign": "^4.1.0", 792 | "signal-exit": "^3.0.0", 793 | "string-width": "^1.0.1", 794 | "strip-ansi": "^3.0.1", 795 | "wide-align": "^1.1.0" 796 | } 797 | }, 798 | "glob": { 799 | "version": "7.1.6", 800 | "bundled": true, 801 | "dev": true, 802 | "optional": true, 803 | "requires": { 804 | "fs.realpath": "^1.0.0", 805 | "inflight": "^1.0.4", 806 | "inherits": "2", 807 | "minimatch": "^3.0.4", 808 | "once": "^1.3.0", 809 | "path-is-absolute": "^1.0.0" 810 | } 811 | }, 812 | "has-unicode": { 813 | "version": "2.0.1", 814 | "bundled": true, 815 | "dev": true, 816 | "optional": true 817 | }, 818 | "iconv-lite": { 819 | "version": "0.4.24", 820 | "bundled": true, 821 | "dev": true, 822 | "optional": true, 823 | "requires": { 824 | "safer-buffer": ">= 2.1.2 < 3" 825 | } 826 | }, 827 | "ignore-walk": { 828 | "version": "3.0.3", 829 | "bundled": true, 830 | "dev": true, 831 | "optional": true, 832 | "requires": { 833 | "minimatch": "^3.0.4" 834 | } 835 | }, 836 | "inflight": { 837 | "version": "1.0.6", 838 | "bundled": true, 839 | "dev": true, 840 | "optional": true, 841 | "requires": { 842 | "once": "^1.3.0", 843 | "wrappy": "1" 844 | } 845 | }, 846 | "inherits": { 847 | "version": "2.0.4", 848 | "bundled": true, 849 | "dev": true, 850 | "optional": true 851 | }, 852 | "ini": { 853 | "version": "1.3.5", 854 | "bundled": true, 855 | "dev": true, 856 | "optional": true 857 | }, 858 | "is-fullwidth-code-point": { 859 | "version": "1.0.0", 860 | "bundled": true, 861 | "dev": true, 862 | "optional": true, 863 | "requires": { 864 | "number-is-nan": "^1.0.0" 865 | } 866 | }, 867 | "isarray": { 868 | "version": "1.0.0", 869 | "bundled": true, 870 | "dev": true, 871 | "optional": true 872 | }, 873 | "minimatch": { 874 | "version": "3.0.4", 875 | "bundled": true, 876 | "dev": true, 877 | "optional": true, 878 | "requires": { 879 | "brace-expansion": "^1.1.7" 880 | } 881 | }, 882 | "minimist": { 883 | "version": "0.0.8", 884 | "bundled": true, 885 | "dev": true, 886 | "optional": true 887 | }, 888 | "minipass": { 889 | "version": "2.9.0", 890 | "bundled": true, 891 | "dev": true, 892 | "optional": true, 893 | "requires": { 894 | "safe-buffer": "^5.1.2", 895 | "yallist": "^3.0.0" 896 | } 897 | }, 898 | "minizlib": { 899 | "version": "1.3.3", 900 | "bundled": true, 901 | "dev": true, 902 | "optional": true, 903 | "requires": { 904 | "minipass": "^2.9.0" 905 | } 906 | }, 907 | "mkdirp": { 908 | "version": "0.5.1", 909 | "bundled": true, 910 | "dev": true, 911 | "optional": true, 912 | "requires": { 913 | "minimist": "0.0.8" 914 | } 915 | }, 916 | "ms": { 917 | "version": "2.1.2", 918 | "bundled": true, 919 | "dev": true, 920 | "optional": true 921 | }, 922 | "needle": { 923 | "version": "2.4.0", 924 | "bundled": true, 925 | "dev": true, 926 | "optional": true, 927 | "requires": { 928 | "debug": "^3.2.6", 929 | "iconv-lite": "^0.4.4", 930 | "sax": "^1.2.4" 931 | } 932 | }, 933 | "node-pre-gyp": { 934 | "version": "0.14.0", 935 | "bundled": true, 936 | "dev": true, 937 | "optional": true, 938 | "requires": { 939 | "detect-libc": "^1.0.2", 940 | "mkdirp": "^0.5.1", 941 | "needle": "^2.2.1", 942 | "nopt": "^4.0.1", 943 | "npm-packlist": "^1.1.6", 944 | "npmlog": "^4.0.2", 945 | "rc": "^1.2.7", 946 | "rimraf": "^2.6.1", 947 | "semver": "^5.3.0", 948 | "tar": "^4.4.2" 949 | } 950 | }, 951 | "nopt": { 952 | "version": "4.0.1", 953 | "bundled": true, 954 | "dev": true, 955 | "optional": true, 956 | "requires": { 957 | "abbrev": "1", 958 | "osenv": "^0.1.4" 959 | } 960 | }, 961 | "npm-bundled": { 962 | "version": "1.1.1", 963 | "bundled": true, 964 | "dev": true, 965 | "optional": true, 966 | "requires": { 967 | "npm-normalize-package-bin": "^1.0.1" 968 | } 969 | }, 970 | "npm-normalize-package-bin": { 971 | "version": "1.0.1", 972 | "bundled": true, 973 | "dev": true, 974 | "optional": true 975 | }, 976 | "npm-packlist": { 977 | "version": "1.4.7", 978 | "bundled": true, 979 | "dev": true, 980 | "optional": true, 981 | "requires": { 982 | "ignore-walk": "^3.0.1", 983 | "npm-bundled": "^1.0.1" 984 | } 985 | }, 986 | "npmlog": { 987 | "version": "4.1.2", 988 | "bundled": true, 989 | "dev": true, 990 | "optional": true, 991 | "requires": { 992 | "are-we-there-yet": "~1.1.2", 993 | "console-control-strings": "~1.1.0", 994 | "gauge": "~2.7.3", 995 | "set-blocking": "~2.0.0" 996 | } 997 | }, 998 | "number-is-nan": { 999 | "version": "1.0.1", 1000 | "bundled": true, 1001 | "dev": true, 1002 | "optional": true 1003 | }, 1004 | "object-assign": { 1005 | "version": "4.1.1", 1006 | "bundled": true, 1007 | "dev": true, 1008 | "optional": true 1009 | }, 1010 | "once": { 1011 | "version": "1.4.0", 1012 | "bundled": true, 1013 | "dev": true, 1014 | "optional": true, 1015 | "requires": { 1016 | "wrappy": "1" 1017 | } 1018 | }, 1019 | "os-homedir": { 1020 | "version": "1.0.2", 1021 | "bundled": true, 1022 | "dev": true, 1023 | "optional": true 1024 | }, 1025 | "os-tmpdir": { 1026 | "version": "1.0.2", 1027 | "bundled": true, 1028 | "dev": true, 1029 | "optional": true 1030 | }, 1031 | "osenv": { 1032 | "version": "0.1.5", 1033 | "bundled": true, 1034 | "dev": true, 1035 | "optional": true, 1036 | "requires": { 1037 | "os-homedir": "^1.0.0", 1038 | "os-tmpdir": "^1.0.0" 1039 | } 1040 | }, 1041 | "path-is-absolute": { 1042 | "version": "1.0.1", 1043 | "bundled": true, 1044 | "dev": true, 1045 | "optional": true 1046 | }, 1047 | "process-nextick-args": { 1048 | "version": "2.0.1", 1049 | "bundled": true, 1050 | "dev": true, 1051 | "optional": true 1052 | }, 1053 | "rc": { 1054 | "version": "1.2.8", 1055 | "bundled": true, 1056 | "dev": true, 1057 | "optional": true, 1058 | "requires": { 1059 | "deep-extend": "^0.6.0", 1060 | "ini": "~1.3.0", 1061 | "minimist": "^1.2.0", 1062 | "strip-json-comments": "~2.0.1" 1063 | }, 1064 | "dependencies": { 1065 | "minimist": { 1066 | "version": "1.2.0", 1067 | "bundled": true, 1068 | "dev": true, 1069 | "optional": true 1070 | } 1071 | } 1072 | }, 1073 | "readable-stream": { 1074 | "version": "2.3.6", 1075 | "bundled": true, 1076 | "dev": true, 1077 | "optional": true, 1078 | "requires": { 1079 | "core-util-is": "~1.0.0", 1080 | "inherits": "~2.0.3", 1081 | "isarray": "~1.0.0", 1082 | "process-nextick-args": "~2.0.0", 1083 | "safe-buffer": "~5.1.1", 1084 | "string_decoder": "~1.1.1", 1085 | "util-deprecate": "~1.0.1" 1086 | } 1087 | }, 1088 | "rimraf": { 1089 | "version": "2.7.1", 1090 | "bundled": true, 1091 | "dev": true, 1092 | "optional": true, 1093 | "requires": { 1094 | "glob": "^7.1.3" 1095 | } 1096 | }, 1097 | "safe-buffer": { 1098 | "version": "5.1.2", 1099 | "bundled": true, 1100 | "dev": true, 1101 | "optional": true 1102 | }, 1103 | "safer-buffer": { 1104 | "version": "2.1.2", 1105 | "bundled": true, 1106 | "dev": true, 1107 | "optional": true 1108 | }, 1109 | "sax": { 1110 | "version": "1.2.4", 1111 | "bundled": true, 1112 | "dev": true, 1113 | "optional": true 1114 | }, 1115 | "semver": { 1116 | "version": "5.7.1", 1117 | "bundled": true, 1118 | "dev": true, 1119 | "optional": true 1120 | }, 1121 | "set-blocking": { 1122 | "version": "2.0.0", 1123 | "bundled": true, 1124 | "dev": true, 1125 | "optional": true 1126 | }, 1127 | "signal-exit": { 1128 | "version": "3.0.2", 1129 | "bundled": true, 1130 | "dev": true, 1131 | "optional": true 1132 | }, 1133 | "string-width": { 1134 | "version": "1.0.2", 1135 | "bundled": true, 1136 | "dev": true, 1137 | "optional": true, 1138 | "requires": { 1139 | "code-point-at": "^1.0.0", 1140 | "is-fullwidth-code-point": "^1.0.0", 1141 | "strip-ansi": "^3.0.0" 1142 | } 1143 | }, 1144 | "string_decoder": { 1145 | "version": "1.1.1", 1146 | "bundled": true, 1147 | "dev": true, 1148 | "optional": true, 1149 | "requires": { 1150 | "safe-buffer": "~5.1.0" 1151 | } 1152 | }, 1153 | "strip-ansi": { 1154 | "version": "3.0.1", 1155 | "bundled": true, 1156 | "dev": true, 1157 | "optional": true, 1158 | "requires": { 1159 | "ansi-regex": "^2.0.0" 1160 | } 1161 | }, 1162 | "strip-json-comments": { 1163 | "version": "2.0.1", 1164 | "bundled": true, 1165 | "dev": true, 1166 | "optional": true 1167 | }, 1168 | "tar": { 1169 | "version": "4.4.13", 1170 | "bundled": true, 1171 | "dev": true, 1172 | "optional": true, 1173 | "requires": { 1174 | "chownr": "^1.1.1", 1175 | "fs-minipass": "^1.2.5", 1176 | "minipass": "^2.8.6", 1177 | "minizlib": "^1.2.1", 1178 | "mkdirp": "^0.5.0", 1179 | "safe-buffer": "^5.1.2", 1180 | "yallist": "^3.0.3" 1181 | } 1182 | }, 1183 | "util-deprecate": { 1184 | "version": "1.0.2", 1185 | "bundled": true, 1186 | "dev": true, 1187 | "optional": true 1188 | }, 1189 | "wide-align": { 1190 | "version": "1.1.3", 1191 | "bundled": true, 1192 | "dev": true, 1193 | "optional": true, 1194 | "requires": { 1195 | "string-width": "^1.0.2 || 2" 1196 | } 1197 | }, 1198 | "wrappy": { 1199 | "version": "1.0.2", 1200 | "bundled": true, 1201 | "dev": true, 1202 | "optional": true 1203 | }, 1204 | "yallist": { 1205 | "version": "3.1.1", 1206 | "bundled": true, 1207 | "dev": true, 1208 | "optional": true 1209 | } 1210 | } 1211 | }, 1212 | "get-port": { 1213 | "version": "3.2.0", 1214 | "resolved": "https://registry.npmjs.org/get-port/-/get-port-3.2.0.tgz", 1215 | "integrity": "sha1-3Xzn3hh8Bsi/NTeWrHHgmfCYDrw=" 1216 | }, 1217 | "get-value": { 1218 | "version": "2.0.6", 1219 | "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", 1220 | "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=", 1221 | "dev": true 1222 | }, 1223 | "glob-parent": { 1224 | "version": "3.1.0", 1225 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", 1226 | "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", 1227 | "dev": true, 1228 | "requires": { 1229 | "is-glob": "^3.1.0", 1230 | "path-dirname": "^1.0.0" 1231 | }, 1232 | "dependencies": { 1233 | "is-glob": { 1234 | "version": "3.1.0", 1235 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", 1236 | "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", 1237 | "dev": true, 1238 | "requires": { 1239 | "is-extglob": "^2.1.0" 1240 | } 1241 | } 1242 | } 1243 | }, 1244 | "graceful-fs": { 1245 | "version": "4.2.3", 1246 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.3.tgz", 1247 | "integrity": "sha512-a30VEBm4PEdx1dRB7MFK7BejejvCvBronbLjht+sHuGYj8PHs7M/5Z+rt5lw551vZ7yfTCj4Vuyy3mSJytDWRQ==", 1248 | "dev": true 1249 | }, 1250 | "has-flag": { 1251 | "version": "3.0.0", 1252 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 1253 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 1254 | "dev": true 1255 | }, 1256 | "has-value": { 1257 | "version": "1.0.0", 1258 | "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz", 1259 | "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=", 1260 | "dev": true, 1261 | "requires": { 1262 | "get-value": "^2.0.6", 1263 | "has-values": "^1.0.0", 1264 | "isobject": "^3.0.0" 1265 | } 1266 | }, 1267 | "has-values": { 1268 | "version": "1.0.0", 1269 | "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz", 1270 | "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=", 1271 | "dev": true, 1272 | "requires": { 1273 | "is-number": "^3.0.0", 1274 | "kind-of": "^4.0.0" 1275 | }, 1276 | "dependencies": { 1277 | "kind-of": { 1278 | "version": "4.0.0", 1279 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", 1280 | "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", 1281 | "dev": true, 1282 | "requires": { 1283 | "is-buffer": "^1.1.5" 1284 | } 1285 | } 1286 | } 1287 | }, 1288 | "inherits": { 1289 | "version": "2.0.4", 1290 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 1291 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 1292 | "dev": true 1293 | }, 1294 | "is-accessor-descriptor": { 1295 | "version": "0.1.6", 1296 | "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", 1297 | "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", 1298 | "dev": true, 1299 | "requires": { 1300 | "kind-of": "^3.0.2" 1301 | }, 1302 | "dependencies": { 1303 | "kind-of": { 1304 | "version": "3.2.2", 1305 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", 1306 | "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", 1307 | "dev": true, 1308 | "requires": { 1309 | "is-buffer": "^1.1.5" 1310 | } 1311 | } 1312 | } 1313 | }, 1314 | "is-binary-path": { 1315 | "version": "1.0.1", 1316 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", 1317 | "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", 1318 | "dev": true, 1319 | "requires": { 1320 | "binary-extensions": "^1.0.0" 1321 | } 1322 | }, 1323 | "is-buffer": { 1324 | "version": "1.1.6", 1325 | "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", 1326 | "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", 1327 | "dev": true 1328 | }, 1329 | "is-data-descriptor": { 1330 | "version": "0.1.4", 1331 | "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", 1332 | "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", 1333 | "dev": true, 1334 | "requires": { 1335 | "kind-of": "^3.0.2" 1336 | }, 1337 | "dependencies": { 1338 | "kind-of": { 1339 | "version": "3.2.2", 1340 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", 1341 | "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", 1342 | "dev": true, 1343 | "requires": { 1344 | "is-buffer": "^1.1.5" 1345 | } 1346 | } 1347 | } 1348 | }, 1349 | "is-descriptor": { 1350 | "version": "0.1.6", 1351 | "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", 1352 | "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", 1353 | "dev": true, 1354 | "requires": { 1355 | "is-accessor-descriptor": "^0.1.6", 1356 | "is-data-descriptor": "^0.1.4", 1357 | "kind-of": "^5.0.0" 1358 | }, 1359 | "dependencies": { 1360 | "kind-of": { 1361 | "version": "5.1.0", 1362 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", 1363 | "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", 1364 | "dev": true 1365 | } 1366 | } 1367 | }, 1368 | "is-extendable": { 1369 | "version": "0.1.1", 1370 | "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", 1371 | "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=", 1372 | "dev": true 1373 | }, 1374 | "is-extglob": { 1375 | "version": "2.1.1", 1376 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 1377 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 1378 | "dev": true 1379 | }, 1380 | "is-glob": { 1381 | "version": "4.0.1", 1382 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 1383 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 1384 | "dev": true, 1385 | "requires": { 1386 | "is-extglob": "^2.1.1" 1387 | } 1388 | }, 1389 | "is-module": { 1390 | "version": "1.0.0", 1391 | "resolved": "https://registry.npmjs.org/is-module/-/is-module-1.0.0.tgz", 1392 | "integrity": "sha1-Mlj7afeMFNW4FdZkM2tM/7ZEFZE=", 1393 | "dev": true 1394 | }, 1395 | "is-number": { 1396 | "version": "3.0.0", 1397 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", 1398 | "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", 1399 | "dev": true, 1400 | "requires": { 1401 | "kind-of": "^3.0.2" 1402 | }, 1403 | "dependencies": { 1404 | "kind-of": { 1405 | "version": "3.2.2", 1406 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", 1407 | "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", 1408 | "dev": true, 1409 | "requires": { 1410 | "is-buffer": "^1.1.5" 1411 | } 1412 | } 1413 | } 1414 | }, 1415 | "is-plain-object": { 1416 | "version": "2.0.4", 1417 | "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", 1418 | "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", 1419 | "dev": true, 1420 | "requires": { 1421 | "isobject": "^3.0.1" 1422 | } 1423 | }, 1424 | "is-reference": { 1425 | "version": "1.1.4", 1426 | "resolved": "https://registry.npmjs.org/is-reference/-/is-reference-1.1.4.tgz", 1427 | "integrity": "sha512-uJA/CDPO3Tao3GTrxYn6AwkM4nUPJiGGYu5+cB8qbC7WGFlrKZbiRo7SFKxUAEpFUfiHofWCXBUNhvYJMh+6zw==", 1428 | "dev": true, 1429 | "requires": { 1430 | "@types/estree": "0.0.39" 1431 | } 1432 | }, 1433 | "is-windows": { 1434 | "version": "1.0.2", 1435 | "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", 1436 | "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", 1437 | "dev": true 1438 | }, 1439 | "isarray": { 1440 | "version": "1.0.0", 1441 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 1442 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", 1443 | "dev": true 1444 | }, 1445 | "isobject": { 1446 | "version": "3.0.1", 1447 | "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", 1448 | "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", 1449 | "dev": true 1450 | }, 1451 | "jest-worker": { 1452 | "version": "24.9.0", 1453 | "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-24.9.0.tgz", 1454 | "integrity": "sha512-51PE4haMSXcHohnSMdM42anbvZANYTqMrr52tVKPqqsPJMzoP6FYYDVqahX/HrAoKEKz3uUPzSvKs9A3qR4iVw==", 1455 | "dev": true, 1456 | "requires": { 1457 | "merge-stream": "^2.0.0", 1458 | "supports-color": "^6.1.0" 1459 | }, 1460 | "dependencies": { 1461 | "supports-color": { 1462 | "version": "6.1.0", 1463 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", 1464 | "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", 1465 | "dev": true, 1466 | "requires": { 1467 | "has-flag": "^3.0.0" 1468 | } 1469 | } 1470 | } 1471 | }, 1472 | "js-tokens": { 1473 | "version": "4.0.0", 1474 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 1475 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 1476 | "dev": true 1477 | }, 1478 | "kind-of": { 1479 | "version": "6.0.3", 1480 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", 1481 | "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", 1482 | "dev": true 1483 | }, 1484 | "kleur": { 1485 | "version": "3.0.3", 1486 | "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", 1487 | "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==" 1488 | }, 1489 | "livereload": { 1490 | "version": "0.8.2", 1491 | "resolved": "https://registry.npmjs.org/livereload/-/livereload-0.8.2.tgz", 1492 | "integrity": "sha512-8wCvhiCL4cGVoT3U5xoe+UjpiiVZLrlOvr6dbhb1VlyC5QarhrlyRRt4z7EMGO4KSgXj+tKF/dr284F28/wI+g==", 1493 | "dev": true, 1494 | "requires": { 1495 | "chokidar": "^2.1.5", 1496 | "opts": ">= 1.2.0", 1497 | "ws": "^6.2.1" 1498 | } 1499 | }, 1500 | "local-access": { 1501 | "version": "1.0.1", 1502 | "resolved": "https://registry.npmjs.org/local-access/-/local-access-1.0.1.tgz", 1503 | "integrity": "sha512-ykt2pgN0aqIy6KQC1CqdWTWkmUwNgaOS6dcpHVjyBJONA+Xi7AtSB1vuxC/U/0tjIP3wcRudwQk1YYzUvzk2bA==" 1504 | }, 1505 | "magic-string": { 1506 | "version": "0.25.6", 1507 | "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.6.tgz", 1508 | "integrity": "sha512-3a5LOMSGoCTH5rbqobC2HuDNRtE2glHZ8J7pK+QZYppyWA36yuNpsX994rIY2nCuyP7CZYy7lQq/X2jygiZ89g==", 1509 | "dev": true, 1510 | "requires": { 1511 | "sourcemap-codec": "^1.4.4" 1512 | } 1513 | }, 1514 | "map-cache": { 1515 | "version": "0.2.2", 1516 | "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", 1517 | "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=", 1518 | "dev": true 1519 | }, 1520 | "map-visit": { 1521 | "version": "1.0.0", 1522 | "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz", 1523 | "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=", 1524 | "dev": true, 1525 | "requires": { 1526 | "object-visit": "^1.0.0" 1527 | } 1528 | }, 1529 | "merge-stream": { 1530 | "version": "2.0.0", 1531 | "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", 1532 | "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", 1533 | "dev": true 1534 | }, 1535 | "micromatch": { 1536 | "version": "3.1.10", 1537 | "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", 1538 | "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", 1539 | "dev": true, 1540 | "requires": { 1541 | "arr-diff": "^4.0.0", 1542 | "array-unique": "^0.3.2", 1543 | "braces": "^2.3.1", 1544 | "define-property": "^2.0.2", 1545 | "extend-shallow": "^3.0.2", 1546 | "extglob": "^2.0.4", 1547 | "fragment-cache": "^0.2.1", 1548 | "kind-of": "^6.0.2", 1549 | "nanomatch": "^1.2.9", 1550 | "object.pick": "^1.3.0", 1551 | "regex-not": "^1.0.0", 1552 | "snapdragon": "^0.8.1", 1553 | "to-regex": "^3.0.2" 1554 | } 1555 | }, 1556 | "mime": { 1557 | "version": "2.4.4", 1558 | "resolved": "https://registry.npmjs.org/mime/-/mime-2.4.4.tgz", 1559 | "integrity": "sha512-LRxmNwziLPT828z+4YkNzloCFC2YM4wrB99k+AV5ZbEyfGNWfG8SO1FUXLmLDBSo89NrJZ4DIWeLjy1CHGhMGA==" 1560 | }, 1561 | "mixin-deep": { 1562 | "version": "1.3.2", 1563 | "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.2.tgz", 1564 | "integrity": "sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==", 1565 | "dev": true, 1566 | "requires": { 1567 | "for-in": "^1.0.2", 1568 | "is-extendable": "^1.0.1" 1569 | }, 1570 | "dependencies": { 1571 | "is-extendable": { 1572 | "version": "1.0.1", 1573 | "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", 1574 | "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", 1575 | "dev": true, 1576 | "requires": { 1577 | "is-plain-object": "^2.0.4" 1578 | } 1579 | } 1580 | } 1581 | }, 1582 | "mri": { 1583 | "version": "1.1.4", 1584 | "resolved": "https://registry.npmjs.org/mri/-/mri-1.1.4.tgz", 1585 | "integrity": "sha512-6y7IjGPm8AzlvoUrwAaw1tLnUBudaS3752vcd8JtrpGGQn+rXIe63LFVHm/YMwtqAuh+LJPCFdlLYPWM1nYn6w==" 1586 | }, 1587 | "ms": { 1588 | "version": "2.0.0", 1589 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1590 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", 1591 | "dev": true 1592 | }, 1593 | "nan": { 1594 | "version": "2.14.0", 1595 | "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.0.tgz", 1596 | "integrity": "sha512-INOFj37C7k3AfaNTtX8RhsTw7qRy7eLET14cROi9+5HAVbbHuIWUHEauBv5qT4Av2tWasiTY1Jw6puUNqRJXQg==", 1597 | "dev": true, 1598 | "optional": true 1599 | }, 1600 | "nanomatch": { 1601 | "version": "1.2.13", 1602 | "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", 1603 | "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", 1604 | "dev": true, 1605 | "requires": { 1606 | "arr-diff": "^4.0.0", 1607 | "array-unique": "^0.3.2", 1608 | "define-property": "^2.0.2", 1609 | "extend-shallow": "^3.0.2", 1610 | "fragment-cache": "^0.2.1", 1611 | "is-windows": "^1.0.2", 1612 | "kind-of": "^6.0.2", 1613 | "object.pick": "^1.3.0", 1614 | "regex-not": "^1.0.0", 1615 | "snapdragon": "^0.8.1", 1616 | "to-regex": "^3.0.1" 1617 | } 1618 | }, 1619 | "normalize-path": { 1620 | "version": "3.0.0", 1621 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1622 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 1623 | "dev": true 1624 | }, 1625 | "object-copy": { 1626 | "version": "0.1.0", 1627 | "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", 1628 | "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=", 1629 | "dev": true, 1630 | "requires": { 1631 | "copy-descriptor": "^0.1.0", 1632 | "define-property": "^0.2.5", 1633 | "kind-of": "^3.0.3" 1634 | }, 1635 | "dependencies": { 1636 | "define-property": { 1637 | "version": "0.2.5", 1638 | "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", 1639 | "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", 1640 | "dev": true, 1641 | "requires": { 1642 | "is-descriptor": "^0.1.0" 1643 | } 1644 | }, 1645 | "kind-of": { 1646 | "version": "3.2.2", 1647 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", 1648 | "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", 1649 | "dev": true, 1650 | "requires": { 1651 | "is-buffer": "^1.1.5" 1652 | } 1653 | } 1654 | } 1655 | }, 1656 | "object-visit": { 1657 | "version": "1.0.1", 1658 | "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz", 1659 | "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=", 1660 | "dev": true, 1661 | "requires": { 1662 | "isobject": "^3.0.0" 1663 | } 1664 | }, 1665 | "object.pick": { 1666 | "version": "1.3.0", 1667 | "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", 1668 | "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=", 1669 | "dev": true, 1670 | "requires": { 1671 | "isobject": "^3.0.1" 1672 | } 1673 | }, 1674 | "opts": { 1675 | "version": "1.2.7", 1676 | "resolved": "https://registry.npmjs.org/opts/-/opts-1.2.7.tgz", 1677 | "integrity": "sha512-hwZhzGGG/GQ7igxAVFOEun2N4fWul31qE9nfBdCnZGQCB5+L7tN9xZ+94B4aUpLOJx/of3zZs5XsuubayQYQjA==", 1678 | "dev": true 1679 | }, 1680 | "pascalcase": { 1681 | "version": "0.1.1", 1682 | "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", 1683 | "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", 1684 | "dev": true 1685 | }, 1686 | "path-dirname": { 1687 | "version": "1.0.2", 1688 | "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", 1689 | "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", 1690 | "dev": true 1691 | }, 1692 | "path-is-absolute": { 1693 | "version": "1.0.1", 1694 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1695 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 1696 | "dev": true 1697 | }, 1698 | "path-parse": { 1699 | "version": "1.0.6", 1700 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", 1701 | "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", 1702 | "dev": true 1703 | }, 1704 | "posix-character-classes": { 1705 | "version": "0.1.1", 1706 | "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz", 1707 | "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=", 1708 | "dev": true 1709 | }, 1710 | "process-nextick-args": { 1711 | "version": "2.0.1", 1712 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 1713 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", 1714 | "dev": true 1715 | }, 1716 | "readable-stream": { 1717 | "version": "2.3.7", 1718 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", 1719 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", 1720 | "dev": true, 1721 | "requires": { 1722 | "core-util-is": "~1.0.0", 1723 | "inherits": "~2.0.3", 1724 | "isarray": "~1.0.0", 1725 | "process-nextick-args": "~2.0.0", 1726 | "safe-buffer": "~5.1.1", 1727 | "string_decoder": "~1.1.1", 1728 | "util-deprecate": "~1.0.1" 1729 | } 1730 | }, 1731 | "readdirp": { 1732 | "version": "2.2.1", 1733 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", 1734 | "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", 1735 | "dev": true, 1736 | "requires": { 1737 | "graceful-fs": "^4.1.11", 1738 | "micromatch": "^3.1.10", 1739 | "readable-stream": "^2.0.2" 1740 | } 1741 | }, 1742 | "regex-not": { 1743 | "version": "1.0.2", 1744 | "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.2.tgz", 1745 | "integrity": "sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==", 1746 | "dev": true, 1747 | "requires": { 1748 | "extend-shallow": "^3.0.2", 1749 | "safe-regex": "^1.1.0" 1750 | } 1751 | }, 1752 | "remove-trailing-separator": { 1753 | "version": "1.1.0", 1754 | "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", 1755 | "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", 1756 | "dev": true 1757 | }, 1758 | "repeat-element": { 1759 | "version": "1.1.3", 1760 | "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.3.tgz", 1761 | "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==", 1762 | "dev": true 1763 | }, 1764 | "repeat-string": { 1765 | "version": "1.6.1", 1766 | "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", 1767 | "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=", 1768 | "dev": true 1769 | }, 1770 | "require-relative": { 1771 | "version": "0.8.7", 1772 | "resolved": "https://registry.npmjs.org/require-relative/-/require-relative-0.8.7.tgz", 1773 | "integrity": "sha1-eZlTn8ngR6N5KPoZb44VY9q9Nt4=", 1774 | "dev": true 1775 | }, 1776 | "resolve": { 1777 | "version": "1.15.1", 1778 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.15.1.tgz", 1779 | "integrity": "sha512-84oo6ZTtoTUpjgNEr5SJyzQhzL72gaRodsSfyxC/AXRvwu0Yse9H8eF9IpGo7b8YetZhlI6v7ZQ6bKBFV/6S7w==", 1780 | "dev": true, 1781 | "requires": { 1782 | "path-parse": "^1.0.6" 1783 | } 1784 | }, 1785 | "resolve-url": { 1786 | "version": "0.2.1", 1787 | "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", 1788 | "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=", 1789 | "dev": true 1790 | }, 1791 | "ret": { 1792 | "version": "0.1.15", 1793 | "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", 1794 | "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==", 1795 | "dev": true 1796 | }, 1797 | "rollup": { 1798 | "version": "1.31.0", 1799 | "resolved": "https://registry.npmjs.org/rollup/-/rollup-1.31.0.tgz", 1800 | "integrity": "sha512-9C6ovSyNeEwvuRuUUmsTpJcXac1AwSL1a3x+O5lpmQKZqi5mmrjauLeqIjvREC+yNRR8fPdzByojDng+af3nVw==", 1801 | "dev": true, 1802 | "requires": { 1803 | "@types/estree": "*", 1804 | "@types/node": "*", 1805 | "acorn": "^7.1.0" 1806 | } 1807 | }, 1808 | "rollup-plugin-livereload": { 1809 | "version": "1.0.4", 1810 | "resolved": "https://registry.npmjs.org/rollup-plugin-livereload/-/rollup-plugin-livereload-1.0.4.tgz", 1811 | "integrity": "sha512-nbnSP8Mj2mmLZkrf080z3PrdacmpAW6UkmgM+BWClcJ8MSsruPONGTwirhZaNNHjUYvkJ+iF5/pSk4g0KV2uVQ==", 1812 | "dev": true, 1813 | "requires": { 1814 | "livereload": "0.8.0 || ^0.8.2" 1815 | } 1816 | }, 1817 | "rollup-plugin-svelte": { 1818 | "version": "5.1.1", 1819 | "resolved": "https://registry.npmjs.org/rollup-plugin-svelte/-/rollup-plugin-svelte-5.1.1.tgz", 1820 | "integrity": "sha512-wP3CnKHjR4fZUgNm5Iey7eItnxwnH/nAw568WJ8dpMSchBxxZ/DmKSx8e6h8k/B6SwG1wfGvWehadFJHcuFFSw==", 1821 | "dev": true, 1822 | "requires": { 1823 | "require-relative": "^0.8.7", 1824 | "rollup-pluginutils": "^2.3.3", 1825 | "sourcemap-codec": "^1.4.4" 1826 | } 1827 | }, 1828 | "rollup-plugin-terser": { 1829 | "version": "5.2.0", 1830 | "resolved": "https://registry.npmjs.org/rollup-plugin-terser/-/rollup-plugin-terser-5.2.0.tgz", 1831 | "integrity": "sha512-jQI+nYhtDBc9HFRBz8iGttQg7li9klmzR62RG2W2nN6hJ/FI2K2ItYQ7kJ7/zn+vs+BP1AEccmVRjRN989I+Nw==", 1832 | "dev": true, 1833 | "requires": { 1834 | "@babel/code-frame": "^7.5.5", 1835 | "jest-worker": "^24.9.0", 1836 | "rollup-pluginutils": "^2.8.2", 1837 | "serialize-javascript": "^2.1.2", 1838 | "terser": "^4.6.2" 1839 | } 1840 | }, 1841 | "rollup-pluginutils": { 1842 | "version": "2.8.2", 1843 | "resolved": "https://registry.npmjs.org/rollup-pluginutils/-/rollup-pluginutils-2.8.2.tgz", 1844 | "integrity": "sha512-EEp9NhnUkwY8aif6bxgovPHMoMoNr2FulJziTndpt5H9RdwC47GSGuII9XxpSdzVGM0GWrNPHV6ie1LTNJPaLQ==", 1845 | "dev": true, 1846 | "requires": { 1847 | "estree-walker": "^0.6.1" 1848 | }, 1849 | "dependencies": { 1850 | "estree-walker": { 1851 | "version": "0.6.1", 1852 | "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-0.6.1.tgz", 1853 | "integrity": "sha512-SqmZANLWS0mnatqbSfRP5g8OXZC12Fgg1IwNtLsyHDzJizORW4khDfjPqJZsemPWBB2uqykUah5YpQ6epsqC/w==", 1854 | "dev": true 1855 | } 1856 | } 1857 | }, 1858 | "sade": { 1859 | "version": "1.7.0", 1860 | "resolved": "https://registry.npmjs.org/sade/-/sade-1.7.0.tgz", 1861 | "integrity": "sha512-HSkPpZzN7q4EFN5PVW8nTfDn1rJZh4sKbPQqz33AXokIo6SMDeVJ3RA4e0ZASlnMK6PywEMZxKXudEn5dxSWew==", 1862 | "requires": { 1863 | "mri": "^1.1.0" 1864 | } 1865 | }, 1866 | "safe-buffer": { 1867 | "version": "5.1.2", 1868 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1869 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", 1870 | "dev": true 1871 | }, 1872 | "safe-regex": { 1873 | "version": "1.1.0", 1874 | "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", 1875 | "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", 1876 | "dev": true, 1877 | "requires": { 1878 | "ret": "~0.1.10" 1879 | } 1880 | }, 1881 | "serialize-javascript": { 1882 | "version": "2.1.2", 1883 | "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-2.1.2.tgz", 1884 | "integrity": "sha512-rs9OggEUF0V4jUSecXazOYsLfu7OGK2qIn3c7IPBiffz32XniEp/TX9Xmc9LQfK2nQ2QKHvZ2oygKUGU0lG4jQ==", 1885 | "dev": true 1886 | }, 1887 | "set-value": { 1888 | "version": "2.0.1", 1889 | "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.1.tgz", 1890 | "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", 1891 | "dev": true, 1892 | "requires": { 1893 | "extend-shallow": "^2.0.1", 1894 | "is-extendable": "^0.1.1", 1895 | "is-plain-object": "^2.0.3", 1896 | "split-string": "^3.0.1" 1897 | }, 1898 | "dependencies": { 1899 | "extend-shallow": { 1900 | "version": "2.0.1", 1901 | "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", 1902 | "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", 1903 | "dev": true, 1904 | "requires": { 1905 | "is-extendable": "^0.1.0" 1906 | } 1907 | } 1908 | } 1909 | }, 1910 | "sirv": { 1911 | "version": "0.4.2", 1912 | "resolved": "https://registry.npmjs.org/sirv/-/sirv-0.4.2.tgz", 1913 | "integrity": "sha512-dQbZnsMaIiTQPZmbGmktz+c74zt/hyrJEB4tdp2Jj0RNv9J6B/OWR5RyrZEvIn9fyh9Zlg2OlE2XzKz6wMKGAw==", 1914 | "requires": { 1915 | "@polka/url": "^0.5.0", 1916 | "mime": "^2.3.1" 1917 | } 1918 | }, 1919 | "sirv-cli": { 1920 | "version": "0.4.5", 1921 | "resolved": "https://registry.npmjs.org/sirv-cli/-/sirv-cli-0.4.5.tgz", 1922 | "integrity": "sha512-Fl6icSm0EwPrXSGid2xphMp//WNTSX2yENRAGnJuuZNmdc8LvE/BtdZD3MPn28ifAfDqTMwbB3dpcZojAIOiBg==", 1923 | "requires": { 1924 | "console-clear": "^1.1.0", 1925 | "get-port": "^3.2.0", 1926 | "kleur": "^3.0.0", 1927 | "local-access": "^1.0.1", 1928 | "sade": "^1.4.0", 1929 | "sirv": "^0.4.2", 1930 | "tinydate": "^1.0.0" 1931 | } 1932 | }, 1933 | "snapdragon": { 1934 | "version": "0.8.2", 1935 | "resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz", 1936 | "integrity": "sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==", 1937 | "dev": true, 1938 | "requires": { 1939 | "base": "^0.11.1", 1940 | "debug": "^2.2.0", 1941 | "define-property": "^0.2.5", 1942 | "extend-shallow": "^2.0.1", 1943 | "map-cache": "^0.2.2", 1944 | "source-map": "^0.5.6", 1945 | "source-map-resolve": "^0.5.0", 1946 | "use": "^3.1.0" 1947 | }, 1948 | "dependencies": { 1949 | "define-property": { 1950 | "version": "0.2.5", 1951 | "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", 1952 | "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", 1953 | "dev": true, 1954 | "requires": { 1955 | "is-descriptor": "^0.1.0" 1956 | } 1957 | }, 1958 | "extend-shallow": { 1959 | "version": "2.0.1", 1960 | "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", 1961 | "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", 1962 | "dev": true, 1963 | "requires": { 1964 | "is-extendable": "^0.1.0" 1965 | } 1966 | } 1967 | } 1968 | }, 1969 | "snapdragon-node": { 1970 | "version": "2.1.1", 1971 | "resolved": "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz", 1972 | "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==", 1973 | "dev": true, 1974 | "requires": { 1975 | "define-property": "^1.0.0", 1976 | "isobject": "^3.0.0", 1977 | "snapdragon-util": "^3.0.1" 1978 | }, 1979 | "dependencies": { 1980 | "define-property": { 1981 | "version": "1.0.0", 1982 | "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", 1983 | "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", 1984 | "dev": true, 1985 | "requires": { 1986 | "is-descriptor": "^1.0.0" 1987 | } 1988 | }, 1989 | "is-accessor-descriptor": { 1990 | "version": "1.0.0", 1991 | "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", 1992 | "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", 1993 | "dev": true, 1994 | "requires": { 1995 | "kind-of": "^6.0.0" 1996 | } 1997 | }, 1998 | "is-data-descriptor": { 1999 | "version": "1.0.0", 2000 | "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", 2001 | "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", 2002 | "dev": true, 2003 | "requires": { 2004 | "kind-of": "^6.0.0" 2005 | } 2006 | }, 2007 | "is-descriptor": { 2008 | "version": "1.0.2", 2009 | "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", 2010 | "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", 2011 | "dev": true, 2012 | "requires": { 2013 | "is-accessor-descriptor": "^1.0.0", 2014 | "is-data-descriptor": "^1.0.0", 2015 | "kind-of": "^6.0.2" 2016 | } 2017 | } 2018 | } 2019 | }, 2020 | "snapdragon-util": { 2021 | "version": "3.0.1", 2022 | "resolved": "https://registry.npmjs.org/snapdragon-util/-/snapdragon-util-3.0.1.tgz", 2023 | "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==", 2024 | "dev": true, 2025 | "requires": { 2026 | "kind-of": "^3.2.0" 2027 | }, 2028 | "dependencies": { 2029 | "kind-of": { 2030 | "version": "3.2.2", 2031 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", 2032 | "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", 2033 | "dev": true, 2034 | "requires": { 2035 | "is-buffer": "^1.1.5" 2036 | } 2037 | } 2038 | } 2039 | }, 2040 | "source-map": { 2041 | "version": "0.5.7", 2042 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", 2043 | "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", 2044 | "dev": true 2045 | }, 2046 | "source-map-resolve": { 2047 | "version": "0.5.3", 2048 | "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.3.tgz", 2049 | "integrity": "sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==", 2050 | "dev": true, 2051 | "requires": { 2052 | "atob": "^2.1.2", 2053 | "decode-uri-component": "^0.2.0", 2054 | "resolve-url": "^0.2.1", 2055 | "source-map-url": "^0.4.0", 2056 | "urix": "^0.1.0" 2057 | } 2058 | }, 2059 | "source-map-support": { 2060 | "version": "0.5.16", 2061 | "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.16.tgz", 2062 | "integrity": "sha512-efyLRJDr68D9hBBNIPWFjhpFzURh+KJykQwvMyW5UiZzYwoF6l4YMMDIJJEyFWxWCqfyxLzz6tSfUFR+kXXsVQ==", 2063 | "dev": true, 2064 | "requires": { 2065 | "buffer-from": "^1.0.0", 2066 | "source-map": "^0.6.0" 2067 | }, 2068 | "dependencies": { 2069 | "source-map": { 2070 | "version": "0.6.1", 2071 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 2072 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", 2073 | "dev": true 2074 | } 2075 | } 2076 | }, 2077 | "source-map-url": { 2078 | "version": "0.4.0", 2079 | "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.0.tgz", 2080 | "integrity": "sha1-PpNdfd1zYxuXZZlW1VEo6HtQhKM=", 2081 | "dev": true 2082 | }, 2083 | "sourcemap-codec": { 2084 | "version": "1.4.8", 2085 | "resolved": "https://registry.npmjs.org/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz", 2086 | "integrity": "sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==", 2087 | "dev": true 2088 | }, 2089 | "split-string": { 2090 | "version": "3.1.0", 2091 | "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz", 2092 | "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", 2093 | "dev": true, 2094 | "requires": { 2095 | "extend-shallow": "^3.0.0" 2096 | } 2097 | }, 2098 | "static-extend": { 2099 | "version": "0.1.2", 2100 | "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz", 2101 | "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=", 2102 | "dev": true, 2103 | "requires": { 2104 | "define-property": "^0.2.5", 2105 | "object-copy": "^0.1.0" 2106 | }, 2107 | "dependencies": { 2108 | "define-property": { 2109 | "version": "0.2.5", 2110 | "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", 2111 | "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", 2112 | "dev": true, 2113 | "requires": { 2114 | "is-descriptor": "^0.1.0" 2115 | } 2116 | } 2117 | } 2118 | }, 2119 | "string_decoder": { 2120 | "version": "1.1.1", 2121 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 2122 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 2123 | "dev": true, 2124 | "requires": { 2125 | "safe-buffer": "~5.1.0" 2126 | } 2127 | }, 2128 | "supports-color": { 2129 | "version": "5.5.0", 2130 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 2131 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 2132 | "dev": true, 2133 | "requires": { 2134 | "has-flag": "^3.0.0" 2135 | } 2136 | }, 2137 | "svelte": { 2138 | "version": "3.18.2", 2139 | "resolved": "https://registry.npmjs.org/svelte/-/svelte-3.18.2.tgz", 2140 | "integrity": "sha512-jRk7jdYULb9V4Z+0BKlfofombmdIIQph4leojrOSHzvZBRmCredz7fZsJBiUDLO6h83XYekuLbwfy5zx1i95GQ==", 2141 | "dev": true 2142 | }, 2143 | "terser": { 2144 | "version": "4.6.3", 2145 | "resolved": "https://registry.npmjs.org/terser/-/terser-4.6.3.tgz", 2146 | "integrity": "sha512-Lw+ieAXmY69d09IIc/yqeBqXpEQIpDGZqT34ui1QWXIUpR2RjbqEkT8X7Lgex19hslSqcWM5iMN2kM11eMsESQ==", 2147 | "dev": true, 2148 | "requires": { 2149 | "commander": "^2.20.0", 2150 | "source-map": "~0.6.1", 2151 | "source-map-support": "~0.5.12" 2152 | }, 2153 | "dependencies": { 2154 | "source-map": { 2155 | "version": "0.6.1", 2156 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 2157 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", 2158 | "dev": true 2159 | } 2160 | } 2161 | }, 2162 | "tinydate": { 2163 | "version": "1.2.0", 2164 | "resolved": "https://registry.npmjs.org/tinydate/-/tinydate-1.2.0.tgz", 2165 | "integrity": "sha512-3GwPk8VhDFnUZ2TrgkhXJs6hcMAIIw4x/xkz+ayK6dGoQmp2nUwKzBXK0WnMsqkh6vfUhpqQicQF3rbshfyJkg==" 2166 | }, 2167 | "to-object-path": { 2168 | "version": "0.3.0", 2169 | "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", 2170 | "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", 2171 | "dev": true, 2172 | "requires": { 2173 | "kind-of": "^3.0.2" 2174 | }, 2175 | "dependencies": { 2176 | "kind-of": { 2177 | "version": "3.2.2", 2178 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", 2179 | "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", 2180 | "dev": true, 2181 | "requires": { 2182 | "is-buffer": "^1.1.5" 2183 | } 2184 | } 2185 | } 2186 | }, 2187 | "to-regex": { 2188 | "version": "3.0.2", 2189 | "resolved": "https://registry.npmjs.org/to-regex/-/to-regex-3.0.2.tgz", 2190 | "integrity": "sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==", 2191 | "dev": true, 2192 | "requires": { 2193 | "define-property": "^2.0.2", 2194 | "extend-shallow": "^3.0.2", 2195 | "regex-not": "^1.0.2", 2196 | "safe-regex": "^1.1.0" 2197 | } 2198 | }, 2199 | "to-regex-range": { 2200 | "version": "2.1.1", 2201 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", 2202 | "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", 2203 | "dev": true, 2204 | "requires": { 2205 | "is-number": "^3.0.0", 2206 | "repeat-string": "^1.6.1" 2207 | } 2208 | }, 2209 | "typescript": { 2210 | "version": "3.7.5", 2211 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.7.5.tgz", 2212 | "integrity": "sha512-/P5lkRXkWHNAbcJIiHPfRoKqyd7bsyCma1hZNUGfn20qm64T6ZBlrzprymeu918H+mB/0rIg2gGK/BXkhhYgBw==" 2213 | }, 2214 | "union-value": { 2215 | "version": "1.0.1", 2216 | "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.1.tgz", 2217 | "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", 2218 | "dev": true, 2219 | "requires": { 2220 | "arr-union": "^3.1.0", 2221 | "get-value": "^2.0.6", 2222 | "is-extendable": "^0.1.1", 2223 | "set-value": "^2.0.1" 2224 | } 2225 | }, 2226 | "unset-value": { 2227 | "version": "1.0.0", 2228 | "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", 2229 | "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=", 2230 | "dev": true, 2231 | "requires": { 2232 | "has-value": "^0.3.1", 2233 | "isobject": "^3.0.0" 2234 | }, 2235 | "dependencies": { 2236 | "has-value": { 2237 | "version": "0.3.1", 2238 | "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", 2239 | "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", 2240 | "dev": true, 2241 | "requires": { 2242 | "get-value": "^2.0.3", 2243 | "has-values": "^0.1.4", 2244 | "isobject": "^2.0.0" 2245 | }, 2246 | "dependencies": { 2247 | "isobject": { 2248 | "version": "2.1.0", 2249 | "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", 2250 | "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", 2251 | "dev": true, 2252 | "requires": { 2253 | "isarray": "1.0.0" 2254 | } 2255 | } 2256 | } 2257 | }, 2258 | "has-values": { 2259 | "version": "0.1.4", 2260 | "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", 2261 | "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=", 2262 | "dev": true 2263 | } 2264 | } 2265 | }, 2266 | "upath": { 2267 | "version": "1.2.0", 2268 | "resolved": "https://registry.npmjs.org/upath/-/upath-1.2.0.tgz", 2269 | "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==", 2270 | "dev": true 2271 | }, 2272 | "urix": { 2273 | "version": "0.1.0", 2274 | "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", 2275 | "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", 2276 | "dev": true 2277 | }, 2278 | "use": { 2279 | "version": "3.1.1", 2280 | "resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz", 2281 | "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", 2282 | "dev": true 2283 | }, 2284 | "util-deprecate": { 2285 | "version": "1.0.2", 2286 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 2287 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", 2288 | "dev": true 2289 | }, 2290 | "ws": { 2291 | "version": "6.2.1", 2292 | "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz", 2293 | "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==", 2294 | "dev": true, 2295 | "requires": { 2296 | "async-limiter": "~1.0.0" 2297 | } 2298 | } 2299 | } 2300 | } 2301 | -------------------------------------------------------------------------------- /frontend/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "svelte-app", 3 | "version": "1.0.0", 4 | "scripts": { 5 | "build": "rollup -c", 6 | "dev": "rollup -c -w", 7 | "start": "sirv public" 8 | }, 9 | "devDependencies": { 10 | "@rollup/plugin-commonjs": "^11.0.0", 11 | "@rollup/plugin-node-resolve": "^7.0.0", 12 | "rollup": "^1.20.0", 13 | "rollup-plugin-livereload": "^1.0.0", 14 | "rollup-plugin-svelte": "^5.0.3", 15 | "rollup-plugin-terser": "^5.1.2", 16 | "svelte": "^3.0.0" 17 | }, 18 | "dependencies": { 19 | "comlink": "^4.2.0", 20 | "sirv-cli": "^0.4.4", 21 | "typescript": "^3.7.5" 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /frontend/public/favicon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/noopkat/comlink-websocket-chat-experiment/9f3b414bd7486e54a69b0d0d7d361cff2d4dd453/frontend/public/favicon.png -------------------------------------------------------------------------------- /frontend/public/global.css: -------------------------------------------------------------------------------- 1 | html, body { 2 | position: relative; 3 | width: 100%; 4 | height: 100%; 5 | } 6 | 7 | body { 8 | color: #333; 9 | margin: 0; 10 | padding: 8px; 11 | box-sizing: border-box; 12 | font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen-Sans, Ubuntu, Cantarell, "Helvetica Neue", sans-serif; 13 | } 14 | 15 | a { 16 | color: rgb(0,100,200); 17 | text-decoration: none; 18 | } 19 | 20 | a:hover { 21 | text-decoration: underline; 22 | } 23 | 24 | a:visited { 25 | color: rgb(0,80,160); 26 | } 27 | 28 | label { 29 | display: block; 30 | } 31 | 32 | input, button, select, textarea { 33 | font-family: inherit; 34 | font-size: inherit; 35 | padding: 0.4em; 36 | margin: 0 0 0.5em 0; 37 | box-sizing: border-box; 38 | border: 1px solid #ccc; 39 | border-radius: 2px; 40 | } 41 | 42 | input:disabled { 43 | color: #ccc; 44 | } 45 | 46 | input[type="range"] { 47 | height: 0; 48 | } 49 | 50 | button { 51 | color: #333; 52 | background-color: #f4f4f4; 53 | outline: none; 54 | } 55 | 56 | button:disabled { 57 | color: #999; 58 | } 59 | 60 | button:not(:disabled):active { 61 | background-color: #ddd; 62 | } 63 | 64 | button:focus { 65 | border-color: #666; 66 | } 67 | -------------------------------------------------------------------------------- /frontend/public/images/smiley.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/noopkat/comlink-websocket-chat-experiment/9f3b414bd7486e54a69b0d0d7d361cff2d4dd453/frontend/public/images/smiley.png -------------------------------------------------------------------------------- /frontend/public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Svelte app 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | -------------------------------------------------------------------------------- /frontend/rollup.config.js: -------------------------------------------------------------------------------- 1 | import svelte from "rollup-plugin-svelte"; 2 | 3 | import resolve from "@rollup/plugin-node-resolve"; 4 | import commonjs from "@rollup/plugin-commonjs"; 5 | import livereload from "rollup-plugin-livereload"; 6 | import { terser } from "rollup-plugin-terser"; 7 | 8 | const production = !process.env.ROLLUP_WATCH; 9 | 10 | export default [ 11 | { 12 | input: "src/main.js", 13 | output: { 14 | sourcemap: true, 15 | format: "iife", 16 | name: "app", 17 | file: "public/build/bundle.js" 18 | }, 19 | plugins: [ 20 | svelte({ 21 | // enable run-time checks when not in production 22 | dev: !production, 23 | // we'll extract any component CSS out into 24 | // a separate file - better for performance 25 | css: css => { 26 | css.write("public/build/bundle.css"); 27 | } 28 | }), 29 | 30 | // If you have external dependencies installed from 31 | // npm, you'll most likely need these plugins. In 32 | // some cases you'll need additional configuration - 33 | // consult the documentation for details: 34 | // https://github.com/rollup/plugins/tree/master/packages/commonjs 35 | resolve({ 36 | browser: true, 37 | dedupe: ["svelte"] 38 | }), 39 | commonjs(), 40 | 41 | // In dev mode, call `npm run start` once 42 | // the bundle has been generated 43 | !production && serve(), 44 | 45 | // Watch the `public` directory and refresh the 46 | // browser on changes when not in production 47 | // !production && livereload('public'), 48 | 49 | // If we're building for production (npm run build 50 | // instead of npm run dev), minify 51 | production && terser() 52 | ], 53 | watch: { 54 | clearScreen: false 55 | } 56 | }, 57 | { 58 | input: "src/worker.js", 59 | output: { 60 | sourcemap: true, 61 | format: "iife", 62 | name: "app", 63 | file: "public/build/worker.js" 64 | }, 65 | plugins: [ 66 | resolve() 67 | ] 68 | } 69 | ]; 70 | 71 | function serve() { 72 | let started = false; 73 | 74 | return { 75 | writeBundle() { 76 | if (!started) { 77 | started = true; 78 | 79 | require("child_process").spawn("npm", ["run", "start", "--", "--dev"], { 80 | stdio: ["ignore", "inherit", "inherit"], 81 | shell: true 82 | }); 83 | } 84 | } 85 | }; 86 | } 87 | -------------------------------------------------------------------------------- /frontend/src/App.svelte: -------------------------------------------------------------------------------- 1 | 6 | 7 |
8 |

Hello {name}!

9 | 10 |
11 | 12 | -------------------------------------------------------------------------------- /frontend/src/Chat.svelte: -------------------------------------------------------------------------------- 1 | 46 | 47 | 54 | 55 |
56 |
57 | {#each messageList as message} 58 |
59 | {@html message} 60 |
61 | 62 | {/each} 63 |
64 |
65 | 66 | 67 |
68 |
-------------------------------------------------------------------------------- /frontend/src/main.js: -------------------------------------------------------------------------------- 1 | import App from './App.svelte'; 2 | 3 | const app = new App({ 4 | target: document.body, 5 | props: { 6 | name: 'Suz' 7 | } 8 | }); 9 | 10 | export default app; -------------------------------------------------------------------------------- /frontend/src/string-channel.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright 2019 Google Inc. All Rights Reserved. 3 | * Licensed under the Apache License, Version 2.0 (the "License"); 4 | * you may not use this file except in compliance with the License. 5 | * You may obtain a copy of the License at 6 | * http://www.apache.org/licenses/LICENSE-2.0 7 | * Unless required by applicable law or agreed to in writing, software 8 | * distributed under the License is distributed on an "AS IS" BASIS, 9 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 10 | * See the License for the specific language governing permissions and 11 | * limitations under the License. 12 | */ 13 | function generateUID() { 14 | return new Array(4) 15 | .fill(0) 16 | .map(() => Math.floor(Math.random() * Number.MAX_SAFE_INTEGER).toString(16)) 17 | .join("-"); 18 | } 19 | function* iterateAllProperties(value, path = [], visited = new WeakSet()) { 20 | if (value === undefined) 21 | return; 22 | if (visited.has(value)) 23 | return; 24 | if (typeof value === "string") 25 | return; 26 | if (typeof value === "object") 27 | visited.add(value); 28 | yield { value, path }; 29 | // Emit these objects, but don’t traverse them. 30 | if (ArrayBuffer.isView(value)) 31 | return; 32 | if (value instanceof ArrayBuffer) 33 | return; 34 | const keys = Object.keys(value); 35 | for (const key of keys) { 36 | yield* iterateAllProperties(value[key], [...path, key], visited); 37 | } 38 | } 39 | function findAllTransferables(obj) { 40 | const transferables = []; 41 | for (const entry of iterateAllProperties(obj)) { 42 | if (isTransferable(entry.value)) { 43 | transferables.push(entry); 44 | } 45 | } 46 | return transferables; 47 | } 48 | function replaceValueAtPath(value, path, newValue) { 49 | const lastProp = path[path.length - 1]; 50 | for (const prop of path.slice(0, -1)) { 51 | value = value[prop]; 52 | } 53 | const oldValue = value[lastProp]; 54 | value[lastProp] = newValue; 55 | return oldValue; 56 | } 57 | function padLeft(str, pad, length) { 58 | return (pad.repeat(length) + str).slice(-length); 59 | } 60 | function hexEncode(buffer) { 61 | return [...new Uint8Array(buffer)] 62 | .map(v => padLeft(v.toString(16), "0", 2)) 63 | .join(""); 64 | } 65 | function hexDecode(s) { 66 | return new Uint8Array(s 67 | .split(/(..)/) 68 | .filter(Boolean) 69 | .map(v => parseInt(v, 16))).buffer; 70 | } 71 | function addMessageListener(target, listener) { 72 | if ("on" in target) { 73 | return target.on("message", (data) => listener({ data })); 74 | } 75 | target.addEventListener("message", listener); 76 | } 77 | function getTypedArrayType(v) { 78 | if (!ArrayBuffer.isView(v)) { 79 | return 0 /* Raw */; 80 | } 81 | if (v instanceof Int8Array) { 82 | return 1 /* Int8 */; 83 | } 84 | if (v instanceof Uint8Array) { 85 | return 2 /* Uint8 */; 86 | } 87 | if (v instanceof Uint8ClampedArray) { 88 | return 3 /* Uint8Clamped */; 89 | } 90 | if (v instanceof Int16Array) { 91 | return 4 /* Int16 */; 92 | } 93 | if (v instanceof Uint16Array) { 94 | return 5 /* Uint16 */; 95 | } 96 | if (v instanceof Int32Array) { 97 | return 6 /* Int32 */; 98 | } 99 | if (v instanceof Uint32Array) { 100 | return 7 /* Uint32 */; 101 | } 102 | if (v instanceof Float32Array) { 103 | return 8 /* Float32 */; 104 | } 105 | if (v instanceof Float64Array) { 106 | return 9 /* Float64 */; 107 | } 108 | if (v instanceof BigInt64Array) { 109 | return 10 /* BigInt64 */; 110 | } 111 | if (v instanceof BigUint64Array) { 112 | return 11 /* BigUint64 */; 113 | } 114 | throw Error("Unknown ArrayBufferView type"); 115 | } 116 | function getTypedViewConstructor(type) { 117 | switch (type) { 118 | case 0 /* Raw */: 119 | return v => v; 120 | case 1 /* Int8 */: 121 | return v => new Int8Array(v); 122 | case 2 /* Uint8 */: 123 | return v => new Uint8Array(v); 124 | case 3 /* Uint8Clamped */: 125 | return v => new Uint8ClampedArray(v); 126 | case 4 /* Int16 */: 127 | return v => new Int16Array(v); 128 | case 5 /* Uint16 */: 129 | return v => new Uint16Array(v); 130 | case 6 /* Int32 */: 131 | return v => new Int32Array(v); 132 | case 7 /* Uint32 */: 133 | return v => new Uint32Array(v); 134 | case 8 /* Float32 */: 135 | return v => new Float32Array(v); 136 | case 9 /* Float64 */: 137 | return v => new Float64Array(v); 138 | case 10 /* BigInt64 */: 139 | return v => new BigInt64Array(v); 140 | case 11 /* BigUint64 */: 141 | return v => new BigUint64Array(v); 142 | } 143 | } 144 | const messagePortMap = new Map(); 145 | function makeTransferable(v, ep) { 146 | if (v instanceof MessagePort) { 147 | const uid = generateUID(); 148 | messagePortMap.set(uid, ep); 149 | const wrapped = wrap(ep, uid); 150 | addMessageListener(v, ({ data }) => { 151 | wrapped.postMessage(data, findAllTransferables(data).map(v => v.value)); 152 | }); 153 | v.start(); 154 | addMessageListener(wrapped, ({ data }) => { 155 | v.postMessage(data, findAllTransferables(data).map(v => v.value)); 156 | }); 157 | return { 158 | type: 0 /* MessagePort */, 159 | path: [], 160 | value: uid 161 | }; 162 | } 163 | else if (v instanceof ArrayBuffer || ArrayBuffer.isView(v)) { 164 | const buffer = v.buffer || v; 165 | return { 166 | type: 1 /* TypedArray */, 167 | subtype: getTypedArrayType(v), 168 | path: [], 169 | value: hexEncode(buffer) 170 | }; 171 | } 172 | throw Error("Not transferable"); 173 | } 174 | function isTransferable(v) { 175 | if (v instanceof MessagePort) { 176 | return true; 177 | } 178 | if (v instanceof ArrayBuffer) { 179 | return true; 180 | } 181 | if (ArrayBuffer.isView(v)) { 182 | return true; 183 | } 184 | return false; 185 | } 186 | function deserializeTransferable(transfer, ep) { 187 | switch (transfer.type) { 188 | case 0 /* MessagePort */: 189 | const port = wrap(ep, transfer.value); 190 | return [port, port]; 191 | case 1 /* TypedArray */: 192 | const constructor = getTypedViewConstructor(transfer.subtype); 193 | const buffer = hexDecode(transfer.value); 194 | return [constructor(buffer), buffer]; 195 | default: 196 | throw Error("Unknown transferable"); 197 | } 198 | } 199 | export function wrap(ep, uid = "") { 200 | const { port1, port2 } = new MessageChannel(); 201 | ep.addMessageListener(msg => { 202 | let payload; 203 | payload = JSON.parse(msg); 204 | if (payload.uid !== uid) { 205 | return; 206 | } 207 | const transferables = payload.transfer 208 | .map(transfer => { 209 | const [replacement, transferable] = deserializeTransferable(transfer, ep); 210 | replaceValueAtPath(payload.data, transfer.path, replacement); 211 | return transferable; 212 | }) 213 | .filter(Boolean); 214 | port2.postMessage(payload.data, transferables); 215 | }); 216 | addMessageListener(port2, ({ data }) => { 217 | const transfer = []; 218 | for (const { path } of findAllTransferables(data)) { 219 | const oldValue = replaceValueAtPath(data, path, null); 220 | const serializedTransferable = makeTransferable(oldValue, ep); 221 | serializedTransferable.path = path; 222 | transfer.push(serializedTransferable); 223 | } 224 | ep.send(JSON.stringify({ uid, data, transfer })); 225 | }); 226 | port2.start(); 227 | return port1; 228 | } 229 | -------------------------------------------------------------------------------- /frontend/src/string-channel.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright 2019 Google Inc. All Rights Reserved. 3 | * Licensed under the Apache License, Version 2.0 (the "License"); 4 | * you may not use this file except in compliance with the License. 5 | * You may obtain a copy of the License at 6 | * http://www.apache.org/licenses/LICENSE-2.0 7 | * Unless required by applicable law or agreed to in writing, software 8 | * distributed under the License is distributed on an "AS IS" BASIS, 9 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 10 | * See the License for the specific language governing permissions and 11 | * limitations under the License. 12 | */ 13 | 14 | export type StringListener = (msg: string) => void; 15 | export interface StringChannelEndpoint { 16 | addMessageListener(listener: StringListener): void; 17 | send(msg: string): void; 18 | } 19 | 20 | function generateUID(): string { 21 | return new Array(4) 22 | .fill(0) 23 | .map(() => Math.floor(Math.random() * Number.MAX_SAFE_INTEGER).toString(16)) 24 | .join("-"); 25 | } 26 | 27 | interface PropertyIteratorEntry { 28 | value: T; 29 | path: string[]; 30 | } 31 | 32 | function* iterateAllProperties( 33 | value: {} | undefined, 34 | path: string[] = [], 35 | visited: WeakSet<{}> = new WeakSet<{}>() 36 | ): Iterable { 37 | if (value === undefined) return; 38 | if (visited.has(value)) return; 39 | if (typeof value === "string") return; 40 | if (typeof value === "object") visited.add(value); 41 | yield { value, path }; 42 | 43 | // Emit these objects, but don’t traverse them. 44 | if (ArrayBuffer.isView(value)) return; 45 | if (value instanceof ArrayBuffer) return; 46 | 47 | const keys = Object.keys(value); 48 | for (const key of keys) { 49 | yield* iterateAllProperties((value as any)[key], [...path, key], visited); 50 | } 51 | } 52 | 53 | function findAllTransferables(obj: {}): PropertyIteratorEntry[] { 54 | const transferables: PropertyIteratorEntry[] = []; 55 | for (const entry of iterateAllProperties(obj)) { 56 | if (isTransferable(entry.value)) { 57 | transferables.push(entry as any); 58 | } 59 | } 60 | return transferables; 61 | } 62 | 63 | function replaceValueAtPath(value: any, path: string[], newValue: any): {} { 64 | const lastProp = path[path.length - 1]; 65 | for (const prop of path.slice(0, -1)) { 66 | value = value[prop]; 67 | } 68 | const oldValue = value[lastProp]; 69 | value[lastProp] = newValue; 70 | return oldValue; 71 | } 72 | 73 | function padLeft(str: string, pad: string, length: number): string { 74 | return (pad.repeat(length) + str).slice(-length); 75 | } 76 | 77 | function hexEncode(buffer: ArrayBuffer): string { 78 | return [...new Uint8Array(buffer)] 79 | .map(v => padLeft(v.toString(16), "0", 2)) 80 | .join(""); 81 | } 82 | 83 | function hexDecode(s: string): ArrayBuffer { 84 | return new Uint8Array( 85 | s 86 | .split(/(..)/) 87 | .filter(Boolean) 88 | .map(v => parseInt(v, 16)) 89 | ).buffer; 90 | } 91 | 92 | function addMessageListener( 93 | target: EventTarget, 94 | listener: (ev: MessageEvent) => void 95 | ) { 96 | if ("on" in target) { 97 | return (target as any).on("message", (data: any) => 98 | listener({ data } as any) 99 | ); 100 | } 101 | target.addEventListener("message", listener as any); 102 | } 103 | 104 | const enum SerializedTransferableType { 105 | MessagePort, 106 | TypedArray 107 | } 108 | 109 | const enum TypedArrayType { 110 | Raw, 111 | Int8, 112 | Uint8, 113 | Uint8Clamped, 114 | Int16, 115 | Uint16, 116 | Int32, 117 | Uint32, 118 | Float32, 119 | Float64, 120 | BigInt64, 121 | BigUint64 122 | } 123 | 124 | interface SerializedTransferableTypedArray { 125 | type: SerializedTransferableType.TypedArray; 126 | subtype: TypedArrayType; 127 | path: string[]; 128 | value: string; 129 | } 130 | 131 | function getTypedArrayType(v: ArrayBuffer | ArrayBufferView): TypedArrayType { 132 | if (!ArrayBuffer.isView(v)) { 133 | return TypedArrayType.Raw; 134 | } 135 | if (v instanceof Int8Array) { 136 | return TypedArrayType.Int8; 137 | } 138 | if (v instanceof Uint8Array) { 139 | return TypedArrayType.Uint8; 140 | } 141 | if (v instanceof Uint8ClampedArray) { 142 | return TypedArrayType.Uint8Clamped; 143 | } 144 | if (v instanceof Int16Array) { 145 | return TypedArrayType.Int16; 146 | } 147 | if (v instanceof Uint16Array) { 148 | return TypedArrayType.Uint16; 149 | } 150 | if (v instanceof Int32Array) { 151 | return TypedArrayType.Int32; 152 | } 153 | if (v instanceof Uint32Array) { 154 | return TypedArrayType.Uint32; 155 | } 156 | if (v instanceof Float32Array) { 157 | return TypedArrayType.Float32; 158 | } 159 | if (v instanceof Float64Array) { 160 | return TypedArrayType.Float64; 161 | } 162 | if (v instanceof BigInt64Array) { 163 | return TypedArrayType.BigInt64; 164 | } 165 | if (v instanceof BigUint64Array) { 166 | return TypedArrayType.BigUint64; 167 | } 168 | throw Error("Unknown ArrayBufferView type"); 169 | } 170 | 171 | function getTypedViewConstructor( 172 | type: TypedArrayType 173 | ): (v: ArrayBuffer) => ArrayBufferView | ArrayBuffer { 174 | switch (type) { 175 | case TypedArrayType.Raw: 176 | return v => v; 177 | case TypedArrayType.Int8: 178 | return v => new Int8Array(v); 179 | case TypedArrayType.Uint8: 180 | return v => new Uint8Array(v); 181 | case TypedArrayType.Uint8Clamped: 182 | return v => new Uint8ClampedArray(v); 183 | case TypedArrayType.Int16: 184 | return v => new Int16Array(v); 185 | case TypedArrayType.Uint16: 186 | return v => new Uint16Array(v); 187 | case TypedArrayType.Int32: 188 | return v => new Int32Array(v); 189 | case TypedArrayType.Uint32: 190 | return v => new Uint32Array(v); 191 | case TypedArrayType.Float32: 192 | return v => new Float32Array(v); 193 | case TypedArrayType.Float64: 194 | return v => new Float64Array(v); 195 | case TypedArrayType.BigInt64: 196 | return v => new BigInt64Array(v); 197 | case TypedArrayType.BigUint64: 198 | return v => new BigUint64Array(v); 199 | } 200 | } 201 | 202 | interface SerializedTransferableMessagePort { 203 | type: SerializedTransferableType.MessagePort; 204 | path: string[]; 205 | value: string; 206 | } 207 | 208 | type SerializedTransferable = 209 | | SerializedTransferableMessagePort 210 | | SerializedTransferableTypedArray; 211 | const messagePortMap = new Map(); 212 | function makeTransferable( 213 | v: unknown, 214 | ep: StringChannelEndpoint 215 | ): SerializedTransferable { 216 | if (v instanceof MessagePort) { 217 | const uid = generateUID(); 218 | messagePortMap.set(uid, ep); 219 | const wrapped = wrap(ep, uid); 220 | addMessageListener(v, ({ data }) => { 221 | wrapped.postMessage( 222 | data, 223 | findAllTransferables(data).map(v => v.value) 224 | ); 225 | }); 226 | v.start(); 227 | addMessageListener(wrapped, ({ data }) => { 228 | v.postMessage( 229 | data, 230 | findAllTransferables(data).map(v => v.value) 231 | ); 232 | }); 233 | 234 | return { 235 | type: SerializedTransferableType.MessagePort, 236 | path: [], 237 | value: uid 238 | }; 239 | } else if (v instanceof ArrayBuffer || ArrayBuffer.isView(v)) { 240 | const buffer = (v as any).buffer || v; 241 | return { 242 | type: SerializedTransferableType.TypedArray, 243 | subtype: getTypedArrayType(v), 244 | path: [], 245 | value: hexEncode(buffer) 246 | }; 247 | } 248 | throw Error("Not transferable"); 249 | } 250 | 251 | function isTransferable(v: any): v is Transferable { 252 | if (v instanceof MessagePort) { 253 | return true; 254 | } 255 | if (v instanceof ArrayBuffer) { 256 | return true; 257 | } 258 | if (ArrayBuffer.isView(v)) { 259 | return true; 260 | } 261 | return false; 262 | } 263 | 264 | function deserializeTransferable( 265 | transfer: SerializedTransferable, 266 | ep: StringChannelEndpoint 267 | ): [any, Transferable] { 268 | switch (transfer.type) { 269 | case SerializedTransferableType.MessagePort: 270 | const port = wrap(ep, transfer.value); 271 | return [port, port]; 272 | case SerializedTransferableType.TypedArray: 273 | const constructor = getTypedViewConstructor(transfer.subtype); 274 | const buffer = hexDecode(transfer.value); 275 | return [constructor(buffer), buffer]; 276 | default: 277 | throw Error("Unknown transferable"); 278 | } 279 | } 280 | 281 | interface StringChannelPayload { 282 | uid: string; 283 | data: any; 284 | transfer: SerializedTransferable[]; 285 | } 286 | 287 | export function wrap(ep: StringChannelEndpoint, uid = "") { 288 | const { port1, port2 } = new MessageChannel(); 289 | 290 | ep.addMessageListener(msg => { 291 | let payload: StringChannelPayload; 292 | payload = JSON.parse(msg); 293 | if (payload.uid !== uid) { 294 | return; 295 | } 296 | const transferables: Transferable[] = payload.transfer 297 | .map(transfer => { 298 | const [replacement, transferable] = deserializeTransferable( 299 | transfer, 300 | ep 301 | ); 302 | replaceValueAtPath(payload.data, transfer.path, replacement); 303 | return transferable; 304 | }) 305 | .filter(Boolean); 306 | port2.postMessage(payload.data, transferables); 307 | }); 308 | 309 | addMessageListener(port2, ({ data }) => { 310 | const transfer: SerializedTransferable[] = []; 311 | for (const { path } of findAllTransferables(data)) { 312 | const oldValue = replaceValueAtPath(data, path, null); 313 | const serializedTransferable = makeTransferable(oldValue, ep); 314 | serializedTransferable.path = path; 315 | transfer.push(serializedTransferable); 316 | } 317 | ep.send(JSON.stringify({ uid, data, transfer })); 318 | }); 319 | port2.start(); 320 | return port1; 321 | } -------------------------------------------------------------------------------- /frontend/src/worker.js: -------------------------------------------------------------------------------- 1 | import * as Comlink from "comlink"; 2 | 3 | const smileys = new Map([ 4 | [/:\)/g, "/images/smiley.png"] 5 | ]); 6 | 7 | const api = { 8 | sayHello(name) { 9 | console.log(`Hello ${name}!`); 10 | }, 11 | processMessage(message) { 12 | for(const [regexp, image] of smileys.entries()) { 13 | message = message.replace(regexp, ``); 14 | } 15 | return message; 16 | } 17 | } 18 | 19 | Comlink.expose(api); -------------------------------------------------------------------------------- /server/index.js: -------------------------------------------------------------------------------- 1 | const express = require("express"); 2 | const app = express(); 3 | const wsapp = require("express-ws")(app); 4 | const Comlink = require("comlink"); 5 | const nodeEndpoint = require("comlink/dist/umd/node-adapter.js"); 6 | const {wrap} = require("./string-channel.js"); 7 | 8 | const { MessageChannel, MessagePort } = require("worker_threads"); 9 | globalThis.MessageChannel = MessageChannel; 10 | globalThis.MessagePort = MessagePort; 11 | 12 | function websocketEndpoint(ws) { 13 | return nodeEndpoint( 14 | wrap({ 15 | addMessageListener(f) { 16 | ws.addEventListener("message", ev => f(ev.data)); 17 | }, 18 | send(msg) { 19 | ws.send(msg); 20 | } 21 | }) 22 | ); 23 | } 24 | 25 | const listeners = new Set(); 26 | 27 | class ChatClient { 28 | setName(name) { 29 | this._name = name; 30 | } 31 | sendMessage() { 32 | if(!this._name) { 33 | return; 34 | } 35 | } 36 | } 37 | 38 | app.ws("/ws2", (ws, req) => { 39 | Comlink.expose(new ChatClient(), websocketEndpoint(ws)); 40 | }); 41 | 42 | app.ws("/ws", (ws, req) => { 43 | listeners.add(ws); 44 | ws.on("message", data => { 45 | for (const listener of listeners) { 46 | if (listener === ws) { 47 | // Don’t send message to yourself. 48 | continue; 49 | } 50 | listener.send(data); 51 | } 52 | }); 53 | ws.on("close", () => { 54 | listeners.delete(ws); 55 | }); 56 | }); 57 | app.use("/", express.static("../frontend/public")); 58 | 59 | app.listen(8081); 60 | 61 | 62 | 63 | 64 | 65 | app.get("/test", (req, res) => { 66 | res.setHeader("Content-Type", "text/html"); 67 | res.send(` 68 | 69 | 70 | 71 |

72 |     
84 |   `);
85 | });
86 | 


--------------------------------------------------------------------------------
/server/package-lock.json:
--------------------------------------------------------------------------------
  1 | {
  2 |   "name": "server",
  3 |   "version": "1.0.0",
  4 |   "lockfileVersion": 1,
  5 |   "requires": true,
  6 |   "dependencies": {
  7 |     "accepts": {
  8 |       "version": "1.3.7",
  9 |       "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
 10 |       "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
 11 |       "requires": {
 12 |         "mime-types": "~2.1.24",
 13 |         "negotiator": "0.6.2"
 14 |       }
 15 |     },
 16 |     "array-flatten": {
 17 |       "version": "1.1.1",
 18 |       "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
 19 |       "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
 20 |     },
 21 |     "async-limiter": {
 22 |       "version": "1.0.1",
 23 |       "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz",
 24 |       "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ=="
 25 |     },
 26 |     "body-parser": {
 27 |       "version": "1.19.0",
 28 |       "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
 29 |       "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
 30 |       "requires": {
 31 |         "bytes": "3.1.0",
 32 |         "content-type": "~1.0.4",
 33 |         "debug": "2.6.9",
 34 |         "depd": "~1.1.2",
 35 |         "http-errors": "1.7.2",
 36 |         "iconv-lite": "0.4.24",
 37 |         "on-finished": "~2.3.0",
 38 |         "qs": "6.7.0",
 39 |         "raw-body": "2.4.0",
 40 |         "type-is": "~1.6.17"
 41 |       }
 42 |     },
 43 |     "bytes": {
 44 |       "version": "3.1.0",
 45 |       "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
 46 |       "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg=="
 47 |     },
 48 |     "comlink": {
 49 |       "version": "4.2.0",
 50 |       "resolved": "https://registry.npmjs.org/comlink/-/comlink-4.2.0.tgz",
 51 |       "integrity": "sha512-33hF3yYzZicIWLREuvluTGSZkbAXYwRmrA9IYQo0/P+0O45qgvzR0FHX4tRZztObFMTs/RhnN0G0UBNDAZSiCg=="
 52 |     },
 53 |     "content-disposition": {
 54 |       "version": "0.5.3",
 55 |       "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
 56 |       "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
 57 |       "requires": {
 58 |         "safe-buffer": "5.1.2"
 59 |       }
 60 |     },
 61 |     "content-type": {
 62 |       "version": "1.0.4",
 63 |       "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
 64 |       "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
 65 |     },
 66 |     "cookie": {
 67 |       "version": "0.4.0",
 68 |       "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
 69 |       "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg=="
 70 |     },
 71 |     "cookie-signature": {
 72 |       "version": "1.0.6",
 73 |       "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
 74 |       "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
 75 |     },
 76 |     "debug": {
 77 |       "version": "2.6.9",
 78 |       "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
 79 |       "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
 80 |       "requires": {
 81 |         "ms": "2.0.0"
 82 |       }
 83 |     },
 84 |     "depd": {
 85 |       "version": "1.1.2",
 86 |       "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
 87 |       "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
 88 |     },
 89 |     "destroy": {
 90 |       "version": "1.0.4",
 91 |       "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
 92 |       "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
 93 |     },
 94 |     "ee-first": {
 95 |       "version": "1.1.1",
 96 |       "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
 97 |       "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
 98 |     },
 99 |     "encodeurl": {
100 |       "version": "1.0.2",
101 |       "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
102 |       "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
103 |     },
104 |     "escape-html": {
105 |       "version": "1.0.3",
106 |       "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
107 |       "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
108 |     },
109 |     "etag": {
110 |       "version": "1.8.1",
111 |       "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
112 |       "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
113 |     },
114 |     "express": {
115 |       "version": "4.17.1",
116 |       "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
117 |       "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
118 |       "requires": {
119 |         "accepts": "~1.3.7",
120 |         "array-flatten": "1.1.1",
121 |         "body-parser": "1.19.0",
122 |         "content-disposition": "0.5.3",
123 |         "content-type": "~1.0.4",
124 |         "cookie": "0.4.0",
125 |         "cookie-signature": "1.0.6",
126 |         "debug": "2.6.9",
127 |         "depd": "~1.1.2",
128 |         "encodeurl": "~1.0.2",
129 |         "escape-html": "~1.0.3",
130 |         "etag": "~1.8.1",
131 |         "finalhandler": "~1.1.2",
132 |         "fresh": "0.5.2",
133 |         "merge-descriptors": "1.0.1",
134 |         "methods": "~1.1.2",
135 |         "on-finished": "~2.3.0",
136 |         "parseurl": "~1.3.3",
137 |         "path-to-regexp": "0.1.7",
138 |         "proxy-addr": "~2.0.5",
139 |         "qs": "6.7.0",
140 |         "range-parser": "~1.2.1",
141 |         "safe-buffer": "5.1.2",
142 |         "send": "0.17.1",
143 |         "serve-static": "1.14.1",
144 |         "setprototypeof": "1.1.1",
145 |         "statuses": "~1.5.0",
146 |         "type-is": "~1.6.18",
147 |         "utils-merge": "1.0.1",
148 |         "vary": "~1.1.2"
149 |       }
150 |     },
151 |     "express-ws": {
152 |       "version": "4.0.0",
153 |       "resolved": "https://registry.npmjs.org/express-ws/-/express-ws-4.0.0.tgz",
154 |       "integrity": "sha512-KEyUw8AwRET2iFjFsI1EJQrJ/fHeGiJtgpYgEWG3yDv4l/To/m3a2GaYfeGyB3lsWdvbesjF5XCMx+SVBgAAYw==",
155 |       "requires": {
156 |         "ws": "^5.2.0"
157 |       }
158 |     },
159 |     "finalhandler": {
160 |       "version": "1.1.2",
161 |       "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
162 |       "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
163 |       "requires": {
164 |         "debug": "2.6.9",
165 |         "encodeurl": "~1.0.2",
166 |         "escape-html": "~1.0.3",
167 |         "on-finished": "~2.3.0",
168 |         "parseurl": "~1.3.3",
169 |         "statuses": "~1.5.0",
170 |         "unpipe": "~1.0.0"
171 |       }
172 |     },
173 |     "forwarded": {
174 |       "version": "0.1.2",
175 |       "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
176 |       "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
177 |     },
178 |     "fresh": {
179 |       "version": "0.5.2",
180 |       "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
181 |       "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
182 |     },
183 |     "http-errors": {
184 |       "version": "1.7.2",
185 |       "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
186 |       "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
187 |       "requires": {
188 |         "depd": "~1.1.2",
189 |         "inherits": "2.0.3",
190 |         "setprototypeof": "1.1.1",
191 |         "statuses": ">= 1.5.0 < 2",
192 |         "toidentifier": "1.0.0"
193 |       }
194 |     },
195 |     "iconv-lite": {
196 |       "version": "0.4.24",
197 |       "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
198 |       "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
199 |       "requires": {
200 |         "safer-buffer": ">= 2.1.2 < 3"
201 |       }
202 |     },
203 |     "inherits": {
204 |       "version": "2.0.3",
205 |       "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
206 |       "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
207 |     },
208 |     "ipaddr.js": {
209 |       "version": "1.9.0",
210 |       "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.0.tgz",
211 |       "integrity": "sha512-M4Sjn6N/+O6/IXSJseKqHoFc+5FdGJ22sXqnjTpdZweHK64MzEPAyQZyEU3R/KRv2GLoa7nNtg/C2Ev6m7z+eA=="
212 |     },
213 |     "media-typer": {
214 |       "version": "0.3.0",
215 |       "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
216 |       "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
217 |     },
218 |     "merge-descriptors": {
219 |       "version": "1.0.1",
220 |       "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
221 |       "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
222 |     },
223 |     "methods": {
224 |       "version": "1.1.2",
225 |       "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
226 |       "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
227 |     },
228 |     "mime": {
229 |       "version": "1.6.0",
230 |       "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
231 |       "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
232 |     },
233 |     "mime-db": {
234 |       "version": "1.43.0",
235 |       "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.43.0.tgz",
236 |       "integrity": "sha512-+5dsGEEovYbT8UY9yD7eE4XTc4UwJ1jBYlgaQQF38ENsKR3wj/8q8RFZrF9WIZpB2V1ArTVFUva8sAul1NzRzQ=="
237 |     },
238 |     "mime-types": {
239 |       "version": "2.1.26",
240 |       "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.26.tgz",
241 |       "integrity": "sha512-01paPWYgLrkqAyrlDorC1uDwl2p3qZT7yl806vW7DvDoxwXi46jsjFbg+WdwotBIk6/MbEhO/dh5aZ5sNj/dWQ==",
242 |       "requires": {
243 |         "mime-db": "1.43.0"
244 |       }
245 |     },
246 |     "ms": {
247 |       "version": "2.0.0",
248 |       "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
249 |       "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
250 |     },
251 |     "negotiator": {
252 |       "version": "0.6.2",
253 |       "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
254 |       "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
255 |     },
256 |     "on-finished": {
257 |       "version": "2.3.0",
258 |       "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
259 |       "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
260 |       "requires": {
261 |         "ee-first": "1.1.1"
262 |       }
263 |     },
264 |     "parseurl": {
265 |       "version": "1.3.3",
266 |       "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
267 |       "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
268 |     },
269 |     "path-to-regexp": {
270 |       "version": "0.1.7",
271 |       "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
272 |       "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
273 |     },
274 |     "prettier": {
275 |       "version": "1.19.1",
276 |       "resolved": "https://registry.npmjs.org/prettier/-/prettier-1.19.1.tgz",
277 |       "integrity": "sha512-s7PoyDv/II1ObgQunCbB9PdLmUcBZcnWOcxDh7O0N/UwDEsHyqkW+Qh28jW+mVuCdx7gLB0BotYI1Y6uI9iyew==",
278 |       "dev": true
279 |     },
280 |     "proxy-addr": {
281 |       "version": "2.0.5",
282 |       "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.5.tgz",
283 |       "integrity": "sha512-t/7RxHXPH6cJtP0pRG6smSr9QJidhB+3kXu0KgXnbGYMgzEnUxRQ4/LDdfOwZEMyIh3/xHb8PX3t+lfL9z+YVQ==",
284 |       "requires": {
285 |         "forwarded": "~0.1.2",
286 |         "ipaddr.js": "1.9.0"
287 |       }
288 |     },
289 |     "qs": {
290 |       "version": "6.7.0",
291 |       "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
292 |       "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ=="
293 |     },
294 |     "range-parser": {
295 |       "version": "1.2.1",
296 |       "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
297 |       "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
298 |     },
299 |     "raw-body": {
300 |       "version": "2.4.0",
301 |       "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
302 |       "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
303 |       "requires": {
304 |         "bytes": "3.1.0",
305 |         "http-errors": "1.7.2",
306 |         "iconv-lite": "0.4.24",
307 |         "unpipe": "1.0.0"
308 |       }
309 |     },
310 |     "safe-buffer": {
311 |       "version": "5.1.2",
312 |       "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
313 |       "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
314 |     },
315 |     "safer-buffer": {
316 |       "version": "2.1.2",
317 |       "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
318 |       "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
319 |     },
320 |     "send": {
321 |       "version": "0.17.1",
322 |       "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
323 |       "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
324 |       "requires": {
325 |         "debug": "2.6.9",
326 |         "depd": "~1.1.2",
327 |         "destroy": "~1.0.4",
328 |         "encodeurl": "~1.0.2",
329 |         "escape-html": "~1.0.3",
330 |         "etag": "~1.8.1",
331 |         "fresh": "0.5.2",
332 |         "http-errors": "~1.7.2",
333 |         "mime": "1.6.0",
334 |         "ms": "2.1.1",
335 |         "on-finished": "~2.3.0",
336 |         "range-parser": "~1.2.1",
337 |         "statuses": "~1.5.0"
338 |       },
339 |       "dependencies": {
340 |         "ms": {
341 |           "version": "2.1.1",
342 |           "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
343 |           "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
344 |         }
345 |       }
346 |     },
347 |     "serve-static": {
348 |       "version": "1.14.1",
349 |       "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
350 |       "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
351 |       "requires": {
352 |         "encodeurl": "~1.0.2",
353 |         "escape-html": "~1.0.3",
354 |         "parseurl": "~1.3.3",
355 |         "send": "0.17.1"
356 |       }
357 |     },
358 |     "setprototypeof": {
359 |       "version": "1.1.1",
360 |       "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
361 |       "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
362 |     },
363 |     "statuses": {
364 |       "version": "1.5.0",
365 |       "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
366 |       "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow="
367 |     },
368 |     "toidentifier": {
369 |       "version": "1.0.0",
370 |       "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
371 |       "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw=="
372 |     },
373 |     "type-is": {
374 |       "version": "1.6.18",
375 |       "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
376 |       "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
377 |       "requires": {
378 |         "media-typer": "0.3.0",
379 |         "mime-types": "~2.1.24"
380 |       }
381 |     },
382 |     "typescript": {
383 |       "version": "3.7.5",
384 |       "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.7.5.tgz",
385 |       "integrity": "sha512-/P5lkRXkWHNAbcJIiHPfRoKqyd7bsyCma1hZNUGfn20qm64T6ZBlrzprymeu918H+mB/0rIg2gGK/BXkhhYgBw=="
386 |     },
387 |     "unpipe": {
388 |       "version": "1.0.0",
389 |       "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
390 |       "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
391 |     },
392 |     "utils-merge": {
393 |       "version": "1.0.1",
394 |       "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
395 |       "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
396 |     },
397 |     "vary": {
398 |       "version": "1.1.2",
399 |       "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
400 |       "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
401 |     },
402 |     "ws": {
403 |       "version": "5.2.2",
404 |       "resolved": "https://registry.npmjs.org/ws/-/ws-5.2.2.tgz",
405 |       "integrity": "sha512-jaHFD6PFv6UgoIVda6qZllptQsMlDEJkTQcybzzXDYM1XO9Y8em691FGMPmM46WGyLU4z9KMgQN+qrux/nhlHA==",
406 |       "requires": {
407 |         "async-limiter": "~1.0.0"
408 |       }
409 |     }
410 |   }
411 | }
412 | 


--------------------------------------------------------------------------------
/server/package.json:
--------------------------------------------------------------------------------
 1 | {
 2 |   "name": "server",
 3 |   "version": "1.0.0",
 4 |   "description": "",
 5 |   "main": "index.js",
 6 |   "scripts": {
 7 |     "test": "echo \"Error: no test specified\" && exit 1"
 8 |   },
 9 |   "keywords": [],
10 |   "author": "",
11 |   "license": "MIT",
12 |   "dependencies": {
13 |     "comlink": "^4.2.0",
14 |     "express": "^4.17.1",
15 |     "express-ws": "^4.0.0",
16 |     "typescript": "^3.7.5"
17 |   },
18 |   "devDependencies": {
19 |     "prettier": "^1.19.1"
20 |   }
21 | }
22 | 


--------------------------------------------------------------------------------
/server/string-channel.js:
--------------------------------------------------------------------------------
  1 | "use strict";
  2 | /**
  3 |  * Copyright 2019 Google Inc. All Rights Reserved.
  4 |  * Licensed under the Apache License, Version 2.0 (the "License");
  5 |  * you may not use this file except in compliance with the License.
  6 |  * You may obtain a copy of the License at
  7 |  *     http://www.apache.org/licenses/LICENSE-2.0
  8 |  * Unless required by applicable law or agreed to in writing, software
  9 |  * distributed under the License is distributed on an "AS IS" BASIS,
 10 |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 11 |  * See the License for the specific language governing permissions and
 12 |  * limitations under the License.
 13 |  */
 14 | Object.defineProperty(exports, "__esModule", { value: true });
 15 | function generateUID() {
 16 |     return new Array(4)
 17 |         .fill(0)
 18 |         .map(() => Math.floor(Math.random() * Number.MAX_SAFE_INTEGER).toString(16))
 19 |         .join("-");
 20 | }
 21 | function* iterateAllProperties(value, path = [], visited = new WeakSet()) {
 22 |     if (value === undefined)
 23 |         return;
 24 |     if (visited.has(value))
 25 |         return;
 26 |     if (typeof value === "string")
 27 |         return;
 28 |     if (typeof value === "object")
 29 |         visited.add(value);
 30 |     yield { value, path };
 31 |     // Emit these objects, but don’t traverse them.
 32 |     if (ArrayBuffer.isView(value))
 33 |         return;
 34 |     if (value instanceof ArrayBuffer)
 35 |         return;
 36 |     const keys = Object.keys(value);
 37 |     for (const key of keys) {
 38 |         yield* iterateAllProperties(value[key], [...path, key], visited);
 39 |     }
 40 | }
 41 | function findAllTransferables(obj) {
 42 |     const transferables = [];
 43 |     for (const entry of iterateAllProperties(obj)) {
 44 |         if (isTransferable(entry.value)) {
 45 |             transferables.push(entry);
 46 |         }
 47 |     }
 48 |     return transferables;
 49 | }
 50 | function replaceValueAtPath(value, path, newValue) {
 51 |     const lastProp = path[path.length - 1];
 52 |     for (const prop of path.slice(0, -1)) {
 53 |         value = value[prop];
 54 |     }
 55 |     const oldValue = value[lastProp];
 56 |     value[lastProp] = newValue;
 57 |     return oldValue;
 58 | }
 59 | function padLeft(str, pad, length) {
 60 |     return (pad.repeat(length) + str).slice(-length);
 61 | }
 62 | function hexEncode(buffer) {
 63 |     return [...new Uint8Array(buffer)]
 64 |         .map(v => padLeft(v.toString(16), "0", 2))
 65 |         .join("");
 66 | }
 67 | function hexDecode(s) {
 68 |     return new Uint8Array(s
 69 |         .split(/(..)/)
 70 |         .filter(Boolean)
 71 |         .map(v => parseInt(v, 16))).buffer;
 72 | }
 73 | function addMessageListener(target, listener) {
 74 |     if ("on" in target) {
 75 |         return target.on("message", (data) => listener({ data }));
 76 |     }
 77 |     target.addEventListener("message", listener);
 78 | }
 79 | function getTypedArrayType(v) {
 80 |     if (!ArrayBuffer.isView(v)) {
 81 |         return 0 /* Raw */;
 82 |     }
 83 |     if (v instanceof Int8Array) {
 84 |         return 1 /* Int8 */;
 85 |     }
 86 |     if (v instanceof Uint8Array) {
 87 |         return 2 /* Uint8 */;
 88 |     }
 89 |     if (v instanceof Uint8ClampedArray) {
 90 |         return 3 /* Uint8Clamped */;
 91 |     }
 92 |     if (v instanceof Int16Array) {
 93 |         return 4 /* Int16 */;
 94 |     }
 95 |     if (v instanceof Uint16Array) {
 96 |         return 5 /* Uint16 */;
 97 |     }
 98 |     if (v instanceof Int32Array) {
 99 |         return 6 /* Int32 */;
100 |     }
101 |     if (v instanceof Uint32Array) {
102 |         return 7 /* Uint32 */;
103 |     }
104 |     if (v instanceof Float32Array) {
105 |         return 8 /* Float32 */;
106 |     }
107 |     if (v instanceof Float64Array) {
108 |         return 9 /* Float64 */;
109 |     }
110 |     if (v instanceof BigInt64Array) {
111 |         return 10 /* BigInt64 */;
112 |     }
113 |     if (v instanceof BigUint64Array) {
114 |         return 11 /* BigUint64 */;
115 |     }
116 |     throw Error("Unknown ArrayBufferView type");
117 | }
118 | function getTypedViewConstructor(type) {
119 |     switch (type) {
120 |         case 0 /* Raw */:
121 |             return v => v;
122 |         case 1 /* Int8 */:
123 |             return v => new Int8Array(v);
124 |         case 2 /* Uint8 */:
125 |             return v => new Uint8Array(v);
126 |         case 3 /* Uint8Clamped */:
127 |             return v => new Uint8ClampedArray(v);
128 |         case 4 /* Int16 */:
129 |             return v => new Int16Array(v);
130 |         case 5 /* Uint16 */:
131 |             return v => new Uint16Array(v);
132 |         case 6 /* Int32 */:
133 |             return v => new Int32Array(v);
134 |         case 7 /* Uint32 */:
135 |             return v => new Uint32Array(v);
136 |         case 8 /* Float32 */:
137 |             return v => new Float32Array(v);
138 |         case 9 /* Float64 */:
139 |             return v => new Float64Array(v);
140 |         case 10 /* BigInt64 */:
141 |             return v => new BigInt64Array(v);
142 |         case 11 /* BigUint64 */:
143 |             return v => new BigUint64Array(v);
144 |     }
145 | }
146 | const messagePortMap = new Map();
147 | function makeTransferable(v, ep) {
148 |     if (v instanceof MessagePort) {
149 |         const uid = generateUID();
150 |         messagePortMap.set(uid, ep);
151 |         const wrapped = wrap(ep, uid);
152 |         addMessageListener(v, ({ data }) => {
153 |             wrapped.postMessage(data, findAllTransferables(data).map(v => v.value));
154 |         });
155 |         v.start();
156 |         addMessageListener(wrapped, ({ data }) => {
157 |             v.postMessage(data, findAllTransferables(data).map(v => v.value));
158 |         });
159 |         return {
160 |             type: 0 /* MessagePort */,
161 |             path: [],
162 |             value: uid
163 |         };
164 |     }
165 |     else if (v instanceof ArrayBuffer || ArrayBuffer.isView(v)) {
166 |         const buffer = v.buffer || v;
167 |         return {
168 |             type: 1 /* TypedArray */,
169 |             subtype: getTypedArrayType(v),
170 |             path: [],
171 |             value: hexEncode(buffer)
172 |         };
173 |     }
174 |     throw Error("Not transferable");
175 | }
176 | function isTransferable(v) {
177 |     if (v instanceof MessagePort) {
178 |         return true;
179 |     }
180 |     if (v instanceof ArrayBuffer) {
181 |         return true;
182 |     }
183 |     if (ArrayBuffer.isView(v)) {
184 |         return true;
185 |     }
186 |     return false;
187 | }
188 | function deserializeTransferable(transfer, ep) {
189 |     switch (transfer.type) {
190 |         case 0 /* MessagePort */:
191 |             const port = wrap(ep, transfer.value);
192 |             return [port, port];
193 |         case 1 /* TypedArray */:
194 |             const constructor = getTypedViewConstructor(transfer.subtype);
195 |             const buffer = hexDecode(transfer.value);
196 |             return [constructor(buffer), buffer];
197 |         default:
198 |             throw Error("Unknown transferable");
199 |     }
200 | }
201 | function wrap(ep, uid = "") {
202 |     const { port1, port2 } = new MessageChannel();
203 |     ep.addMessageListener(msg => {
204 |         let payload;
205 |         payload = JSON.parse(msg);
206 |         if (payload.uid !== uid) {
207 |             return;
208 |         }
209 |         const transferables = payload.transfer
210 |             .map(transfer => {
211 |             const [replacement, transferable] = deserializeTransferable(transfer, ep);
212 |             replaceValueAtPath(payload.data, transfer.path, replacement);
213 |             return transferable;
214 |         })
215 |             .filter(Boolean);
216 |         port2.postMessage(payload.data, transferables);
217 |     });
218 |     addMessageListener(port2, ({ data }) => {
219 |         const transfer = [];
220 |         for (const { path } of findAllTransferables(data)) {
221 |             const oldValue = replaceValueAtPath(data, path, null);
222 |             const serializedTransferable = makeTransferable(oldValue, ep);
223 |             serializedTransferable.path = path;
224 |             transfer.push(serializedTransferable);
225 |         }
226 |         ep.send(JSON.stringify({ uid, data, transfer }));
227 |     });
228 |     port2.start();
229 |     return port1;
230 | }
231 | exports.wrap = wrap;
232 | 


--------------------------------------------------------------------------------
/server/string-channel.ts:
--------------------------------------------------------------------------------
  1 | /**
  2 |  * Copyright 2019 Google Inc. All Rights Reserved.
  3 |  * Licensed under the Apache License, Version 2.0 (the "License");
  4 |  * you may not use this file except in compliance with the License.
  5 |  * You may obtain a copy of the License at
  6 |  *     http://www.apache.org/licenses/LICENSE-2.0
  7 |  * Unless required by applicable law or agreed to in writing, software
  8 |  * distributed under the License is distributed on an "AS IS" BASIS,
  9 |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 10 |  * See the License for the specific language governing permissions and
 11 |  * limitations under the License.
 12 |  */
 13 | 
 14 | export type StringListener = (msg: string) => void;
 15 | export interface StringChannelEndpoint {
 16 |   addMessageListener(listener: StringListener): void;
 17 |   send(msg: string): void;
 18 | }
 19 | 
 20 | function generateUID(): string {
 21 |   return new Array(4)
 22 |     .fill(0)
 23 |     .map(() => Math.floor(Math.random() * Number.MAX_SAFE_INTEGER).toString(16))
 24 |     .join("-");
 25 | }
 26 | 
 27 | interface PropertyIteratorEntry {
 28 |   value: T;
 29 |   path: string[];
 30 | }
 31 | 
 32 | function* iterateAllProperties(
 33 |   value: {} | undefined,
 34 |   path: string[] = [],
 35 |   visited: WeakSet<{}> = new WeakSet<{}>()
 36 | ): Iterable {
 37 |   if (value === undefined) return;
 38 |   if (visited.has(value)) return;
 39 |   if (typeof value === "string") return;
 40 |   if (typeof value === "object") visited.add(value);
 41 |   yield { value, path };
 42 | 
 43 |   // Emit these objects, but don’t traverse them.
 44 |   if (ArrayBuffer.isView(value)) return;
 45 |   if (value instanceof ArrayBuffer) return;
 46 | 
 47 |   const keys = Object.keys(value);
 48 |   for (const key of keys) {
 49 |     yield* iterateAllProperties((value as any)[key], [...path, key], visited);
 50 |   }
 51 | }
 52 | 
 53 | function findAllTransferables(obj: {}): PropertyIteratorEntry[] {
 54 |   const transferables: PropertyIteratorEntry[] = [];
 55 |   for (const entry of iterateAllProperties(obj)) {
 56 |     if (isTransferable(entry.value)) {
 57 |       transferables.push(entry as any);
 58 |     }
 59 |   }
 60 |   return transferables;
 61 | }
 62 | 
 63 | function replaceValueAtPath(value: any, path: string[], newValue: any): {} {
 64 |   const lastProp = path[path.length - 1];
 65 |   for (const prop of path.slice(0, -1)) {
 66 |     value = value[prop];
 67 |   }
 68 |   const oldValue = value[lastProp];
 69 |   value[lastProp] = newValue;
 70 |   return oldValue;
 71 | }
 72 | 
 73 | function padLeft(str: string, pad: string, length: number): string {
 74 |   return (pad.repeat(length) + str).slice(-length);
 75 | }
 76 | 
 77 | function hexEncode(buffer: ArrayBuffer): string {
 78 |   return [...new Uint8Array(buffer)]
 79 |     .map(v => padLeft(v.toString(16), "0", 2))
 80 |     .join("");
 81 | }
 82 | 
 83 | function hexDecode(s: string): ArrayBuffer {
 84 |   return new Uint8Array(
 85 |     s
 86 |       .split(/(..)/)
 87 |       .filter(Boolean)
 88 |       .map(v => parseInt(v, 16))
 89 |   ).buffer;
 90 | }
 91 | 
 92 | function addMessageListener(
 93 |   target: EventTarget,
 94 |   listener: (ev: MessageEvent) => void
 95 | ) {
 96 |   if ("on" in target) {
 97 |     return (target as any).on("message", (data: any) =>
 98 |       listener({ data } as any)
 99 |     );
100 |   }
101 |   target.addEventListener("message", listener as any);
102 | }
103 | 
104 | const enum SerializedTransferableType {
105 |   MessagePort,
106 |   TypedArray
107 | }
108 | 
109 | const enum TypedArrayType {
110 |   Raw,
111 |   Int8,
112 |   Uint8,
113 |   Uint8Clamped,
114 |   Int16,
115 |   Uint16,
116 |   Int32,
117 |   Uint32,
118 |   Float32,
119 |   Float64,
120 |   BigInt64,
121 |   BigUint64
122 | }
123 | 
124 | interface SerializedTransferableTypedArray {
125 |   type: SerializedTransferableType.TypedArray;
126 |   subtype: TypedArrayType;
127 |   path: string[];
128 |   value: string;
129 | }
130 | 
131 | function getTypedArrayType(v: ArrayBuffer | ArrayBufferView): TypedArrayType {
132 |   if (!ArrayBuffer.isView(v)) {
133 |     return TypedArrayType.Raw;
134 |   }
135 |   if (v instanceof Int8Array) {
136 |     return TypedArrayType.Int8;
137 |   }
138 |   if (v instanceof Uint8Array) {
139 |     return TypedArrayType.Uint8;
140 |   }
141 |   if (v instanceof Uint8ClampedArray) {
142 |     return TypedArrayType.Uint8Clamped;
143 |   }
144 |   if (v instanceof Int16Array) {
145 |     return TypedArrayType.Int16;
146 |   }
147 |   if (v instanceof Uint16Array) {
148 |     return TypedArrayType.Uint16;
149 |   }
150 |   if (v instanceof Int32Array) {
151 |     return TypedArrayType.Int32;
152 |   }
153 |   if (v instanceof Uint32Array) {
154 |     return TypedArrayType.Uint32;
155 |   }
156 |   if (v instanceof Float32Array) {
157 |     return TypedArrayType.Float32;
158 |   }
159 |   if (v instanceof Float64Array) {
160 |     return TypedArrayType.Float64;
161 |   }
162 |   if (v instanceof BigInt64Array) {
163 |     return TypedArrayType.BigInt64;
164 |   }
165 |   if (v instanceof BigUint64Array) {
166 |     return TypedArrayType.BigUint64;
167 |   }
168 |   throw Error("Unknown ArrayBufferView type");
169 | }
170 | 
171 | function getTypedViewConstructor(
172 |   type: TypedArrayType
173 | ): (v: ArrayBuffer) => ArrayBufferView | ArrayBuffer {
174 |   switch (type) {
175 |     case TypedArrayType.Raw:
176 |       return v => v;
177 |     case TypedArrayType.Int8:
178 |       return v => new Int8Array(v);
179 |     case TypedArrayType.Uint8:
180 |       return v => new Uint8Array(v);
181 |     case TypedArrayType.Uint8Clamped:
182 |       return v => new Uint8ClampedArray(v);
183 |     case TypedArrayType.Int16:
184 |       return v => new Int16Array(v);
185 |     case TypedArrayType.Uint16:
186 |       return v => new Uint16Array(v);
187 |     case TypedArrayType.Int32:
188 |       return v => new Int32Array(v);
189 |     case TypedArrayType.Uint32:
190 |       return v => new Uint32Array(v);
191 |     case TypedArrayType.Float32:
192 |       return v => new Float32Array(v);
193 |     case TypedArrayType.Float64:
194 |       return v => new Float64Array(v);
195 |     case TypedArrayType.BigInt64:
196 |       return v => new BigInt64Array(v);
197 |     case TypedArrayType.BigUint64:
198 |       return v => new BigUint64Array(v);
199 |   }
200 | }
201 | 
202 | interface SerializedTransferableMessagePort {
203 |   type: SerializedTransferableType.MessagePort;
204 |   path: string[];
205 |   value: string;
206 | }
207 | 
208 | type SerializedTransferable =
209 |   | SerializedTransferableMessagePort
210 |   | SerializedTransferableTypedArray;
211 | const messagePortMap = new Map();
212 | function makeTransferable(
213 |   v: unknown,
214 |   ep: StringChannelEndpoint
215 | ): SerializedTransferable {
216 |   if (v instanceof MessagePort) {
217 |     const uid = generateUID();
218 |     messagePortMap.set(uid, ep);
219 |     const wrapped = wrap(ep, uid);
220 |     addMessageListener(v, ({ data }) => {
221 |       wrapped.postMessage(
222 |         data,
223 |         findAllTransferables(data).map(v => v.value)
224 |       );
225 |     });
226 |     v.start();
227 |     addMessageListener(wrapped, ({ data }) => {
228 |       v.postMessage(
229 |         data,
230 |         findAllTransferables(data).map(v => v.value)
231 |       );
232 |     });
233 | 
234 |     return {
235 |       type: SerializedTransferableType.MessagePort,
236 |       path: [],
237 |       value: uid
238 |     };
239 |   } else if (v instanceof ArrayBuffer || ArrayBuffer.isView(v)) {
240 |     const buffer = (v as any).buffer || v;
241 |     return {
242 |       type: SerializedTransferableType.TypedArray,
243 |       subtype: getTypedArrayType(v),
244 |       path: [],
245 |       value: hexEncode(buffer)
246 |     };
247 |   }
248 |   throw Error("Not transferable");
249 | }
250 | 
251 | function isTransferable(v: any): v is Transferable {
252 |   if (v instanceof MessagePort) {
253 |     return true;
254 |   }
255 |   if (v instanceof ArrayBuffer) {
256 |     return true;
257 |   }
258 |   if (ArrayBuffer.isView(v)) {
259 |     return true;
260 |   }
261 |   return false;
262 | }
263 | 
264 | function deserializeTransferable(
265 |   transfer: SerializedTransferable,
266 |   ep: StringChannelEndpoint
267 | ): [any, Transferable] {
268 |   switch (transfer.type) {
269 |     case SerializedTransferableType.MessagePort:
270 |       const port = wrap(ep, transfer.value);
271 |       return [port, port];
272 |     case SerializedTransferableType.TypedArray:
273 |       const constructor = getTypedViewConstructor(transfer.subtype);
274 |       const buffer = hexDecode(transfer.value);
275 |       return [constructor(buffer), buffer];
276 |     default:
277 |       throw Error("Unknown transferable");
278 |   }
279 | }
280 | 
281 | interface StringChannelPayload {
282 |   uid: string;
283 |   data: any;
284 |   transfer: SerializedTransferable[];
285 | }
286 | 
287 | export function wrap(ep: StringChannelEndpoint, uid = "") {
288 |   const { port1, port2 } = new MessageChannel();
289 | 
290 |   ep.addMessageListener(msg => {
291 |     let payload: StringChannelPayload;
292 |     payload = JSON.parse(msg);
293 |     if (payload.uid !== uid) {
294 |       return;
295 |     }
296 |     const transferables: Transferable[] = payload.transfer
297 |       .map(transfer => {
298 |         const [replacement, transferable] = deserializeTransferable(
299 |           transfer,
300 |           ep
301 |         );
302 |         replaceValueAtPath(payload.data, transfer.path, replacement);
303 |         return transferable;
304 |       })
305 |       .filter(Boolean);
306 |     port2.postMessage(payload.data, transferables);
307 |   });
308 | 
309 |   addMessageListener(port2, ({ data }) => {
310 |     const transfer: SerializedTransferable[] = [];
311 |     for (const { path } of findAllTransferables(data)) {
312 |       const oldValue = replaceValueAtPath(data, path, null);
313 |       const serializedTransferable = makeTransferable(oldValue, ep);
314 |       serializedTransferable.path = path;
315 |       transfer.push(serializedTransferable);
316 |     }
317 |     ep.send(JSON.stringify({ uid, data, transfer }));
318 |   });
319 |   port2.start();
320 |   return port1;
321 | }


--------------------------------------------------------------------------------