├── .dockerignore ├── .eslintrc.json ├── .gitignore ├── Dockerfile ├── LICENSE ├── README.md ├── app.js ├── config ├── config-sample.yaml └── email-config-schema.yaml ├── package-lock.json ├── package.json └── src ├── bridge.js ├── email ├── email.js ├── index.js └── outbound.js └── log.js /.dockerignore: -------------------------------------------------------------------------------- 1 | config/config-sample.yaml 2 | 3 | # Dependency directories 4 | node_modules/ 5 | 6 | # Optional npm cache directory 7 | .npm 8 | 9 | # npm debug logs 10 | npm-debug.log 11 | 12 | # dotenv environment variables file 13 | .env 14 | 15 | # db files 16 | *.db 17 | 18 | # AS registration config 19 | email-registration.yaml 20 | 21 | #editor configs 22 | .vscode 23 | .idea 24 | -------------------------------------------------------------------------------- /.eslintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "parserOptions": { 3 | "ecmaVersion": 9, 4 | "ecmaFeatures": { 5 | "jsx": false 6 | } 7 | }, 8 | "env": { 9 | "node": true, 10 | "jasmine": true 11 | }, 12 | "rules": { 13 | "consistent-return": "error", 14 | "curly": "error", 15 | "default-case": "error", 16 | "guard-for-in": "error", 17 | "no-alert": "error", 18 | "no-caller": "error", 19 | "no-cond-assign": "error", 20 | "no-constant-condition": "error", 21 | "no-debugger": "error", 22 | "no-dupe-args": "error", 23 | "no-dupe-keys": "error", 24 | "no-duplicate-case": "error", 25 | "no-else-return": "error", 26 | "no-empty": "error", 27 | "no-empty-character-class": "error", 28 | "no-eq-null": "error", 29 | "no-ex-assign": "error", 30 | "no-extend-native": "error", 31 | "no-extra-boolean-cast": "error", 32 | "no-extra-semi": "error", 33 | "no-fallthrough": "error", 34 | "no-func-assign": "error", 35 | "no-invalid-regexp": "error", 36 | "no-invalid-this": "error", 37 | "no-irregular-whitespace": "error", 38 | "no-lone-blocks": "error", 39 | "no-loop-func": "error", 40 | "no-multi-spaces": "error", 41 | "no-new-wrappers": "error", 42 | "no-new": "error", 43 | "no-octal": "error", 44 | "no-negated-in-lhs": "error", 45 | "no-obj-calls": "error", 46 | "no-redeclare": "error", 47 | "no-regex-spaces": "error", 48 | "no-return-assign": "error", 49 | "no-self-compare": "error", 50 | "no-unreachable": "error", 51 | "no-unexpected-multiline": "error", 52 | "no-unused-expressions": "error", 53 | "no-use-before-define": "off", 54 | "use-isnan": "error", 55 | "valid-typeof": "error", 56 | "array-bracket-spacing": ["error", "never"], 57 | "max-len": ["error", 120], 58 | "brace-style": ["error", "stroustrup", { "allowSingleLine": true }], 59 | "comma-spacing": ["error", {"before": false, "after": true}], 60 | "comma-style": ["error", "last"], 61 | "computed-property-spacing": ["error", "never"], 62 | "consistent-this": ["error", "self"], 63 | "eol-last": "error", 64 | "new-cap": "error", 65 | "new-parens": "error", 66 | "no-mixed-spaces-and-tabs": "error", 67 | "no-nested-ternary": "error", 68 | "no-spaced-func": "error", 69 | "no-trailing-spaces": "error", 70 | "keyword-spacing": ["error", {"before": true, "after": true}], 71 | "space-before-blocks": ["error", "always"], 72 | "camelcase": ["error", { "properties": "never" }], 73 | "no-unused-vars": "off", 74 | "strict": ["error", "never" ], 75 | "no-var": "error", 76 | "object-curly-spacing": ["error", "always"], 77 | "semi": ["error", "always"] 78 | } 79 | } 80 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Dependency directories 2 | node_modules/ 3 | 4 | # Optional npm cache directory 5 | .npm 6 | 7 | # npm debug logs 8 | npm-debug.log 9 | 10 | # dotenv environment variables file 11 | .env 12 | 13 | # db files 14 | *.db 15 | 16 | # AS registration config 17 | email-registration.yaml 18 | 19 | #editor configs 20 | .vscode 21 | .idea 22 | -------------------------------------------------------------------------------- /Dockerfile: -------------------------------------------------------------------------------- 1 | FROM node:16-slim 2 | 3 | WORKDIR /app 4 | COPY . . 5 | RUN npm ci --cache /tmp/empty-cache 6 | 7 | EXPOSE 25/tcp 8 | EXPOSE 8090/tcp 9 | VOLUME ["/config"] 10 | 11 | CMD [ "node", "app.js", "-c", "/config/config.yaml", "-f", "/config/email-registration.yaml", "-p", "8090" ] 12 | -------------------------------------------------------------------------------- /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 | # Matrix Email Bridge 2 | 3 | First class email bridge for Matrix. 4 | This bridge enables Email users and Matrix users to communicate back and forth. 5 | 6 | [This was made during Google Summer of Code 2021](https://matrix.org/blog/2021/05/20/google-summer-of-code-2021#abhinav-krishna-c-k-first-class-email-bridge) 7 | 8 | 9 | ### Setup: 10 | 11 | To set up the bridge, simply clone this repository. 12 | 13 | 1. `git clone https://github.com/abbyck/matrix-appservice-email.git` 14 | 2. Install the dependencies `cd matrix-appservice-email && npm i` 15 | 3. Generate the Application Service registration file with 16 | `node app.js -r -u "http://localhost:8090" -f /config/email-registration.yaml` (http://localhost:8090 is the URL that the AS will listen to. 17 | 4. Add the registration details to your homeserver configuration by adding the `email-registration.yaml`. 18 | 5. Make a copy of the sample `config/config-sample.yaml` and carefully change the options as required. 19 | 6. Run the bridge! 20 | ```node app.js -c config/config.yaml``` 21 | 22 | #### Recommended options for the bridge. 23 | 24 | Most of the available options are detailed on the sample configuration file. 25 | It's _highly recommended to use_ the following settings to ensure email delivery. 26 | * Correct [MX records](https://www.cloudflare.com/en-in/learning/dns/dns-records/dns-mx-record/) and [SPF records](https://en.wikipedia.org/wiki/Sender_Policy_Framework). 27 | * [DKIM (DomainKeys Identified Mail)](https://en.wikipedia.org/wiki/DomainKeys_Identified_Mail). 28 | - You may use [opendkim-genkey](http://www.opendkim.org/opendkim-genkey.8.html) to generate the DKIM keys. 29 | * TLS Keys for encrypted outbounds to supported servers. 30 | 31 | ### Docker 32 | 33 | A Dockerfile is included. To use it: 34 | 35 | ```sh 36 | # To build a docker image 37 | docker build . -t matrix-appservice-email 38 | # Create a new config file 39 | mkdir data 40 | cp config/config.yaml data/config.yaml 41 | # ...and edit that file. 42 | # Generate a registration file 43 | docker run --rm -v $PWD/data:/config matrix-appservice-email node app.js -r -u "http://localhost:8090" -f /config/email-registration.yaml 44 | # To run with default ports (port 25 for SMTP, port 8090 for bridge traffic) 45 | docker run --rm -v $PWD/data:/config matrix-appservice-email 46 | # To run with custom ports 47 | docker run --rm -v $PWD/data:/config -p 127.0.0.1:25:1111/tcp -p 127.0.0.1:8090:2222/tcp matrix-appservice-email 48 | ``` 49 | 50 | ### Features 51 | * Email users can join public Matrix rooms which has room aliases. 52 | * To join a room, send an email to `room+test_example.com@example.com` and the user would be automatically joined to a 53 | matrix room `#test:example.com` 54 | * To Direct message a user on the matrix side, hit a mail to `user+alice_example.com@example.com` and a DM will be created with the user `@alice:example.com` 55 | * Subsequent replies will be sent to the `from` address. 56 | * They can send and receive messages to Matrix rooms. 57 | * Outbound mails have the ability to get DKIM signed and, 58 | * Use TLS while communicating with supported mail servers (Configuration required). 59 | 60 | #### To be implemented 61 | * [Inbound spam protection](https://github.com/abbyck/matrix-appservice-email/issues/6): Running an email server 62 | publicly means, you have to fight a lot of spam 63 | * [Attachments/File Handling](https://github.com/abbyck/matrix-appservice-email/issues/7) 64 | 65 | See something missing? Hit a message to [@abbyck:matrix.org](https://matrix.to/#/@abbyck:matrix.org). 66 | -------------------------------------------------------------------------------- /app.js: -------------------------------------------------------------------------------- 1 | const { AppServiceRegistration, Cli, } = require("matrix-appservice-bridge"); 2 | const { bridge } = require('./src/bridge'); 3 | 4 | 5 | new Cli({ 6 | registrationPath: "email-registration.yaml", 7 | generateRegistration: function (reg, callback) { 8 | reg.setId(AppServiceRegistration.generateToken()); 9 | reg.setHomeserverToken(AppServiceRegistration.generateToken()); 10 | reg.setAppServiceToken(AppServiceRegistration.generateToken()); 11 | reg.setSenderLocalpart("email"); 12 | reg.addRegexPattern("users", "@_email_.*", true); 13 | callback(reg); 14 | }, 15 | bridgeConfig: { 16 | schema: "config/email-config-schema.yaml" 17 | }, 18 | run: bridge 19 | }).run(); 20 | -------------------------------------------------------------------------------- /config/config-sample.yaml: -------------------------------------------------------------------------------- 1 | # Sample config 2 | bridge: 3 | # Domain part of the bridge, e.g. matrix.org 4 | domain: "localhost" 5 | # HomeServer URL 6 | homeserverUrl: "http://localhost:8008" 7 | # The TCP port on which the appservice runs on. 8 | port: 8090 9 | email: 10 | # MX inbound domain. 11 | # This is the domain to which inbound mails are sent by email users (eg: matrix.org) 12 | mxDomain: "localhost" 13 | # SMTP inbound port(Usually 25). 14 | inboundPort: 2525 15 | # SMTP outbound port(Usually 25). 16 | outboundPort: 2500 17 | # SMTP next hop address (If you want to relay outbound mails through some server. Leave empty, for no relay) 18 | smtpHost: 'localhost' 19 | #DKIM signing options 20 | dkim: 21 | # Set to `true` for enabling DKIM signing 22 | enabled: false 23 | # DKIM private key file location (eg: './dkim-private.pem'). 24 | dkimKey: '' 25 | #DKIM selector 26 | selector: '' 27 | # TLS Key and Cert for Opportunistic TLS upgrades (STARTTLS). 28 | tls: 29 | # Set to `true` for enabling STARTTLS support. 30 | enabled: false 31 | # TLS Key file location. 32 | tlsKey: '' 33 | # TLS Certificate file location. 34 | tlsCert: '' 35 | -------------------------------------------------------------------------------- /config/email-config-schema.yaml: -------------------------------------------------------------------------------- 1 | "$schema": "http://json-schema.org/draft-04/schema#" 2 | type: "object" 3 | required: ["bridge", "email"] 4 | properties: 5 | bridge: 6 | type: "object" 7 | required: ["domain", "homeserverUrl", "port"] 8 | properties: 9 | domain: 10 | type: "string" 11 | homeserverUrl: 12 | type: "string" 13 | port: 14 | type: "number" 15 | email: 16 | type: "object" 17 | required: ["mxDomain", "inboundPort", "outboundPort"] 18 | properties: 19 | mxDomain: 20 | type: "string" 21 | inboundPort: 22 | type: "number" 23 | outboundPort: 24 | type: "number" 25 | smtpHost: 26 | type: "string" 27 | dkim: 28 | type: "object" 29 | properties: 30 | enabled: 31 | type: "boolean" 32 | dkimKey: 33 | type: "string" 34 | selector: 35 | type: "string" 36 | tls: 37 | type: "object" 38 | properties: 39 | enabled: 40 | type: "boolean" 41 | tlsKey: 42 | type: "string" 43 | tlsCert: 44 | type: "string" 45 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "matrix-appservice-email", 3 | "version": "0.0.1", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@babel/code-frame": { 8 | "version": "7.12.11", 9 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.12.11.tgz", 10 | "integrity": "sha512-Zt1yodBx1UcyiePMSkWnU4hPqhwq7hGi2nFL1LeA3EUl+q2LQx16MISgJ0+z7dnmgvP9QtIleuETGOiOH1RcIw==", 11 | "dev": true, 12 | "requires": { 13 | "@babel/highlight": "^7.10.4" 14 | } 15 | }, 16 | "@babel/helper-validator-identifier": { 17 | "version": "7.14.9", 18 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.14.9.tgz", 19 | "integrity": "sha512-pQYxPY0UP6IHISRitNe8bsijHex4TWZXi2HwKVsjPiltzlhse2znVcm9Ace510VT1kxIHjGJCZZQBX2gJDbo0g==", 20 | "dev": true 21 | }, 22 | "@babel/highlight": { 23 | "version": "7.14.5", 24 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.14.5.tgz", 25 | "integrity": "sha512-qf9u2WFWVV0MppaL877j2dBtQIDgmidgjGk5VIMw3OadXvYaXn66U1BFlH2t4+t3i+8PhedppRv+i40ABzd+gg==", 26 | "dev": true, 27 | "requires": { 28 | "@babel/helper-validator-identifier": "^7.14.5", 29 | "chalk": "^2.0.0", 30 | "js-tokens": "^4.0.0" 31 | }, 32 | "dependencies": { 33 | "ansi-styles": { 34 | "version": "3.2.1", 35 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 36 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 37 | "dev": true, 38 | "requires": { 39 | "color-convert": "^1.9.0" 40 | } 41 | }, 42 | "chalk": { 43 | "version": "2.4.2", 44 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 45 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 46 | "dev": true, 47 | "requires": { 48 | "ansi-styles": "^3.2.1", 49 | "escape-string-regexp": "^1.0.5", 50 | "supports-color": "^5.3.0" 51 | } 52 | }, 53 | "color-convert": { 54 | "version": "1.9.3", 55 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 56 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 57 | "dev": true, 58 | "requires": { 59 | "color-name": "1.1.3" 60 | } 61 | }, 62 | "color-name": { 63 | "version": "1.1.3", 64 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 65 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 66 | "dev": true 67 | }, 68 | "escape-string-regexp": { 69 | "version": "1.0.5", 70 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 71 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 72 | "dev": true 73 | }, 74 | "has-flag": { 75 | "version": "3.0.0", 76 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 77 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 78 | "dev": true 79 | }, 80 | "supports-color": { 81 | "version": "5.5.0", 82 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 83 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 84 | "dev": true, 85 | "requires": { 86 | "has-flag": "^3.0.0" 87 | } 88 | } 89 | } 90 | }, 91 | "@babel/runtime": { 92 | "version": "7.15.3", 93 | "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.15.3.tgz", 94 | "integrity": "sha512-OvwMLqNXkCXSz1kSm58sEsNuhqOx/fKpnUnKnFB5v8uDda5bLNEHNgKPvhDN6IU0LDcnHQ90LlJ0Q6jnyBSIBA==", 95 | "requires": { 96 | "regenerator-runtime": "^0.13.4" 97 | } 98 | }, 99 | "@dabh/diagnostics": { 100 | "version": "2.0.2", 101 | "resolved": "https://registry.npmjs.org/@dabh/diagnostics/-/diagnostics-2.0.2.tgz", 102 | "integrity": "sha512-+A1YivoVDNNVCdfozHSR8v/jyuuLTMXwjWuxPFlFlUapXoGc+Gj9mDlTDDfrwl7rXCl2tNZ0kE8sIBO6YOn96Q==", 103 | "requires": { 104 | "colorspace": "1.1.x", 105 | "enabled": "2.0.x", 106 | "kuler": "^2.0.0" 107 | } 108 | }, 109 | "@eslint/eslintrc": { 110 | "version": "0.4.3", 111 | "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-0.4.3.tgz", 112 | "integrity": "sha512-J6KFFz5QCYUJq3pf0mjEcCJVERbzv71PUIDczuh9JkwGEzced6CO5ADLHB1rbf/+oPBtoPfMYNOpGDzCANlbXw==", 113 | "dev": true, 114 | "requires": { 115 | "ajv": "^6.12.4", 116 | "debug": "^4.1.1", 117 | "espree": "^7.3.0", 118 | "globals": "^13.9.0", 119 | "ignore": "^4.0.6", 120 | "import-fresh": "^3.2.1", 121 | "js-yaml": "^3.13.1", 122 | "minimatch": "^3.0.4", 123 | "strip-json-comments": "^3.1.1" 124 | } 125 | }, 126 | "@humanwhocodes/config-array": { 127 | "version": "0.5.0", 128 | "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.5.0.tgz", 129 | "integrity": "sha512-FagtKFz74XrTl7y6HCzQpwDfXP0yhxe9lHLD1UZxjvZIcbyRz8zTFF/yYNfSfzU414eDwZ1SrO0Qvtyf+wFMQg==", 130 | "dev": true, 131 | "requires": { 132 | "@humanwhocodes/object-schema": "^1.2.0", 133 | "debug": "^4.1.1", 134 | "minimatch": "^3.0.4" 135 | } 136 | }, 137 | "@humanwhocodes/object-schema": { 138 | "version": "1.2.0", 139 | "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.0.tgz", 140 | "integrity": "sha512-wdppn25U8z/2yiaT6YGquE6X8sSv7hNMWSXYSSU1jGv/yd6XqjXgTDJ8KP4NgjTXfJ3GbRjeeb8RTV7a/VpM+w==", 141 | "dev": true 142 | }, 143 | "@selderee/plugin-htmlparser2": { 144 | "version": "0.6.0", 145 | "resolved": "https://registry.npmjs.org/@selderee/plugin-htmlparser2/-/plugin-htmlparser2-0.6.0.tgz", 146 | "integrity": "sha512-J3jpy002TyBjd4N/p6s+s90eX42H2eRhK3SbsZuvTDv977/E8p2U3zikdiehyJja66do7FlxLomZLPlvl2/xaA==", 147 | "requires": { 148 | "domhandler": "^4.2.0", 149 | "selderee": "^0.6.0" 150 | } 151 | }, 152 | "@types/body-parser": { 153 | "version": "1.19.1", 154 | "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.1.tgz", 155 | "integrity": "sha512-a6bTJ21vFOGIkwM0kzh9Yr89ziVxq4vYH2fQ6N8AeipEzai/cFK6aGMArIkUeIdRIgpwQa+2bXiLuUJCpSf2Cg==", 156 | "requires": { 157 | "@types/connect": "*", 158 | "@types/node": "*" 159 | } 160 | }, 161 | "@types/connect": { 162 | "version": "3.4.35", 163 | "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.35.tgz", 164 | "integrity": "sha512-cdeYyv4KWoEgpBISTxWvqYsVy444DOqehiF3fM3ne10AmJ62RSyNkUnxMJXHQWRQQX2eR94m5y1IZyDwBjV9FQ==", 165 | "requires": { 166 | "@types/node": "*" 167 | } 168 | }, 169 | "@types/express": { 170 | "version": "4.17.13", 171 | "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.13.tgz", 172 | "integrity": "sha512-6bSZTPaTIACxn48l50SR+axgrqm6qXFIxrdAKaG6PaJk3+zuUr35hBlgT7vOmJcum+OEaIBLtHV/qloEAFITeA==", 173 | "requires": { 174 | "@types/body-parser": "*", 175 | "@types/express-serve-static-core": "^4.17.18", 176 | "@types/qs": "*", 177 | "@types/serve-static": "*" 178 | } 179 | }, 180 | "@types/express-serve-static-core": { 181 | "version": "4.17.24", 182 | "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.24.tgz", 183 | "integrity": "sha512-3UJuW+Qxhzwjq3xhwXm2onQcFHn76frIYVbTu+kn24LFxI+dEhdfISDFovPB8VpEgW8oQCTpRuCe+0zJxB7NEA==", 184 | "requires": { 185 | "@types/node": "*", 186 | "@types/qs": "*", 187 | "@types/range-parser": "*" 188 | } 189 | }, 190 | "@types/mime": { 191 | "version": "1.3.2", 192 | "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.2.tgz", 193 | "integrity": "sha512-YATxVxgRqNH6nHEIsvg6k2Boc1JHI9ZbH5iWFFv/MTkchz3b1ieGDa5T0a9RznNdI0KhVbdbWSN+KWWrQZRxTw==" 194 | }, 195 | "@types/node": { 196 | "version": "16.7.1", 197 | "resolved": "https://registry.npmjs.org/@types/node/-/node-16.7.1.tgz", 198 | "integrity": "sha512-ncRdc45SoYJ2H4eWU9ReDfp3vtFqDYhjOsKlFFUDEn8V1Bgr2RjYal8YT5byfadWIRluhPFU6JiDOl0H6Sl87A==" 199 | }, 200 | "@types/qs": { 201 | "version": "6.9.7", 202 | "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.7.tgz", 203 | "integrity": "sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw==" 204 | }, 205 | "@types/range-parser": { 206 | "version": "1.2.4", 207 | "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.4.tgz", 208 | "integrity": "sha512-EEhsLsD6UsDM1yFhAvy0Cjr6VwmpMWqFBCb9w07wVugF7w9nfajxLuVmngTIpgS6svCnm6Vaw+MZhoDCKnOfsw==" 209 | }, 210 | "@types/serve-static": { 211 | "version": "1.13.10", 212 | "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.13.10.tgz", 213 | "integrity": "sha512-nCkHGI4w7ZgAdNkrEu0bv+4xNV/XDqW+DydknebMOQwkpDGx8G+HTlj7R7ABI8i8nKxVw0wtKPi1D+lPOkh4YQ==", 214 | "requires": { 215 | "@types/mime": "^1", 216 | "@types/node": "*" 217 | } 218 | }, 219 | "abbrev": { 220 | "version": "1.1.1", 221 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 222 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==" 223 | }, 224 | "accepts": { 225 | "version": "1.3.7", 226 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 227 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 228 | "requires": { 229 | "mime-types": "~2.1.24", 230 | "negotiator": "0.6.2" 231 | } 232 | }, 233 | "acorn": { 234 | "version": "7.4.1", 235 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", 236 | "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", 237 | "dev": true 238 | }, 239 | "acorn-jsx": { 240 | "version": "5.3.2", 241 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", 242 | "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", 243 | "dev": true 244 | }, 245 | "ajv": { 246 | "version": "6.12.6", 247 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", 248 | "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", 249 | "requires": { 250 | "fast-deep-equal": "^3.1.1", 251 | "fast-json-stable-stringify": "^2.0.0", 252 | "json-schema-traverse": "^0.4.1", 253 | "uri-js": "^4.2.2" 254 | } 255 | }, 256 | "another-json": { 257 | "version": "0.2.0", 258 | "resolved": "https://registry.npmjs.org/another-json/-/another-json-0.2.0.tgz", 259 | "integrity": "sha1-tfQBnJc7bdXGUGotk0acttMq7tw=" 260 | }, 261 | "ansi-colors": { 262 | "version": "4.1.1", 263 | "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", 264 | "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", 265 | "dev": true 266 | }, 267 | "ansi-regex": { 268 | "version": "5.0.0", 269 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 270 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 271 | "dev": true 272 | }, 273 | "ansi-styles": { 274 | "version": "4.3.0", 275 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 276 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 277 | "requires": { 278 | "color-convert": "^2.0.1" 279 | } 280 | }, 281 | "argparse": { 282 | "version": "1.0.10", 283 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 284 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 285 | "requires": { 286 | "sprintf-js": "~1.0.2" 287 | } 288 | }, 289 | "array-flatten": { 290 | "version": "1.1.1", 291 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 292 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 293 | }, 294 | "asn1": { 295 | "version": "0.2.4", 296 | "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", 297 | "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", 298 | "requires": { 299 | "safer-buffer": "~2.1.0" 300 | } 301 | }, 302 | "assert-plus": { 303 | "version": "1.0.0", 304 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 305 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 306 | }, 307 | "astral-regex": { 308 | "version": "2.0.0", 309 | "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz", 310 | "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==", 311 | "dev": true 312 | }, 313 | "async": { 314 | "version": "0.2.10", 315 | "resolved": "https://registry.npmjs.org/async/-/async-0.2.10.tgz", 316 | "integrity": "sha1-trvgsGdLnXGXCMo43owjfLUmw9E=" 317 | }, 318 | "asynckit": { 319 | "version": "0.4.0", 320 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 321 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" 322 | }, 323 | "aws-sign2": { 324 | "version": "0.7.0", 325 | "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", 326 | "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=" 327 | }, 328 | "aws4": { 329 | "version": "1.11.0", 330 | "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.11.0.tgz", 331 | "integrity": "sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA==" 332 | }, 333 | "balanced-match": { 334 | "version": "1.0.2", 335 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 336 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 337 | "dev": true 338 | }, 339 | "base-x": { 340 | "version": "3.0.8", 341 | "resolved": "https://registry.npmjs.org/base-x/-/base-x-3.0.8.tgz", 342 | "integrity": "sha512-Rl/1AWP4J/zRrk54hhlxH4drNxPJXYUaKffODVI53/dAsV4t9fBxyxYKAVPU1XBHxYwOWP9h9H0hM2MVw4YfJA==", 343 | "requires": { 344 | "safe-buffer": "^5.0.1" 345 | } 346 | }, 347 | "base32.js": { 348 | "version": "0.1.0", 349 | "resolved": "https://registry.npmjs.org/base32.js/-/base32.js-0.1.0.tgz", 350 | "integrity": "sha1-tYLexpPC8R6JPPBk7mrFthMaIgI=" 351 | }, 352 | "basic-auth": { 353 | "version": "2.0.1", 354 | "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz", 355 | "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==", 356 | "requires": { 357 | "safe-buffer": "5.1.2" 358 | } 359 | }, 360 | "bcrypt-pbkdf": { 361 | "version": "1.0.2", 362 | "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", 363 | "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", 364 | "requires": { 365 | "tweetnacl": "^0.14.3" 366 | } 367 | }, 368 | "binary-search-tree": { 369 | "version": "0.2.5", 370 | "resolved": "https://registry.npmjs.org/binary-search-tree/-/binary-search-tree-0.2.5.tgz", 371 | "integrity": "sha1-fbs7IQ/coIJFDa0jNMMErzm9x4Q=", 372 | "requires": { 373 | "underscore": "~1.4.4" 374 | } 375 | }, 376 | "bintrees": { 377 | "version": "1.0.1", 378 | "resolved": "https://registry.npmjs.org/bintrees/-/bintrees-1.0.1.tgz", 379 | "integrity": "sha1-DmVcm5wkNeqraL9AJyJtK1WjRSQ=" 380 | }, 381 | "body-parser": { 382 | "version": "1.19.0", 383 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", 384 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", 385 | "requires": { 386 | "bytes": "3.1.0", 387 | "content-type": "~1.0.4", 388 | "debug": "2.6.9", 389 | "depd": "~1.1.2", 390 | "http-errors": "1.7.2", 391 | "iconv-lite": "0.4.24", 392 | "on-finished": "~2.3.0", 393 | "qs": "6.7.0", 394 | "raw-body": "2.4.0", 395 | "type-is": "~1.6.17" 396 | }, 397 | "dependencies": { 398 | "debug": { 399 | "version": "2.6.9", 400 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 401 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 402 | "requires": { 403 | "ms": "2.0.0" 404 | } 405 | }, 406 | "iconv-lite": { 407 | "version": "0.4.24", 408 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 409 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 410 | "requires": { 411 | "safer-buffer": ">= 2.1.2 < 3" 412 | } 413 | }, 414 | "ms": { 415 | "version": "2.0.0", 416 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 417 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 418 | } 419 | } 420 | }, 421 | "brace-expansion": { 422 | "version": "1.1.11", 423 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 424 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 425 | "dev": true, 426 | "requires": { 427 | "balanced-match": "^1.0.0", 428 | "concat-map": "0.0.1" 429 | } 430 | }, 431 | "browser-request": { 432 | "version": "0.3.3", 433 | "resolved": "https://registry.npmjs.org/browser-request/-/browser-request-0.3.3.tgz", 434 | "integrity": "sha1-ns5bWsqJopkyJC4Yv5M975h2zBc=" 435 | }, 436 | "bs58": { 437 | "version": "4.0.1", 438 | "resolved": "https://registry.npmjs.org/bs58/-/bs58-4.0.1.tgz", 439 | "integrity": "sha1-vhYedsNU9veIrkBx9j806MTwpCo=", 440 | "requires": { 441 | "base-x": "^3.0.2" 442 | } 443 | }, 444 | "bytes": { 445 | "version": "3.1.0", 446 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", 447 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" 448 | }, 449 | "call-bind": { 450 | "version": "1.0.2", 451 | "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", 452 | "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", 453 | "requires": { 454 | "function-bind": "^1.1.1", 455 | "get-intrinsic": "^1.0.2" 456 | } 457 | }, 458 | "callsites": { 459 | "version": "3.1.0", 460 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", 461 | "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", 462 | "dev": true 463 | }, 464 | "caseless": { 465 | "version": "0.12.0", 466 | "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", 467 | "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=" 468 | }, 469 | "chalk": { 470 | "version": "4.1.2", 471 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", 472 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", 473 | "requires": { 474 | "ansi-styles": "^4.1.0", 475 | "supports-color": "^7.1.0" 476 | } 477 | }, 478 | "color": { 479 | "version": "3.0.0", 480 | "resolved": "https://registry.npmjs.org/color/-/color-3.0.0.tgz", 481 | "integrity": "sha512-jCpd5+s0s0t7p3pHQKpnJ0TpQKKdleP71LWcA0aqiljpiuAkOSUFN/dyH8ZwF0hRmFlrIuRhufds1QyEP9EB+w==", 482 | "requires": { 483 | "color-convert": "^1.9.1", 484 | "color-string": "^1.5.2" 485 | }, 486 | "dependencies": { 487 | "color-convert": { 488 | "version": "1.9.3", 489 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 490 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 491 | "requires": { 492 | "color-name": "1.1.3" 493 | } 494 | }, 495 | "color-name": { 496 | "version": "1.1.3", 497 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 498 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" 499 | } 500 | } 501 | }, 502 | "color-convert": { 503 | "version": "2.0.1", 504 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 505 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 506 | "requires": { 507 | "color-name": "~1.1.4" 508 | } 509 | }, 510 | "color-name": { 511 | "version": "1.1.4", 512 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 513 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" 514 | }, 515 | "color-string": { 516 | "version": "1.5.5", 517 | "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.5.5.tgz", 518 | "integrity": "sha512-jgIoum0OfQfq9Whcfc2z/VhCNcmQjWbey6qBX0vqt7YICflUmBCh9E9CiQD5GSJ+Uehixm3NUwHVhqUAWRivZg==", 519 | "requires": { 520 | "color-name": "^1.0.0", 521 | "simple-swizzle": "^0.2.2" 522 | } 523 | }, 524 | "colors": { 525 | "version": "1.4.0", 526 | "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz", 527 | "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==" 528 | }, 529 | "colorspace": { 530 | "version": "1.1.2", 531 | "resolved": "https://registry.npmjs.org/colorspace/-/colorspace-1.1.2.tgz", 532 | "integrity": "sha512-vt+OoIP2d76xLhjwbBaucYlNSpPsrJWPlBTtwCpQKIu6/CSMutyzX93O/Do0qzpH3YoHEes8YEFXyZ797rEhzQ==", 533 | "requires": { 534 | "color": "3.0.x", 535 | "text-hex": "1.0.x" 536 | } 537 | }, 538 | "combined-stream": { 539 | "version": "1.0.8", 540 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 541 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 542 | "requires": { 543 | "delayed-stream": "~1.0.0" 544 | } 545 | }, 546 | "commander": { 547 | "version": "2.20.3", 548 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", 549 | "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==" 550 | }, 551 | "concat-map": { 552 | "version": "0.0.1", 553 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 554 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 555 | "dev": true 556 | }, 557 | "content-disposition": { 558 | "version": "0.5.3", 559 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", 560 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", 561 | "requires": { 562 | "safe-buffer": "5.1.2" 563 | } 564 | }, 565 | "content-type": { 566 | "version": "1.0.4", 567 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 568 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 569 | }, 570 | "cookie": { 571 | "version": "0.4.0", 572 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 573 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 574 | }, 575 | "cookie-signature": { 576 | "version": "1.0.6", 577 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 578 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 579 | }, 580 | "core-util-is": { 581 | "version": "1.0.2", 582 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 583 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 584 | }, 585 | "cross-spawn": { 586 | "version": "7.0.3", 587 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", 588 | "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", 589 | "dev": true, 590 | "requires": { 591 | "path-key": "^3.1.0", 592 | "shebang-command": "^2.0.0", 593 | "which": "^2.0.1" 594 | } 595 | }, 596 | "cycle": { 597 | "version": "1.0.3", 598 | "resolved": "https://registry.npmjs.org/cycle/-/cycle-1.0.3.tgz", 599 | "integrity": "sha1-IegLK+hYD5i0aPN5QwZisEbDStI=" 600 | }, 601 | "dashdash": { 602 | "version": "1.14.1", 603 | "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", 604 | "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", 605 | "requires": { 606 | "assert-plus": "^1.0.0" 607 | } 608 | }, 609 | "debug": { 610 | "version": "4.3.2", 611 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.2.tgz", 612 | "integrity": "sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw==", 613 | "dev": true, 614 | "requires": { 615 | "ms": "2.1.2" 616 | } 617 | }, 618 | "deep-is": { 619 | "version": "0.1.3", 620 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", 621 | "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", 622 | "dev": true 623 | }, 624 | "deepmerge": { 625 | "version": "4.2.2", 626 | "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.2.2.tgz", 627 | "integrity": "sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg==" 628 | }, 629 | "delayed-stream": { 630 | "version": "1.0.0", 631 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 632 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=" 633 | }, 634 | "depd": { 635 | "version": "1.1.2", 636 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 637 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 638 | }, 639 | "destroy": { 640 | "version": "1.0.4", 641 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 642 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 643 | }, 644 | "discontinuous-range": { 645 | "version": "1.0.0", 646 | "resolved": "https://registry.npmjs.org/discontinuous-range/-/discontinuous-range-1.0.0.tgz", 647 | "integrity": "sha1-44Mx8IRLukm5qctxx3FYWqsbxlo=" 648 | }, 649 | "dkim-signer": { 650 | "version": "0.2.2", 651 | "resolved": "https://registry.npmjs.org/dkim-signer/-/dkim-signer-0.2.2.tgz", 652 | "integrity": "sha1-qoHsBx7u02IngbqpIgRNeADl8wg=", 653 | "requires": { 654 | "libmime": "^2.0.3" 655 | }, 656 | "dependencies": { 657 | "iconv-lite": { 658 | "version": "0.4.15", 659 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.15.tgz", 660 | "integrity": "sha1-/iZaIYrGpXz+hUkn6dBMGYJe3es=" 661 | }, 662 | "libbase64": { 663 | "version": "0.1.0", 664 | "resolved": "https://registry.npmjs.org/libbase64/-/libbase64-0.1.0.tgz", 665 | "integrity": "sha1-YjUag5VjrF/1vSbxL2Dpgwu3UeY=" 666 | }, 667 | "libmime": { 668 | "version": "2.1.3", 669 | "resolved": "https://registry.npmjs.org/libmime/-/libmime-2.1.3.tgz", 670 | "integrity": "sha1-JQF8pataHpiq2+JyUBfPHUikKgw=", 671 | "requires": { 672 | "iconv-lite": "0.4.15", 673 | "libbase64": "0.1.0", 674 | "libqp": "1.1.0" 675 | } 676 | } 677 | } 678 | }, 679 | "doctrine": { 680 | "version": "3.0.0", 681 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", 682 | "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", 683 | "dev": true, 684 | "requires": { 685 | "esutils": "^2.0.2" 686 | } 687 | }, 688 | "dom-serializer": { 689 | "version": "1.3.2", 690 | "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-1.3.2.tgz", 691 | "integrity": "sha512-5c54Bk5Dw4qAxNOI1pFEizPSjVsx5+bpJKmL2kPn8JhBUq2q09tTCa3mjijun2NfK78NMouDYNMBkOrPZiS+ig==", 692 | "requires": { 693 | "domelementtype": "^2.0.1", 694 | "domhandler": "^4.2.0", 695 | "entities": "^2.0.0" 696 | } 697 | }, 698 | "domelementtype": { 699 | "version": "2.2.0", 700 | "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.2.0.tgz", 701 | "integrity": "sha512-DtBMo82pv1dFtUmHyr48beiuq792Sxohr+8Hm9zoxklYPfa6n0Z3Byjj2IV7bmr2IyqClnqEQhfgHJJ5QF0R5A==" 702 | }, 703 | "domhandler": { 704 | "version": "4.2.0", 705 | "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-4.2.0.tgz", 706 | "integrity": "sha512-zk7sgt970kzPks2Bf+dwT/PLzghLnsivb9CcxkvR8Mzr66Olr0Ofd8neSbglHJHaHa2MadfoSdNlKYAaafmWfA==", 707 | "requires": { 708 | "domelementtype": "^2.2.0" 709 | } 710 | }, 711 | "domutils": { 712 | "version": "2.7.0", 713 | "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.7.0.tgz", 714 | "integrity": "sha512-8eaHa17IwJUPAiB+SoTYBo5mCdeMgdcAoXJ59m6DT1vw+5iLS3gNoqYaRowaBKtGVrOF1Jz4yDTgYKLK2kvfJg==", 715 | "requires": { 716 | "dom-serializer": "^1.0.1", 717 | "domelementtype": "^2.2.0", 718 | "domhandler": "^4.2.0" 719 | } 720 | }, 721 | "ecc-jsbn": { 722 | "version": "0.1.2", 723 | "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", 724 | "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", 725 | "requires": { 726 | "jsbn": "~0.1.0", 727 | "safer-buffer": "^2.1.0" 728 | } 729 | }, 730 | "ee-first": { 731 | "version": "1.1.1", 732 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 733 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 734 | }, 735 | "email-addresses": { 736 | "version": "4.0.0", 737 | "resolved": "https://registry.npmjs.org/email-addresses/-/email-addresses-4.0.0.tgz", 738 | "integrity": "sha512-Nas3sSSiD5lSIoqBos0FMjB9h4clHxXuAahHKGJ5doRWavEB7pBHzOxnI7R5f1MuGNrrSnsZFJ81HCBv0DZmnw==" 739 | }, 740 | "emoji-regex": { 741 | "version": "8.0.0", 742 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 743 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 744 | "dev": true 745 | }, 746 | "enabled": { 747 | "version": "2.0.0", 748 | "resolved": "https://registry.npmjs.org/enabled/-/enabled-2.0.0.tgz", 749 | "integrity": "sha512-AKrN98kuwOzMIdAizXGI86UFBoo26CL21UM763y1h/GMSJ4/OHU9k2YlsmBpyScFo/wbLzWQJBMCW4+IO3/+OQ==" 750 | }, 751 | "encodeurl": { 752 | "version": "1.0.2", 753 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 754 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 755 | }, 756 | "encoding-japanese": { 757 | "version": "1.0.30", 758 | "resolved": "https://registry.npmjs.org/encoding-japanese/-/encoding-japanese-1.0.30.tgz", 759 | "integrity": "sha512-bd/DFLAoJetvv7ar/KIpE3CNO8wEuyrt9Xuw6nSMiZ+Vrz/Q21BPsMHvARL2Wz6IKHKXgb+DWZqtRg1vql9cBg==" 760 | }, 761 | "enquirer": { 762 | "version": "2.3.6", 763 | "resolved": "https://registry.npmjs.org/enquirer/-/enquirer-2.3.6.tgz", 764 | "integrity": "sha512-yjNnPr315/FjS4zIsUxYguYUPP2e1NK4d7E7ZOLiyYCcbFBiTMyID+2wvm2w6+pZ/odMA7cRkjhsPbltwBOrLg==", 765 | "dev": true, 766 | "requires": { 767 | "ansi-colors": "^4.1.1" 768 | } 769 | }, 770 | "entities": { 771 | "version": "2.2.0", 772 | "resolved": "https://registry.npmjs.org/entities/-/entities-2.2.0.tgz", 773 | "integrity": "sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==" 774 | }, 775 | "escape-html": { 776 | "version": "1.0.3", 777 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 778 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 779 | }, 780 | "escape-string-regexp": { 781 | "version": "4.0.0", 782 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", 783 | "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", 784 | "dev": true 785 | }, 786 | "eslint": { 787 | "version": "7.32.0", 788 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-7.32.0.tgz", 789 | "integrity": "sha512-VHZ8gX+EDfz+97jGcgyGCyRia/dPOd6Xh9yPv8Bl1+SoaIwD+a/vlrOmGRUyOYu7MwUhc7CxqeaDZU13S4+EpA==", 790 | "dev": true, 791 | "requires": { 792 | "@babel/code-frame": "7.12.11", 793 | "@eslint/eslintrc": "^0.4.3", 794 | "@humanwhocodes/config-array": "^0.5.0", 795 | "ajv": "^6.10.0", 796 | "chalk": "^4.0.0", 797 | "cross-spawn": "^7.0.2", 798 | "debug": "^4.0.1", 799 | "doctrine": "^3.0.0", 800 | "enquirer": "^2.3.5", 801 | "escape-string-regexp": "^4.0.0", 802 | "eslint-scope": "^5.1.1", 803 | "eslint-utils": "^2.1.0", 804 | "eslint-visitor-keys": "^2.0.0", 805 | "espree": "^7.3.1", 806 | "esquery": "^1.4.0", 807 | "esutils": "^2.0.2", 808 | "fast-deep-equal": "^3.1.3", 809 | "file-entry-cache": "^6.0.1", 810 | "functional-red-black-tree": "^1.0.1", 811 | "glob-parent": "^5.1.2", 812 | "globals": "^13.6.0", 813 | "ignore": "^4.0.6", 814 | "import-fresh": "^3.0.0", 815 | "imurmurhash": "^0.1.4", 816 | "is-glob": "^4.0.0", 817 | "js-yaml": "^3.13.1", 818 | "json-stable-stringify-without-jsonify": "^1.0.1", 819 | "levn": "^0.4.1", 820 | "lodash.merge": "^4.6.2", 821 | "minimatch": "^3.0.4", 822 | "natural-compare": "^1.4.0", 823 | "optionator": "^0.9.1", 824 | "progress": "^2.0.0", 825 | "regexpp": "^3.1.0", 826 | "semver": "^7.2.1", 827 | "strip-ansi": "^6.0.0", 828 | "strip-json-comments": "^3.1.0", 829 | "table": "^6.0.9", 830 | "text-table": "^0.2.0", 831 | "v8-compile-cache": "^2.0.3" 832 | } 833 | }, 834 | "eslint-scope": { 835 | "version": "5.1.1", 836 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", 837 | "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", 838 | "dev": true, 839 | "requires": { 840 | "esrecurse": "^4.3.0", 841 | "estraverse": "^4.1.1" 842 | } 843 | }, 844 | "eslint-utils": { 845 | "version": "2.1.0", 846 | "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", 847 | "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", 848 | "dev": true, 849 | "requires": { 850 | "eslint-visitor-keys": "^1.1.0" 851 | }, 852 | "dependencies": { 853 | "eslint-visitor-keys": { 854 | "version": "1.3.0", 855 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", 856 | "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", 857 | "dev": true 858 | } 859 | } 860 | }, 861 | "eslint-visitor-keys": { 862 | "version": "2.1.0", 863 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz", 864 | "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==", 865 | "dev": true 866 | }, 867 | "espree": { 868 | "version": "7.3.1", 869 | "resolved": "https://registry.npmjs.org/espree/-/espree-7.3.1.tgz", 870 | "integrity": "sha512-v3JCNCE64umkFpmkFGqzVKsOT0tN1Zr+ueqLZfpV1Ob8e+CEgPWa+OxCoGH3tnhimMKIaBm4m/vaRpJ/krRz2g==", 871 | "dev": true, 872 | "requires": { 873 | "acorn": "^7.4.0", 874 | "acorn-jsx": "^5.3.1", 875 | "eslint-visitor-keys": "^1.3.0" 876 | }, 877 | "dependencies": { 878 | "eslint-visitor-keys": { 879 | "version": "1.3.0", 880 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", 881 | "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", 882 | "dev": true 883 | } 884 | } 885 | }, 886 | "esprima": { 887 | "version": "4.0.1", 888 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 889 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==" 890 | }, 891 | "esquery": { 892 | "version": "1.4.0", 893 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.4.0.tgz", 894 | "integrity": "sha512-cCDispWt5vHHtwMY2YrAQ4ibFkAL8RbH5YGBnZBc90MolvvfkkQcJro/aZiAQUlQ3qgrYS6D6v8Gc5G5CQsc9w==", 895 | "dev": true, 896 | "requires": { 897 | "estraverse": "^5.1.0" 898 | }, 899 | "dependencies": { 900 | "estraverse": { 901 | "version": "5.2.0", 902 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", 903 | "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", 904 | "dev": true 905 | } 906 | } 907 | }, 908 | "esrecurse": { 909 | "version": "4.3.0", 910 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", 911 | "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", 912 | "dev": true, 913 | "requires": { 914 | "estraverse": "^5.2.0" 915 | }, 916 | "dependencies": { 917 | "estraverse": { 918 | "version": "5.2.0", 919 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", 920 | "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", 921 | "dev": true 922 | } 923 | } 924 | }, 925 | "estraverse": { 926 | "version": "4.3.0", 927 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", 928 | "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", 929 | "dev": true 930 | }, 931 | "esutils": { 932 | "version": "2.0.3", 933 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", 934 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", 935 | "dev": true 936 | }, 937 | "etag": { 938 | "version": "1.8.1", 939 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 940 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 941 | }, 942 | "eventemitter3": { 943 | "version": "4.0.7", 944 | "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz", 945 | "integrity": "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==" 946 | }, 947 | "express": { 948 | "version": "4.17.1", 949 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", 950 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", 951 | "requires": { 952 | "accepts": "~1.3.7", 953 | "array-flatten": "1.1.1", 954 | "body-parser": "1.19.0", 955 | "content-disposition": "0.5.3", 956 | "content-type": "~1.0.4", 957 | "cookie": "0.4.0", 958 | "cookie-signature": "1.0.6", 959 | "debug": "2.6.9", 960 | "depd": "~1.1.2", 961 | "encodeurl": "~1.0.2", 962 | "escape-html": "~1.0.3", 963 | "etag": "~1.8.1", 964 | "finalhandler": "~1.1.2", 965 | "fresh": "0.5.2", 966 | "merge-descriptors": "1.0.1", 967 | "methods": "~1.1.2", 968 | "on-finished": "~2.3.0", 969 | "parseurl": "~1.3.3", 970 | "path-to-regexp": "0.1.7", 971 | "proxy-addr": "~2.0.5", 972 | "qs": "6.7.0", 973 | "range-parser": "~1.2.1", 974 | "safe-buffer": "5.1.2", 975 | "send": "0.17.1", 976 | "serve-static": "1.14.1", 977 | "setprototypeof": "1.1.1", 978 | "statuses": "~1.5.0", 979 | "type-is": "~1.6.18", 980 | "utils-merge": "1.0.1", 981 | "vary": "~1.1.2" 982 | }, 983 | "dependencies": { 984 | "debug": { 985 | "version": "2.6.9", 986 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 987 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 988 | "requires": { 989 | "ms": "2.0.0" 990 | } 991 | }, 992 | "ms": { 993 | "version": "2.0.0", 994 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 995 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 996 | } 997 | } 998 | }, 999 | "extend": { 1000 | "version": "3.0.2", 1001 | "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", 1002 | "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==" 1003 | }, 1004 | "extsprintf": { 1005 | "version": "1.3.0", 1006 | "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", 1007 | "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=" 1008 | }, 1009 | "fast-deep-equal": { 1010 | "version": "3.1.3", 1011 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", 1012 | "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==" 1013 | }, 1014 | "fast-json-stable-stringify": { 1015 | "version": "2.1.0", 1016 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", 1017 | "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==" 1018 | }, 1019 | "fast-levenshtein": { 1020 | "version": "2.0.6", 1021 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", 1022 | "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", 1023 | "dev": true 1024 | }, 1025 | "fast-safe-stringify": { 1026 | "version": "2.0.7", 1027 | "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.0.7.tgz", 1028 | "integrity": "sha512-Utm6CdzT+6xsDk2m8S6uL8VHxNwI6Jub+e9NYTcAms28T84pTa25GJQV9j0CY0N1rM8hK4x6grpF2BQf+2qwVA==" 1029 | }, 1030 | "fecha": { 1031 | "version": "4.2.1", 1032 | "resolved": "https://registry.npmjs.org/fecha/-/fecha-4.2.1.tgz", 1033 | "integrity": "sha512-MMMQ0ludy/nBs1/o0zVOiKTpG7qMbonKUzjJgQFEuvq6INZ1OraKPRAWkBq5vlKLOUMpmNYG1JoN3oDPUQ9m3Q==" 1034 | }, 1035 | "file-entry-cache": { 1036 | "version": "6.0.1", 1037 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", 1038 | "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", 1039 | "dev": true, 1040 | "requires": { 1041 | "flat-cache": "^3.0.4" 1042 | } 1043 | }, 1044 | "file-stream-rotator": { 1045 | "version": "0.2.1", 1046 | "resolved": "https://registry.npmjs.org/file-stream-rotator/-/file-stream-rotator-0.2.1.tgz", 1047 | "integrity": "sha1-DW/qGpp6uiWofP0xtuJp5E6PCvI=", 1048 | "requires": { 1049 | "moment": "^2.11.2" 1050 | } 1051 | }, 1052 | "finalhandler": { 1053 | "version": "1.1.2", 1054 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", 1055 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", 1056 | "requires": { 1057 | "debug": "2.6.9", 1058 | "encodeurl": "~1.0.2", 1059 | "escape-html": "~1.0.3", 1060 | "on-finished": "~2.3.0", 1061 | "parseurl": "~1.3.3", 1062 | "statuses": "~1.5.0", 1063 | "unpipe": "~1.0.0" 1064 | }, 1065 | "dependencies": { 1066 | "debug": { 1067 | "version": "2.6.9", 1068 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 1069 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 1070 | "requires": { 1071 | "ms": "2.0.0" 1072 | } 1073 | }, 1074 | "ms": { 1075 | "version": "2.0.0", 1076 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1077 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 1078 | } 1079 | } 1080 | }, 1081 | "flat-cache": { 1082 | "version": "3.0.4", 1083 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz", 1084 | "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==", 1085 | "dev": true, 1086 | "requires": { 1087 | "flatted": "^3.1.0", 1088 | "rimraf": "^3.0.2" 1089 | } 1090 | }, 1091 | "flatted": { 1092 | "version": "3.2.2", 1093 | "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.2.tgz", 1094 | "integrity": "sha512-JaTY/wtrcSyvXJl4IMFHPKyFur1sE9AUqc0QnhOaJ0CxHtAoIV8pYDzeEfAaNEtGkOfq4gr3LBFmdXW5mOQFnA==", 1095 | "dev": true 1096 | }, 1097 | "fn.name": { 1098 | "version": "1.1.0", 1099 | "resolved": "https://registry.npmjs.org/fn.name/-/fn.name-1.1.0.tgz", 1100 | "integrity": "sha512-GRnmB5gPyJpAhTQdSZTSp9uaPSvl09KoYcMQtsB9rQoOmzs9dH6ffeccH+Z+cv6P68Hu5bC6JjRh4Ah/mHSNRw==" 1101 | }, 1102 | "forever-agent": { 1103 | "version": "0.6.1", 1104 | "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", 1105 | "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=" 1106 | }, 1107 | "form-data": { 1108 | "version": "2.3.3", 1109 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", 1110 | "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", 1111 | "requires": { 1112 | "asynckit": "^0.4.0", 1113 | "combined-stream": "^1.0.6", 1114 | "mime-types": "^2.1.12" 1115 | } 1116 | }, 1117 | "forwarded": { 1118 | "version": "0.2.0", 1119 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 1120 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==" 1121 | }, 1122 | "fresh": { 1123 | "version": "0.5.2", 1124 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 1125 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 1126 | }, 1127 | "fs.realpath": { 1128 | "version": "1.0.0", 1129 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 1130 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", 1131 | "dev": true 1132 | }, 1133 | "function-bind": { 1134 | "version": "1.1.1", 1135 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 1136 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" 1137 | }, 1138 | "functional-red-black-tree": { 1139 | "version": "1.0.1", 1140 | "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", 1141 | "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", 1142 | "dev": true 1143 | }, 1144 | "generate-function": { 1145 | "version": "2.3.1", 1146 | "resolved": "https://registry.npmjs.org/generate-function/-/generate-function-2.3.1.tgz", 1147 | "integrity": "sha512-eeB5GfMNeevm/GRYq20ShmsaGcmI81kIX2K9XQx5miC8KdHaC6Jm0qQ8ZNeGOi7wYB8OsdxKs+Y2oVuTFuVwKQ==", 1148 | "requires": { 1149 | "is-property": "^1.0.2" 1150 | } 1151 | }, 1152 | "generate-object-property": { 1153 | "version": "1.2.0", 1154 | "resolved": "https://registry.npmjs.org/generate-object-property/-/generate-object-property-1.2.0.tgz", 1155 | "integrity": "sha1-nA4cQDCM6AT0eDYYuTf6iPmdUNA=", 1156 | "requires": { 1157 | "is-property": "^1.0.0" 1158 | } 1159 | }, 1160 | "get-intrinsic": { 1161 | "version": "1.1.1", 1162 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.1.tgz", 1163 | "integrity": "sha512-kWZrnVM42QCiEA2Ig1bG8zjoIMOgxWwYCEeNdwY6Tv/cOSeGpcoX4pXHfKUxNKVoArnrEr2e9srnAxxGIraS9Q==", 1164 | "requires": { 1165 | "function-bind": "^1.1.1", 1166 | "has": "^1.0.3", 1167 | "has-symbols": "^1.0.1" 1168 | } 1169 | }, 1170 | "getpass": { 1171 | "version": "0.1.7", 1172 | "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", 1173 | "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", 1174 | "requires": { 1175 | "assert-plus": "^1.0.0" 1176 | } 1177 | }, 1178 | "glob": { 1179 | "version": "7.1.7", 1180 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz", 1181 | "integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==", 1182 | "dev": true, 1183 | "requires": { 1184 | "fs.realpath": "^1.0.0", 1185 | "inflight": "^1.0.4", 1186 | "inherits": "2", 1187 | "minimatch": "^3.0.4", 1188 | "once": "^1.3.0", 1189 | "path-is-absolute": "^1.0.0" 1190 | } 1191 | }, 1192 | "glob-parent": { 1193 | "version": "5.1.2", 1194 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 1195 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 1196 | "dev": true, 1197 | "requires": { 1198 | "is-glob": "^4.0.1" 1199 | } 1200 | }, 1201 | "globals": { 1202 | "version": "13.11.0", 1203 | "resolved": "https://registry.npmjs.org/globals/-/globals-13.11.0.tgz", 1204 | "integrity": "sha512-08/xrJ7wQjK9kkkRoI3OFUBbLx4f+6x3SGwcPvQ0QH6goFDrOU2oyAWrmh3dJezu65buo+HBMzAMQy6rovVC3g==", 1205 | "dev": true, 1206 | "requires": { 1207 | "type-fest": "^0.20.2" 1208 | } 1209 | }, 1210 | "har-schema": { 1211 | "version": "2.0.0", 1212 | "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", 1213 | "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=" 1214 | }, 1215 | "har-validator": { 1216 | "version": "5.1.5", 1217 | "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz", 1218 | "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==", 1219 | "requires": { 1220 | "ajv": "^6.12.3", 1221 | "har-schema": "^2.0.0" 1222 | } 1223 | }, 1224 | "has": { 1225 | "version": "1.0.3", 1226 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 1227 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 1228 | "requires": { 1229 | "function-bind": "^1.1.1" 1230 | } 1231 | }, 1232 | "has-flag": { 1233 | "version": "4.0.0", 1234 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 1235 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" 1236 | }, 1237 | "has-symbols": { 1238 | "version": "1.0.2", 1239 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.2.tgz", 1240 | "integrity": "sha512-chXa79rL/UC2KlX17jo3vRGz0azaWEx5tGqZg5pO3NUyEJVB17dMruQlzCCOfUvElghKcm5194+BCRvi2Rv/Gw==" 1241 | }, 1242 | "he": { 1243 | "version": "1.2.0", 1244 | "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", 1245 | "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==" 1246 | }, 1247 | "html-to-text": { 1248 | "version": "8.0.0", 1249 | "resolved": "https://registry.npmjs.org/html-to-text/-/html-to-text-8.0.0.tgz", 1250 | "integrity": "sha512-fEtul1OerF2aMEV+Wpy+Ue20tug134jOY1GIudtdqZi7D0uTudB2tVJBKfVhTL03dtqeJoF8gk8EPX9SyMEvLg==", 1251 | "requires": { 1252 | "@selderee/plugin-htmlparser2": "^0.6.0", 1253 | "deepmerge": "^4.2.2", 1254 | "he": "^1.2.0", 1255 | "htmlparser2": "^6.1.0", 1256 | "minimist": "^1.2.5", 1257 | "selderee": "^0.6.0" 1258 | } 1259 | }, 1260 | "htmlparser2": { 1261 | "version": "6.1.0", 1262 | "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-6.1.0.tgz", 1263 | "integrity": "sha512-gyyPk6rgonLFEDGoeRgQNaEUvdJ4ktTmmUh/h2t7s+M8oPpIPxgNACWa+6ESR57kXstwqPiCut0V8NRpcwgU7A==", 1264 | "requires": { 1265 | "domelementtype": "^2.0.1", 1266 | "domhandler": "^4.0.0", 1267 | "domutils": "^2.5.2", 1268 | "entities": "^2.0.0" 1269 | } 1270 | }, 1271 | "http-errors": { 1272 | "version": "1.7.2", 1273 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 1274 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 1275 | "requires": { 1276 | "depd": "~1.1.2", 1277 | "inherits": "2.0.3", 1278 | "setprototypeof": "1.1.1", 1279 | "statuses": ">= 1.5.0 < 2", 1280 | "toidentifier": "1.0.0" 1281 | } 1282 | }, 1283 | "http-signature": { 1284 | "version": "1.2.0", 1285 | "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", 1286 | "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", 1287 | "requires": { 1288 | "assert-plus": "^1.0.0", 1289 | "jsprim": "^1.2.2", 1290 | "sshpk": "^1.7.0" 1291 | } 1292 | }, 1293 | "iconv-lite": { 1294 | "version": "0.6.3", 1295 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", 1296 | "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", 1297 | "requires": { 1298 | "safer-buffer": ">= 2.1.2 < 3.0.0" 1299 | } 1300 | }, 1301 | "ignore": { 1302 | "version": "4.0.6", 1303 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", 1304 | "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", 1305 | "dev": true 1306 | }, 1307 | "immediate": { 1308 | "version": "3.0.6", 1309 | "resolved": "https://registry.npmjs.org/immediate/-/immediate-3.0.6.tgz", 1310 | "integrity": "sha1-nbHb0Pr43m++D13V5Wu2BigN5ps=" 1311 | }, 1312 | "import-fresh": { 1313 | "version": "3.3.0", 1314 | "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", 1315 | "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", 1316 | "dev": true, 1317 | "requires": { 1318 | "parent-module": "^1.0.0", 1319 | "resolve-from": "^4.0.0" 1320 | } 1321 | }, 1322 | "imurmurhash": { 1323 | "version": "0.1.4", 1324 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 1325 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 1326 | "dev": true 1327 | }, 1328 | "inflight": { 1329 | "version": "1.0.6", 1330 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 1331 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 1332 | "dev": true, 1333 | "requires": { 1334 | "once": "^1.3.0", 1335 | "wrappy": "1" 1336 | } 1337 | }, 1338 | "inherits": { 1339 | "version": "2.0.3", 1340 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 1341 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 1342 | }, 1343 | "ipaddr.js": { 1344 | "version": "1.9.1", 1345 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 1346 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" 1347 | }, 1348 | "ipv6-normalize": { 1349 | "version": "1.0.1", 1350 | "resolved": "https://registry.npmjs.org/ipv6-normalize/-/ipv6-normalize-1.0.1.tgz", 1351 | "integrity": "sha1-GzJYKQ02X6gyOeiZB93kWS52IKg=" 1352 | }, 1353 | "is-arrayish": { 1354 | "version": "0.3.2", 1355 | "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz", 1356 | "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==" 1357 | }, 1358 | "is-extglob": { 1359 | "version": "2.1.1", 1360 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 1361 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 1362 | "dev": true 1363 | }, 1364 | "is-fullwidth-code-point": { 1365 | "version": "3.0.0", 1366 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 1367 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 1368 | "dev": true 1369 | }, 1370 | "is-glob": { 1371 | "version": "4.0.1", 1372 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 1373 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 1374 | "dev": true, 1375 | "requires": { 1376 | "is-extglob": "^2.1.1" 1377 | } 1378 | }, 1379 | "is-my-ip-valid": { 1380 | "version": "1.0.0", 1381 | "resolved": "https://registry.npmjs.org/is-my-ip-valid/-/is-my-ip-valid-1.0.0.tgz", 1382 | "integrity": "sha512-gmh/eWXROncUzRnIa1Ubrt5b8ep/MGSnfAUI3aRp+sqTCs1tv1Isl8d8F6JmkN3dXKc3ehZMrtiPN9eL03NuaQ==" 1383 | }, 1384 | "is-my-json-valid": { 1385 | "version": "2.20.5", 1386 | "resolved": "https://registry.npmjs.org/is-my-json-valid/-/is-my-json-valid-2.20.5.tgz", 1387 | "integrity": "sha512-VTPuvvGQtxvCeghwspQu1rBgjYUT6FGxPlvFKbYuFtgc4ADsX3U5ihZOYN0qyU6u+d4X9xXb0IT5O6QpXKt87A==", 1388 | "requires": { 1389 | "generate-function": "^2.0.0", 1390 | "generate-object-property": "^1.1.0", 1391 | "is-my-ip-valid": "^1.0.0", 1392 | "jsonpointer": "^4.0.0", 1393 | "xtend": "^4.0.0" 1394 | } 1395 | }, 1396 | "is-property": { 1397 | "version": "1.0.2", 1398 | "resolved": "https://registry.npmjs.org/is-property/-/is-property-1.0.2.tgz", 1399 | "integrity": "sha1-V/4cTkhHTt1lsJkR8msc1Ald2oQ=" 1400 | }, 1401 | "is-stream": { 1402 | "version": "2.0.0", 1403 | "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz", 1404 | "integrity": "sha512-XCoy+WlUr7d1+Z8GgSuXmpuUFC9fOhRXglJMx+dwLKTkL44Cjd4W1Z5P+BQZpr+cR93aGP4S/s7Ftw6Nd/kiEw==" 1405 | }, 1406 | "is-typedarray": { 1407 | "version": "1.0.0", 1408 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 1409 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=" 1410 | }, 1411 | "isarray": { 1412 | "version": "1.0.0", 1413 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 1414 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 1415 | }, 1416 | "isexe": { 1417 | "version": "2.0.0", 1418 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 1419 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 1420 | "dev": true 1421 | }, 1422 | "isstream": { 1423 | "version": "0.1.2", 1424 | "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", 1425 | "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=" 1426 | }, 1427 | "js-tokens": { 1428 | "version": "4.0.0", 1429 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 1430 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 1431 | "dev": true 1432 | }, 1433 | "js-yaml": { 1434 | "version": "3.14.1", 1435 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", 1436 | "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", 1437 | "requires": { 1438 | "argparse": "^1.0.7", 1439 | "esprima": "^4.0.0" 1440 | } 1441 | }, 1442 | "jsbn": { 1443 | "version": "0.1.1", 1444 | "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", 1445 | "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=" 1446 | }, 1447 | "json-schema": { 1448 | "version": "0.2.3", 1449 | "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", 1450 | "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=" 1451 | }, 1452 | "json-schema-traverse": { 1453 | "version": "0.4.1", 1454 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", 1455 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==" 1456 | }, 1457 | "json-stable-stringify-without-jsonify": { 1458 | "version": "1.0.1", 1459 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", 1460 | "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", 1461 | "dev": true 1462 | }, 1463 | "json-stringify-safe": { 1464 | "version": "5.0.1", 1465 | "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", 1466 | "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=" 1467 | }, 1468 | "jsonpointer": { 1469 | "version": "4.1.0", 1470 | "resolved": "https://registry.npmjs.org/jsonpointer/-/jsonpointer-4.1.0.tgz", 1471 | "integrity": "sha512-CXcRvMyTlnR53xMcKnuMzfCA5i/nfblTnnr74CZb6C4vG39eu6w51t7nKmU5MfLfbTgGItliNyjO/ciNPDqClg==" 1472 | }, 1473 | "jsprim": { 1474 | "version": "1.4.1", 1475 | "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", 1476 | "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", 1477 | "requires": { 1478 | "assert-plus": "1.0.0", 1479 | "extsprintf": "1.3.0", 1480 | "json-schema": "0.2.3", 1481 | "verror": "1.10.0" 1482 | } 1483 | }, 1484 | "kuler": { 1485 | "version": "2.0.0", 1486 | "resolved": "https://registry.npmjs.org/kuler/-/kuler-2.0.0.tgz", 1487 | "integrity": "sha512-Xq9nH7KlWZmXAtodXDDRE7vs6DU1gTU8zYDHDiWLSip45Egwq3plLHzPn27NgvzL2r1LMPC1vdqh98sQxtqj4A==" 1488 | }, 1489 | "levn": { 1490 | "version": "0.4.1", 1491 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", 1492 | "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", 1493 | "dev": true, 1494 | "requires": { 1495 | "prelude-ls": "^1.2.1", 1496 | "type-check": "~0.4.0" 1497 | } 1498 | }, 1499 | "libbase64": { 1500 | "version": "1.2.1", 1501 | "resolved": "https://registry.npmjs.org/libbase64/-/libbase64-1.2.1.tgz", 1502 | "integrity": "sha512-l+nePcPbIG1fNlqMzrh68MLkX/gTxk/+vdvAb388Ssi7UuUN31MI44w4Yf33mM3Cm4xDfw48mdf3rkdHszLNew==" 1503 | }, 1504 | "libmime": { 1505 | "version": "5.0.0", 1506 | "resolved": "https://registry.npmjs.org/libmime/-/libmime-5.0.0.tgz", 1507 | "integrity": "sha512-2Bm96d5ktnE217Ib1FldvUaPAaOst6GtZrsxJCwnJgi9lnsoAKIHyU0sae8rNx6DNYbjdqqh8lv5/b9poD8qOg==", 1508 | "requires": { 1509 | "encoding-japanese": "1.0.30", 1510 | "iconv-lite": "0.6.2", 1511 | "libbase64": "1.2.1", 1512 | "libqp": "1.1.0" 1513 | }, 1514 | "dependencies": { 1515 | "iconv-lite": { 1516 | "version": "0.6.2", 1517 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.2.tgz", 1518 | "integrity": "sha512-2y91h5OpQlolefMPmUlivelittSWy0rP+oYVpn6A7GwVHNE8AWzoYOBNmlwks3LobaJxgHCYZAnyNo2GgpNRNQ==", 1519 | "requires": { 1520 | "safer-buffer": ">= 2.1.2 < 3.0.0" 1521 | } 1522 | } 1523 | } 1524 | }, 1525 | "libqp": { 1526 | "version": "1.1.0", 1527 | "resolved": "https://registry.npmjs.org/libqp/-/libqp-1.1.0.tgz", 1528 | "integrity": "sha1-9ebgatdLeU+1tbZpiL9yjvHe2+g=" 1529 | }, 1530 | "lie": { 1531 | "version": "3.1.1", 1532 | "resolved": "https://registry.npmjs.org/lie/-/lie-3.1.1.tgz", 1533 | "integrity": "sha1-mkNrLMd0bKWd56QfpGmz77dr2H4=", 1534 | "requires": { 1535 | "immediate": "~3.0.5" 1536 | } 1537 | }, 1538 | "linkify-it": { 1539 | "version": "3.0.2", 1540 | "resolved": "https://registry.npmjs.org/linkify-it/-/linkify-it-3.0.2.tgz", 1541 | "integrity": "sha512-gDBO4aHNZS6coiZCKVhSNh43F9ioIL4JwRjLZPkoLIY4yZFwg264Y5lu2x6rb1Js42Gh6Yqm2f6L2AJcnkzinQ==", 1542 | "requires": { 1543 | "uc.micro": "^1.0.1" 1544 | } 1545 | }, 1546 | "localforage": { 1547 | "version": "1.10.0", 1548 | "resolved": "https://registry.npmjs.org/localforage/-/localforage-1.10.0.tgz", 1549 | "integrity": "sha512-14/H1aX7hzBBmmh7sGPd+AOMkkIrHM3Z1PAyGgZigA1H1p5O5ANnMyWzvpAETtG68/dC4pC0ncy3+PPGzXZHPg==", 1550 | "requires": { 1551 | "lie": "3.1.1" 1552 | } 1553 | }, 1554 | "lodash.clonedeep": { 1555 | "version": "4.5.0", 1556 | "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz", 1557 | "integrity": "sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8=", 1558 | "dev": true 1559 | }, 1560 | "lodash.merge": { 1561 | "version": "4.6.2", 1562 | "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", 1563 | "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", 1564 | "dev": true 1565 | }, 1566 | "lodash.truncate": { 1567 | "version": "4.4.2", 1568 | "resolved": "https://registry.npmjs.org/lodash.truncate/-/lodash.truncate-4.4.2.tgz", 1569 | "integrity": "sha1-WjUNoLERO4N+z//VgSy+WNbq4ZM=", 1570 | "dev": true 1571 | }, 1572 | "logform": { 1573 | "version": "2.2.0", 1574 | "resolved": "https://registry.npmjs.org/logform/-/logform-2.2.0.tgz", 1575 | "integrity": "sha512-N0qPlqfypFx7UHNn4B3lzS/b0uLqt2hmuoa+PpuXNYgozdJYAyauF5Ky0BWVjrxDlMWiT3qN4zPq3vVAfZy7Yg==", 1576 | "requires": { 1577 | "colors": "^1.2.1", 1578 | "fast-safe-stringify": "^2.0.4", 1579 | "fecha": "^4.2.0", 1580 | "ms": "^2.1.1", 1581 | "triple-beam": "^1.3.0" 1582 | }, 1583 | "dependencies": { 1584 | "ms": { 1585 | "version": "2.1.3", 1586 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1587 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 1588 | } 1589 | } 1590 | }, 1591 | "loglevel": { 1592 | "version": "1.7.1", 1593 | "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.7.1.tgz", 1594 | "integrity": "sha512-Hesni4s5UkWkwCGJMQGAh71PaLUmKFM60dHvq0zi/vDhhrzuk+4GgNbTXJ12YYQJn6ZKBDNIjYcuQGKudvqrIw==" 1595 | }, 1596 | "lru-cache": { 1597 | "version": "6.0.0", 1598 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", 1599 | "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", 1600 | "dev": true, 1601 | "requires": { 1602 | "yallist": "^4.0.0" 1603 | } 1604 | }, 1605 | "mailparser": { 1606 | "version": "3.3.0", 1607 | "resolved": "https://registry.npmjs.org/mailparser/-/mailparser-3.3.0.tgz", 1608 | "integrity": "sha512-cSvqT3y735gN3IFLcda/H/grJ+7HItKw8OOshzlhAHAic5R0c33GB1ZizREWmLJONjFMrKUitFeopZd9rsRfvg==", 1609 | "requires": { 1610 | "encoding-japanese": "1.0.30", 1611 | "he": "1.2.0", 1612 | "html-to-text": "8.0.0", 1613 | "iconv-lite": "0.6.3", 1614 | "libmime": "5.0.0", 1615 | "linkify-it": "3.0.2", 1616 | "mailsplit": "5.0.1", 1617 | "nodemailer": "6.6.3", 1618 | "tlds": "1.221.1" 1619 | } 1620 | }, 1621 | "mailsplit": { 1622 | "version": "5.0.1", 1623 | "resolved": "https://registry.npmjs.org/mailsplit/-/mailsplit-5.0.1.tgz", 1624 | "integrity": "sha512-CcGy1sv8j9jdjKiNIuMZYIKhq4s47nUj9Q98BZfptabH/whmiQX7EvrHx36O4DcyPEsnG152GVNyvqPi9FNIew==", 1625 | "requires": { 1626 | "libbase64": "1.2.1", 1627 | "libmime": "5.0.0", 1628 | "libqp": "1.1.0" 1629 | } 1630 | }, 1631 | "matrix-appservice": { 1632 | "version": "0.8.0", 1633 | "resolved": "https://registry.npmjs.org/matrix-appservice/-/matrix-appservice-0.8.0.tgz", 1634 | "integrity": "sha512-mfgMpmV3dWLtzrd4V/3XtqUD0P44I/mTgsRreW5jMhSaUnnRGZbpptBw2q4/axbLjw2FarlWtOVgertDGMtccA==", 1635 | "requires": { 1636 | "@types/express": "^4.17.8", 1637 | "body-parser": "^1.19.0", 1638 | "express": "^4.17.1", 1639 | "js-yaml": "^3.14.0", 1640 | "morgan": "^1.10.0" 1641 | } 1642 | }, 1643 | "matrix-appservice-bridge": { 1644 | "version": "2.7.0", 1645 | "resolved": "https://registry.npmjs.org/matrix-appservice-bridge/-/matrix-appservice-bridge-2.7.0.tgz", 1646 | "integrity": "sha512-2KVUHMC0fVRX0CmNzkI0X+NqnWqDlXOXKandrf1a0U4OpIUoF4x1lKfC4H95MP1I/NCh4uy6rT96XE2d/8zLUg==", 1647 | "requires": { 1648 | "chalk": "^4.1.0", 1649 | "extend": "^3.0.2", 1650 | "is-my-json-valid": "^2.20.5", 1651 | "js-yaml": "^4.0.0", 1652 | "matrix-appservice": "^0.8.0", 1653 | "matrix-js-sdk": "^9.9.0", 1654 | "nedb": "^1.8.0", 1655 | "nopt": "^5.0.0", 1656 | "p-queue": "^6.6.2", 1657 | "prom-client": "^13.1.0", 1658 | "winston": "^3.3.3", 1659 | "winston-daily-rotate-file": "^4.5.1" 1660 | }, 1661 | "dependencies": { 1662 | "argparse": { 1663 | "version": "2.0.1", 1664 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", 1665 | "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==" 1666 | }, 1667 | "file-stream-rotator": { 1668 | "version": "0.5.7", 1669 | "resolved": "https://registry.npmjs.org/file-stream-rotator/-/file-stream-rotator-0.5.7.tgz", 1670 | "integrity": "sha512-VYb3HZ/GiAGUCrfeakO8Mp54YGswNUHvL7P09WQcXAJNSj3iQ5QraYSp3cIn1MUyw6uzfgN/EFOarCNa4JvUHQ==", 1671 | "requires": { 1672 | "moment": "^2.11.2" 1673 | } 1674 | }, 1675 | "js-yaml": { 1676 | "version": "4.1.0", 1677 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", 1678 | "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", 1679 | "requires": { 1680 | "argparse": "^2.0.1" 1681 | } 1682 | }, 1683 | "matrix-js-sdk": { 1684 | "version": "9.11.0", 1685 | "resolved": "https://registry.npmjs.org/matrix-js-sdk/-/matrix-js-sdk-9.11.0.tgz", 1686 | "integrity": "sha512-wP28ybOxyQ7lbC48QddRORYr8atEwbTqDOsu8H6u9jTTgB2qqczI/bkSoXHtutODuSeLY5x0UuwLcxVCy4yxVQ==", 1687 | "requires": { 1688 | "@babel/runtime": "^7.12.5", 1689 | "another-json": "^0.2.0", 1690 | "browser-request": "^0.3.3", 1691 | "bs58": "^4.0.1", 1692 | "content-type": "^1.0.4", 1693 | "loglevel": "^1.7.1", 1694 | "qs": "^6.9.6", 1695 | "request": "^2.88.2", 1696 | "unhomoglyph": "^1.0.6" 1697 | } 1698 | }, 1699 | "qs": { 1700 | "version": "6.10.1", 1701 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.10.1.tgz", 1702 | "integrity": "sha512-M528Hph6wsSVOBiYUnGf+K/7w0hNshs/duGsNXPUCLH5XAqjEtiPGwNONLV0tBH8NoGb0mvD5JubnUTrujKDTg==", 1703 | "requires": { 1704 | "side-channel": "^1.0.4" 1705 | } 1706 | }, 1707 | "winston-daily-rotate-file": { 1708 | "version": "4.5.5", 1709 | "resolved": "https://registry.npmjs.org/winston-daily-rotate-file/-/winston-daily-rotate-file-4.5.5.tgz", 1710 | "integrity": "sha512-ds0WahIjiDhKCiMXmY799pDBW+58ByqIBtUcsqr4oDoXrAI3Zn+hbgFdUxzMfqA93OG0mPLYVMiotqTgE/WeWQ==", 1711 | "requires": { 1712 | "file-stream-rotator": "^0.5.7", 1713 | "object-hash": "^2.0.1", 1714 | "triple-beam": "^1.3.0", 1715 | "winston-transport": "^4.4.0" 1716 | } 1717 | } 1718 | } 1719 | }, 1720 | "media-typer": { 1721 | "version": "0.3.0", 1722 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 1723 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 1724 | }, 1725 | "merge-descriptors": { 1726 | "version": "1.0.1", 1727 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 1728 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 1729 | }, 1730 | "methods": { 1731 | "version": "1.1.2", 1732 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 1733 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 1734 | }, 1735 | "mime": { 1736 | "version": "1.6.0", 1737 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 1738 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 1739 | }, 1740 | "mime-db": { 1741 | "version": "1.49.0", 1742 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.49.0.tgz", 1743 | "integrity": "sha512-CIc8j9URtOVApSFCQIF+VBkX1RwXp/oMMOrqdyXSBXq5RWNEsRfyj1kiRnQgmNXmHxPoFIxOroKA3zcU9P+nAA==" 1744 | }, 1745 | "mime-types": { 1746 | "version": "2.1.32", 1747 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.32.tgz", 1748 | "integrity": "sha512-hJGaVS4G4c9TSMYh2n6SQAGrC4RnfU+daP8G7cSCmaqNjiOoUY0VHCMS42pxnQmVF1GWwFhbHWn3RIxCqTmZ9A==", 1749 | "requires": { 1750 | "mime-db": "1.49.0" 1751 | } 1752 | }, 1753 | "minimatch": { 1754 | "version": "3.0.4", 1755 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 1756 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 1757 | "dev": true, 1758 | "requires": { 1759 | "brace-expansion": "^1.1.7" 1760 | } 1761 | }, 1762 | "minimist": { 1763 | "version": "1.2.5", 1764 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 1765 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==" 1766 | }, 1767 | "mkdirp": { 1768 | "version": "0.5.5", 1769 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", 1770 | "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", 1771 | "requires": { 1772 | "minimist": "^1.2.5" 1773 | } 1774 | }, 1775 | "moment": { 1776 | "version": "2.29.1", 1777 | "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz", 1778 | "integrity": "sha512-kHmoybcPV8Sqy59DwNDY3Jefr64lK/by/da0ViFcuA4DH0vQg5Q6Ze5VimxkfQNSC+Mls/Kx53s7TjP1RhFEDQ==" 1779 | }, 1780 | "moo": { 1781 | "version": "0.5.1", 1782 | "resolved": "https://registry.npmjs.org/moo/-/moo-0.5.1.tgz", 1783 | "integrity": "sha512-I1mnb5xn4fO80BH9BLcF0yLypy2UKl+Cb01Fu0hJRkJjlCRtxZMWkTdAtDd5ZqCOxtCkhmRwyI57vWT+1iZ67w==" 1784 | }, 1785 | "morgan": { 1786 | "version": "1.10.0", 1787 | "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.10.0.tgz", 1788 | "integrity": "sha512-AbegBVI4sh6El+1gNwvD5YIck7nSA36weD7xvIxG4in80j/UoK8AEGaWnnz8v1GxonMCltmlNs5ZKbGvl9b1XQ==", 1789 | "requires": { 1790 | "basic-auth": "~2.0.1", 1791 | "debug": "2.6.9", 1792 | "depd": "~2.0.0", 1793 | "on-finished": "~2.3.0", 1794 | "on-headers": "~1.0.2" 1795 | }, 1796 | "dependencies": { 1797 | "debug": { 1798 | "version": "2.6.9", 1799 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 1800 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 1801 | "requires": { 1802 | "ms": "2.0.0" 1803 | } 1804 | }, 1805 | "depd": { 1806 | "version": "2.0.0", 1807 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 1808 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==" 1809 | }, 1810 | "ms": { 1811 | "version": "2.0.0", 1812 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1813 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 1814 | } 1815 | } 1816 | }, 1817 | "ms": { 1818 | "version": "2.1.2", 1819 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1820 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 1821 | "dev": true 1822 | }, 1823 | "natural-compare": { 1824 | "version": "1.4.0", 1825 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", 1826 | "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", 1827 | "dev": true 1828 | }, 1829 | "nearley": { 1830 | "version": "2.20.1", 1831 | "resolved": "https://registry.npmjs.org/nearley/-/nearley-2.20.1.tgz", 1832 | "integrity": "sha512-+Mc8UaAebFzgV+KpI5n7DasuuQCHA89dmwm7JXw3TV43ukfNQ9DnBH3Mdb2g/I4Fdxc26pwimBWvjIw0UAILSQ==", 1833 | "requires": { 1834 | "commander": "^2.19.0", 1835 | "moo": "^0.5.0", 1836 | "railroad-diagrams": "^1.0.0", 1837 | "randexp": "0.4.6" 1838 | } 1839 | }, 1840 | "nedb": { 1841 | "version": "1.8.0", 1842 | "resolved": "https://registry.npmjs.org/nedb/-/nedb-1.8.0.tgz", 1843 | "integrity": "sha1-DjUCzYLABNU1WkPJ5VV3vXvZHYg=", 1844 | "requires": { 1845 | "async": "0.2.10", 1846 | "binary-search-tree": "0.2.5", 1847 | "localforage": "^1.3.0", 1848 | "mkdirp": "~0.5.1", 1849 | "underscore": "~1.4.4" 1850 | } 1851 | }, 1852 | "negotiator": { 1853 | "version": "0.6.2", 1854 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 1855 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 1856 | }, 1857 | "nodemailer": { 1858 | "version": "6.6.3", 1859 | "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-6.6.3.tgz", 1860 | "integrity": "sha512-faZFufgTMrphYoDjvyVpbpJcYzwyFnbAMmQtj1lVBYAUSm3SOy2fIdd9+Mr4UxPosBa0JRw9bJoIwQn+nswiew==" 1861 | }, 1862 | "nopt": { 1863 | "version": "5.0.0", 1864 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-5.0.0.tgz", 1865 | "integrity": "sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==", 1866 | "requires": { 1867 | "abbrev": "1" 1868 | } 1869 | }, 1870 | "oauth-sign": { 1871 | "version": "0.9.0", 1872 | "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", 1873 | "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==" 1874 | }, 1875 | "object-hash": { 1876 | "version": "2.2.0", 1877 | "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-2.2.0.tgz", 1878 | "integrity": "sha512-gScRMn0bS5fH+IuwyIFgnh9zBdo4DV+6GhygmWM9HyNJSgS0hScp1f5vjtm7oIIOiT9trXrShAkLFSc2IqKNgw==" 1879 | }, 1880 | "object-inspect": { 1881 | "version": "1.11.0", 1882 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.11.0.tgz", 1883 | "integrity": "sha512-jp7ikS6Sd3GxQfZJPyH3cjcbJF6GZPClgdV+EFygjFLQ5FmW/dRUnTd9PQ9k0JhoNDabWFbpF1yCdSWCC6gexg==" 1884 | }, 1885 | "on-finished": { 1886 | "version": "2.3.0", 1887 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 1888 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 1889 | "requires": { 1890 | "ee-first": "1.1.1" 1891 | } 1892 | }, 1893 | "on-headers": { 1894 | "version": "1.0.2", 1895 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", 1896 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==" 1897 | }, 1898 | "once": { 1899 | "version": "1.4.0", 1900 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1901 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1902 | "dev": true, 1903 | "requires": { 1904 | "wrappy": "1" 1905 | } 1906 | }, 1907 | "one-time": { 1908 | "version": "1.0.0", 1909 | "resolved": "https://registry.npmjs.org/one-time/-/one-time-1.0.0.tgz", 1910 | "integrity": "sha512-5DXOiRKwuSEcQ/l0kGCF6Q3jcADFv5tSmRaJck/OqkVFcOzutB134KRSfF0xDrL39MNnqxbHBbUUcjZIhTgb2g==", 1911 | "requires": { 1912 | "fn.name": "1.x.x" 1913 | } 1914 | }, 1915 | "optionator": { 1916 | "version": "0.9.1", 1917 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.1.tgz", 1918 | "integrity": "sha512-74RlY5FCnhq4jRxVUPKDaRwrVNXMqsGsiW6AJw4XK8hmtm10wC0ypZBLw5IIp85NZMr91+qd1RvvENwg7jjRFw==", 1919 | "dev": true, 1920 | "requires": { 1921 | "deep-is": "^0.1.3", 1922 | "fast-levenshtein": "^2.0.6", 1923 | "levn": "^0.4.1", 1924 | "prelude-ls": "^1.2.1", 1925 | "type-check": "^0.4.0", 1926 | "word-wrap": "^1.2.3" 1927 | } 1928 | }, 1929 | "p-finally": { 1930 | "version": "1.0.0", 1931 | "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", 1932 | "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=" 1933 | }, 1934 | "p-queue": { 1935 | "version": "6.6.2", 1936 | "resolved": "https://registry.npmjs.org/p-queue/-/p-queue-6.6.2.tgz", 1937 | "integrity": "sha512-RwFpb72c/BhQLEXIZ5K2e+AhgNVmIejGlTgiB9MzZ0e93GRvqZ7uSi0dvRF7/XIXDeNkra2fNHBxTyPDGySpjQ==", 1938 | "requires": { 1939 | "eventemitter3": "^4.0.4", 1940 | "p-timeout": "^3.2.0" 1941 | } 1942 | }, 1943 | "p-timeout": { 1944 | "version": "3.2.0", 1945 | "resolved": "https://registry.npmjs.org/p-timeout/-/p-timeout-3.2.0.tgz", 1946 | "integrity": "sha512-rhIwUycgwwKcP9yTOOFK/AKsAopjjCakVqLHePO3CC6Mir1Z99xT+R63jZxAT5lFZLa2inS5h+ZS2GvR99/FBg==", 1947 | "requires": { 1948 | "p-finally": "^1.0.0" 1949 | } 1950 | }, 1951 | "parent-module": { 1952 | "version": "1.0.1", 1953 | "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", 1954 | "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", 1955 | "dev": true, 1956 | "requires": { 1957 | "callsites": "^3.0.0" 1958 | } 1959 | }, 1960 | "parseley": { 1961 | "version": "0.7.0", 1962 | "resolved": "https://registry.npmjs.org/parseley/-/parseley-0.7.0.tgz", 1963 | "integrity": "sha512-xyOytsdDu077M3/46Am+2cGXEKM9U9QclBDv7fimY7e+BBlxh2JcBp2mgNsmkyA9uvgyTjVzDi7cP1v4hcFxbw==", 1964 | "requires": { 1965 | "moo": "^0.5.1", 1966 | "nearley": "^2.20.1" 1967 | } 1968 | }, 1969 | "parseurl": { 1970 | "version": "1.3.3", 1971 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 1972 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 1973 | }, 1974 | "path-is-absolute": { 1975 | "version": "1.0.1", 1976 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1977 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 1978 | "dev": true 1979 | }, 1980 | "path-key": { 1981 | "version": "3.1.1", 1982 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", 1983 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", 1984 | "dev": true 1985 | }, 1986 | "path-to-regexp": { 1987 | "version": "0.1.7", 1988 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 1989 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 1990 | }, 1991 | "performance-now": { 1992 | "version": "2.1.0", 1993 | "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", 1994 | "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=" 1995 | }, 1996 | "prelude-ls": { 1997 | "version": "1.2.1", 1998 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", 1999 | "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", 2000 | "dev": true 2001 | }, 2002 | "process-nextick-args": { 2003 | "version": "2.0.1", 2004 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 2005 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" 2006 | }, 2007 | "progress": { 2008 | "version": "2.0.3", 2009 | "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", 2010 | "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", 2011 | "dev": true 2012 | }, 2013 | "prom-client": { 2014 | "version": "13.2.0", 2015 | "resolved": "https://registry.npmjs.org/prom-client/-/prom-client-13.2.0.tgz", 2016 | "integrity": "sha512-wGr5mlNNdRNzEhRYXgboUU2LxHWIojxscJKmtG3R8f4/KiWqyYgXTLHs0+Ted7tG3zFT7pgHJbtomzZ1L0ARaQ==", 2017 | "requires": { 2018 | "tdigest": "^0.1.1" 2019 | } 2020 | }, 2021 | "proxy-addr": { 2022 | "version": "2.0.7", 2023 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 2024 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 2025 | "requires": { 2026 | "forwarded": "0.2.0", 2027 | "ipaddr.js": "1.9.1" 2028 | } 2029 | }, 2030 | "psl": { 2031 | "version": "1.8.0", 2032 | "resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz", 2033 | "integrity": "sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ==" 2034 | }, 2035 | "punycode": { 2036 | "version": "2.1.1", 2037 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 2038 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==" 2039 | }, 2040 | "qs": { 2041 | "version": "6.7.0", 2042 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", 2043 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" 2044 | }, 2045 | "railroad-diagrams": { 2046 | "version": "1.0.0", 2047 | "resolved": "https://registry.npmjs.org/railroad-diagrams/-/railroad-diagrams-1.0.0.tgz", 2048 | "integrity": "sha1-635iZ1SN3t+4mcG5Dlc3RVnN234=" 2049 | }, 2050 | "randexp": { 2051 | "version": "0.4.6", 2052 | "resolved": "https://registry.npmjs.org/randexp/-/randexp-0.4.6.tgz", 2053 | "integrity": "sha512-80WNmd9DA0tmZrw9qQa62GPPWfuXJknrmVmLcxvq4uZBdYqb1wYoKTmnlGUchvVWe0XiLupYkBoXVOxz3C8DYQ==", 2054 | "requires": { 2055 | "discontinuous-range": "1.0.0", 2056 | "ret": "~0.1.10" 2057 | } 2058 | }, 2059 | "range-parser": { 2060 | "version": "1.2.1", 2061 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 2062 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 2063 | }, 2064 | "raw-body": { 2065 | "version": "2.4.0", 2066 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", 2067 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", 2068 | "requires": { 2069 | "bytes": "3.1.0", 2070 | "http-errors": "1.7.2", 2071 | "iconv-lite": "0.4.24", 2072 | "unpipe": "1.0.0" 2073 | }, 2074 | "dependencies": { 2075 | "iconv-lite": { 2076 | "version": "0.4.24", 2077 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 2078 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 2079 | "requires": { 2080 | "safer-buffer": ">= 2.1.2 < 3" 2081 | } 2082 | } 2083 | } 2084 | }, 2085 | "readable-stream": { 2086 | "version": "3.6.0", 2087 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", 2088 | "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", 2089 | "requires": { 2090 | "inherits": "^2.0.3", 2091 | "string_decoder": "^1.1.1", 2092 | "util-deprecate": "^1.0.1" 2093 | } 2094 | }, 2095 | "regenerator-runtime": { 2096 | "version": "0.13.9", 2097 | "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.9.tgz", 2098 | "integrity": "sha512-p3VT+cOEgxFsRRA9X4lkI1E+k2/CtnKtU4gcxyaCUreilL/vqI6CdZ3wxVUx3UOUg+gnUOQQcRI7BmSI656MYA==" 2099 | }, 2100 | "regexpp": { 2101 | "version": "3.2.0", 2102 | "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz", 2103 | "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==", 2104 | "dev": true 2105 | }, 2106 | "request": { 2107 | "version": "2.88.2", 2108 | "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz", 2109 | "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", 2110 | "requires": { 2111 | "aws-sign2": "~0.7.0", 2112 | "aws4": "^1.8.0", 2113 | "caseless": "~0.12.0", 2114 | "combined-stream": "~1.0.6", 2115 | "extend": "~3.0.2", 2116 | "forever-agent": "~0.6.1", 2117 | "form-data": "~2.3.2", 2118 | "har-validator": "~5.1.3", 2119 | "http-signature": "~1.2.0", 2120 | "is-typedarray": "~1.0.0", 2121 | "isstream": "~0.1.2", 2122 | "json-stringify-safe": "~5.0.1", 2123 | "mime-types": "~2.1.19", 2124 | "oauth-sign": "~0.9.0", 2125 | "performance-now": "^2.1.0", 2126 | "qs": "~6.5.2", 2127 | "safe-buffer": "^5.1.2", 2128 | "tough-cookie": "~2.5.0", 2129 | "tunnel-agent": "^0.6.0", 2130 | "uuid": "^3.3.2" 2131 | }, 2132 | "dependencies": { 2133 | "qs": { 2134 | "version": "6.5.2", 2135 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", 2136 | "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==" 2137 | } 2138 | } 2139 | }, 2140 | "require-from-string": { 2141 | "version": "2.0.2", 2142 | "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", 2143 | "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", 2144 | "dev": true 2145 | }, 2146 | "resolve-from": { 2147 | "version": "4.0.0", 2148 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", 2149 | "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", 2150 | "dev": true 2151 | }, 2152 | "ret": { 2153 | "version": "0.1.15", 2154 | "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", 2155 | "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==" 2156 | }, 2157 | "rimraf": { 2158 | "version": "3.0.2", 2159 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", 2160 | "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", 2161 | "dev": true, 2162 | "requires": { 2163 | "glob": "^7.1.3" 2164 | } 2165 | }, 2166 | "safe-buffer": { 2167 | "version": "5.1.2", 2168 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 2169 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 2170 | }, 2171 | "safer-buffer": { 2172 | "version": "2.1.2", 2173 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 2174 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 2175 | }, 2176 | "selderee": { 2177 | "version": "0.6.0", 2178 | "resolved": "https://registry.npmjs.org/selderee/-/selderee-0.6.0.tgz", 2179 | "integrity": "sha512-ibqWGV5aChDvfVdqNYuaJP/HnVBhlRGSRrlbttmlMpHcLuTqqbMH36QkSs9GEgj5M88JDYLI8eyP94JaQ8xRlg==", 2180 | "requires": { 2181 | "parseley": "^0.7.0" 2182 | } 2183 | }, 2184 | "semver": { 2185 | "version": "7.3.5", 2186 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", 2187 | "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", 2188 | "dev": true, 2189 | "requires": { 2190 | "lru-cache": "^6.0.0" 2191 | } 2192 | }, 2193 | "send": { 2194 | "version": "0.17.1", 2195 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", 2196 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", 2197 | "requires": { 2198 | "debug": "2.6.9", 2199 | "depd": "~1.1.2", 2200 | "destroy": "~1.0.4", 2201 | "encodeurl": "~1.0.2", 2202 | "escape-html": "~1.0.3", 2203 | "etag": "~1.8.1", 2204 | "fresh": "0.5.2", 2205 | "http-errors": "~1.7.2", 2206 | "mime": "1.6.0", 2207 | "ms": "2.1.1", 2208 | "on-finished": "~2.3.0", 2209 | "range-parser": "~1.2.1", 2210 | "statuses": "~1.5.0" 2211 | }, 2212 | "dependencies": { 2213 | "debug": { 2214 | "version": "2.6.9", 2215 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 2216 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 2217 | "requires": { 2218 | "ms": "2.0.0" 2219 | }, 2220 | "dependencies": { 2221 | "ms": { 2222 | "version": "2.0.0", 2223 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 2224 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 2225 | } 2226 | } 2227 | }, 2228 | "ms": { 2229 | "version": "2.1.1", 2230 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 2231 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 2232 | } 2233 | } 2234 | }, 2235 | "serve-static": { 2236 | "version": "1.14.1", 2237 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", 2238 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", 2239 | "requires": { 2240 | "encodeurl": "~1.0.2", 2241 | "escape-html": "~1.0.3", 2242 | "parseurl": "~1.3.3", 2243 | "send": "0.17.1" 2244 | } 2245 | }, 2246 | "setprototypeof": { 2247 | "version": "1.1.1", 2248 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", 2249 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" 2250 | }, 2251 | "shebang-command": { 2252 | "version": "2.0.0", 2253 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", 2254 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", 2255 | "dev": true, 2256 | "requires": { 2257 | "shebang-regex": "^3.0.0" 2258 | } 2259 | }, 2260 | "shebang-regex": { 2261 | "version": "3.0.0", 2262 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", 2263 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", 2264 | "dev": true 2265 | }, 2266 | "side-channel": { 2267 | "version": "1.0.4", 2268 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz", 2269 | "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", 2270 | "requires": { 2271 | "call-bind": "^1.0.0", 2272 | "get-intrinsic": "^1.0.2", 2273 | "object-inspect": "^1.9.0" 2274 | } 2275 | }, 2276 | "simple-swizzle": { 2277 | "version": "0.2.2", 2278 | "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.2.tgz", 2279 | "integrity": "sha1-pNprY1/8zMoz9w0Xy5JZLeleVXo=", 2280 | "requires": { 2281 | "is-arrayish": "^0.3.1" 2282 | } 2283 | }, 2284 | "slice-ansi": { 2285 | "version": "4.0.0", 2286 | "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-4.0.0.tgz", 2287 | "integrity": "sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==", 2288 | "dev": true, 2289 | "requires": { 2290 | "ansi-styles": "^4.0.0", 2291 | "astral-regex": "^2.0.0", 2292 | "is-fullwidth-code-point": "^3.0.0" 2293 | } 2294 | }, 2295 | "smtp-server": { 2296 | "version": "3.9.0", 2297 | "resolved": "https://registry.npmjs.org/smtp-server/-/smtp-server-3.9.0.tgz", 2298 | "integrity": "sha512-CHws5GkHjfIikue6vSdp3uRnmW85l0JJQibVwDs7S5aIUppXJA9Y60XcdxcaCLXmAnd8V8wbtav5KY92nCZeag==", 2299 | "requires": { 2300 | "base32.js": "0.1.0", 2301 | "ipv6-normalize": "1.0.1", 2302 | "nodemailer": "6.6.1" 2303 | }, 2304 | "dependencies": { 2305 | "nodemailer": { 2306 | "version": "6.6.1", 2307 | "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-6.6.1.tgz", 2308 | "integrity": "sha512-1xzFN3gqv+/qJ6YRyxBxfTYstLNt0FCtZaFRvf4Sg9wxNGWbwFmGXVpfSi6ThGK6aRxAo+KjHtYSW8NvCsNSAg==" 2309 | } 2310 | } 2311 | }, 2312 | "sprintf-js": { 2313 | "version": "1.0.3", 2314 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 2315 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=" 2316 | }, 2317 | "sshpk": { 2318 | "version": "1.16.1", 2319 | "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", 2320 | "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", 2321 | "requires": { 2322 | "asn1": "~0.2.3", 2323 | "assert-plus": "^1.0.0", 2324 | "bcrypt-pbkdf": "^1.0.0", 2325 | "dashdash": "^1.12.0", 2326 | "ecc-jsbn": "~0.1.1", 2327 | "getpass": "^0.1.1", 2328 | "jsbn": "~0.1.0", 2329 | "safer-buffer": "^2.0.2", 2330 | "tweetnacl": "~0.14.0" 2331 | } 2332 | }, 2333 | "stack-trace": { 2334 | "version": "0.0.10", 2335 | "resolved": "https://registry.npmjs.org/stack-trace/-/stack-trace-0.0.10.tgz", 2336 | "integrity": "sha1-VHxws0fo0ytOEI6hoqFZ5f3eGcA=" 2337 | }, 2338 | "statuses": { 2339 | "version": "1.5.0", 2340 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", 2341 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" 2342 | }, 2343 | "string-width": { 2344 | "version": "4.2.2", 2345 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", 2346 | "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", 2347 | "dev": true, 2348 | "requires": { 2349 | "emoji-regex": "^8.0.0", 2350 | "is-fullwidth-code-point": "^3.0.0", 2351 | "strip-ansi": "^6.0.0" 2352 | } 2353 | }, 2354 | "string_decoder": { 2355 | "version": "1.3.0", 2356 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", 2357 | "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", 2358 | "requires": { 2359 | "safe-buffer": "~5.2.0" 2360 | }, 2361 | "dependencies": { 2362 | "safe-buffer": { 2363 | "version": "5.2.1", 2364 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 2365 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" 2366 | } 2367 | } 2368 | }, 2369 | "strip-ansi": { 2370 | "version": "6.0.0", 2371 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 2372 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 2373 | "dev": true, 2374 | "requires": { 2375 | "ansi-regex": "^5.0.0" 2376 | } 2377 | }, 2378 | "strip-json-comments": { 2379 | "version": "3.1.1", 2380 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", 2381 | "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", 2382 | "dev": true 2383 | }, 2384 | "supports-color": { 2385 | "version": "7.2.0", 2386 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 2387 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 2388 | "requires": { 2389 | "has-flag": "^4.0.0" 2390 | } 2391 | }, 2392 | "table": { 2393 | "version": "6.7.1", 2394 | "resolved": "https://registry.npmjs.org/table/-/table-6.7.1.tgz", 2395 | "integrity": "sha512-ZGum47Yi6KOOFDE8m223td53ath2enHcYLgOCjGr5ngu8bdIARQk6mN/wRMv4yMRcHnCSnHbCEha4sobQx5yWg==", 2396 | "dev": true, 2397 | "requires": { 2398 | "ajv": "^8.0.1", 2399 | "lodash.clonedeep": "^4.5.0", 2400 | "lodash.truncate": "^4.4.2", 2401 | "slice-ansi": "^4.0.0", 2402 | "string-width": "^4.2.0", 2403 | "strip-ansi": "^6.0.0" 2404 | }, 2405 | "dependencies": { 2406 | "ajv": { 2407 | "version": "8.6.2", 2408 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.6.2.tgz", 2409 | "integrity": "sha512-9807RlWAgT564wT+DjeyU5OFMPjmzxVobvDFmNAhY+5zD6A2ly3jDp6sgnfyDtlIQ+7H97oc/DGCzzfu9rjw9w==", 2410 | "dev": true, 2411 | "requires": { 2412 | "fast-deep-equal": "^3.1.1", 2413 | "json-schema-traverse": "^1.0.0", 2414 | "require-from-string": "^2.0.2", 2415 | "uri-js": "^4.2.2" 2416 | } 2417 | }, 2418 | "json-schema-traverse": { 2419 | "version": "1.0.0", 2420 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", 2421 | "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", 2422 | "dev": true 2423 | } 2424 | } 2425 | }, 2426 | "tdigest": { 2427 | "version": "0.1.1", 2428 | "resolved": "https://registry.npmjs.org/tdigest/-/tdigest-0.1.1.tgz", 2429 | "integrity": "sha1-Ljyyw56kSeVdHmzZEReszKRYgCE=", 2430 | "requires": { 2431 | "bintrees": "1.0.1" 2432 | } 2433 | }, 2434 | "text-hex": { 2435 | "version": "1.0.0", 2436 | "resolved": "https://registry.npmjs.org/text-hex/-/text-hex-1.0.0.tgz", 2437 | "integrity": "sha512-uuVGNWzgJ4yhRaNSiubPY7OjISw4sw4E5Uv0wbjp+OzcbmVU/rsT8ujgcXJhn9ypzsgr5vlzpPqP+MBBKcGvbg==" 2438 | }, 2439 | "text-table": { 2440 | "version": "0.2.0", 2441 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", 2442 | "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", 2443 | "dev": true 2444 | }, 2445 | "tlds": { 2446 | "version": "1.221.1", 2447 | "resolved": "https://registry.npmjs.org/tlds/-/tlds-1.221.1.tgz", 2448 | "integrity": "sha512-N1Afn/SLeOQRpxMwHBuNFJ3GvGrdtY4XPXKPFcx8he0U9Jg9ZkvTKE1k3jQDtCmlFn44UxjVtouF6PT4rEGd3Q==" 2449 | }, 2450 | "toidentifier": { 2451 | "version": "1.0.0", 2452 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", 2453 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" 2454 | }, 2455 | "tough-cookie": { 2456 | "version": "2.5.0", 2457 | "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz", 2458 | "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==", 2459 | "requires": { 2460 | "psl": "^1.1.28", 2461 | "punycode": "^2.1.1" 2462 | } 2463 | }, 2464 | "triple-beam": { 2465 | "version": "1.3.0", 2466 | "resolved": "https://registry.npmjs.org/triple-beam/-/triple-beam-1.3.0.tgz", 2467 | "integrity": "sha512-XrHUvV5HpdLmIj4uVMxHggLbFSZYIn7HEWsqePZcI50pco+MPqJ50wMGY794X7AOOhxOBAjbkqfAbEe/QMp2Lw==" 2468 | }, 2469 | "tunnel-agent": { 2470 | "version": "0.6.0", 2471 | "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", 2472 | "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", 2473 | "requires": { 2474 | "safe-buffer": "^5.0.1" 2475 | } 2476 | }, 2477 | "tweetnacl": { 2478 | "version": "0.14.5", 2479 | "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", 2480 | "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=" 2481 | }, 2482 | "type-check": { 2483 | "version": "0.4.0", 2484 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", 2485 | "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", 2486 | "dev": true, 2487 | "requires": { 2488 | "prelude-ls": "^1.2.1" 2489 | } 2490 | }, 2491 | "type-fest": { 2492 | "version": "0.20.2", 2493 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", 2494 | "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", 2495 | "dev": true 2496 | }, 2497 | "type-is": { 2498 | "version": "1.6.18", 2499 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 2500 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 2501 | "requires": { 2502 | "media-typer": "0.3.0", 2503 | "mime-types": "~2.1.24" 2504 | } 2505 | }, 2506 | "uc.micro": { 2507 | "version": "1.0.6", 2508 | "resolved": "https://registry.npmjs.org/uc.micro/-/uc.micro-1.0.6.tgz", 2509 | "integrity": "sha512-8Y75pvTYkLJW2hWQHXxoqRgV7qb9B+9vFEtidML+7koHUFapnVJAZ6cKs+Qjz5Aw3aZWHMC6u0wJE3At+nSGwA==" 2510 | }, 2511 | "underscore": { 2512 | "version": "1.4.4", 2513 | "resolved": "https://registry.npmjs.org/underscore/-/underscore-1.4.4.tgz", 2514 | "integrity": "sha1-YaajIBBiKvoHljvzJSA88SI51gQ=" 2515 | }, 2516 | "unhomoglyph": { 2517 | "version": "1.0.6", 2518 | "resolved": "https://registry.npmjs.org/unhomoglyph/-/unhomoglyph-1.0.6.tgz", 2519 | "integrity": "sha512-7uvcWI3hWshSADBu4JpnyYbTVc7YlhF5GDW/oPD5AxIxl34k4wXR3WDkPnzLxkN32LiTCTKMQLtKVZiwki3zGg==" 2520 | }, 2521 | "unpipe": { 2522 | "version": "1.0.0", 2523 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 2524 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 2525 | }, 2526 | "uri-js": { 2527 | "version": "4.4.1", 2528 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", 2529 | "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", 2530 | "requires": { 2531 | "punycode": "^2.1.0" 2532 | } 2533 | }, 2534 | "util-deprecate": { 2535 | "version": "1.0.2", 2536 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 2537 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" 2538 | }, 2539 | "utils-merge": { 2540 | "version": "1.0.1", 2541 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 2542 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 2543 | }, 2544 | "uuid": { 2545 | "version": "3.4.0", 2546 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", 2547 | "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==" 2548 | }, 2549 | "v8-compile-cache": { 2550 | "version": "2.3.0", 2551 | "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz", 2552 | "integrity": "sha512-l8lCEmLcLYZh4nbunNZvQCJc5pv7+RCwa8q/LdUx8u7lsWvPDKmpodJAJNwkAhJC//dFY48KuIEmjtd4RViDrA==", 2553 | "dev": true 2554 | }, 2555 | "vary": { 2556 | "version": "1.1.2", 2557 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 2558 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 2559 | }, 2560 | "verror": { 2561 | "version": "1.10.0", 2562 | "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", 2563 | "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", 2564 | "requires": { 2565 | "assert-plus": "^1.0.0", 2566 | "core-util-is": "1.0.2", 2567 | "extsprintf": "^1.2.0" 2568 | } 2569 | }, 2570 | "which": { 2571 | "version": "2.0.2", 2572 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 2573 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 2574 | "dev": true, 2575 | "requires": { 2576 | "isexe": "^2.0.0" 2577 | } 2578 | }, 2579 | "winston": { 2580 | "version": "3.3.3", 2581 | "resolved": "https://registry.npmjs.org/winston/-/winston-3.3.3.tgz", 2582 | "integrity": "sha512-oEXTISQnC8VlSAKf1KYSSd7J6IWuRPQqDdo8eoRNaYKLvwSb5+79Z3Yi1lrl6KDpU6/VWaxpakDAtb1oQ4n9aw==", 2583 | "requires": { 2584 | "@dabh/diagnostics": "^2.0.2", 2585 | "async": "^3.1.0", 2586 | "is-stream": "^2.0.0", 2587 | "logform": "^2.2.0", 2588 | "one-time": "^1.0.0", 2589 | "readable-stream": "^3.4.0", 2590 | "stack-trace": "0.0.x", 2591 | "triple-beam": "^1.3.0", 2592 | "winston-transport": "^4.4.0" 2593 | }, 2594 | "dependencies": { 2595 | "async": { 2596 | "version": "3.2.0", 2597 | "resolved": "https://registry.npmjs.org/async/-/async-3.2.0.tgz", 2598 | "integrity": "sha512-TR2mEZFVOj2pLStYxLht7TyfuRzaydfpxr3k9RpHIzMgw7A64dzsdqCxH1WJyQdoe8T10nDXd9wnEigmiuHIZw==" 2599 | } 2600 | } 2601 | }, 2602 | "winston-compat": { 2603 | "version": "0.0.1", 2604 | "resolved": "https://registry.npmjs.org/winston-compat/-/winston-compat-0.0.1.tgz", 2605 | "integrity": "sha1-iAn6rNP/sc5qPjfbn1SDS3QGhp0=", 2606 | "requires": { 2607 | "cycle": "~1.0.3" 2608 | } 2609 | }, 2610 | "winston-daily-rotate-file": { 2611 | "version": "2.0.0-rc.6", 2612 | "resolved": "https://registry.npmjs.org/winston-daily-rotate-file/-/winston-daily-rotate-file-2.0.0-rc.6.tgz", 2613 | "integrity": "sha512-MBN+Xyw4X8Ie0xlD91oFZuMPOw6BruPJfV3tI0/FN6qBH5y3mH/J7rZ5nphzogofouic3EBWQZyrB8ZHF+nSuA==", 2614 | "requires": { 2615 | "file-stream-rotator": "^0.2.1", 2616 | "semver": "^5.5.0", 2617 | "triple-beam": "^1.1.0", 2618 | "winston-compat": "0.0.1" 2619 | }, 2620 | "dependencies": { 2621 | "semver": { 2622 | "version": "5.7.1", 2623 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 2624 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" 2625 | } 2626 | } 2627 | }, 2628 | "winston-transport": { 2629 | "version": "4.4.0", 2630 | "resolved": "https://registry.npmjs.org/winston-transport/-/winston-transport-4.4.0.tgz", 2631 | "integrity": "sha512-Lc7/p3GtqtqPBYYtS6KCN3c77/2QCev51DvcJKbkFPQNoj1sinkGwLGFDxkXY9J6p9+EPnYs+D90uwbnaiURTw==", 2632 | "requires": { 2633 | "readable-stream": "^2.3.7", 2634 | "triple-beam": "^1.2.0" 2635 | }, 2636 | "dependencies": { 2637 | "readable-stream": { 2638 | "version": "2.3.7", 2639 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", 2640 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", 2641 | "requires": { 2642 | "core-util-is": "~1.0.0", 2643 | "inherits": "~2.0.3", 2644 | "isarray": "~1.0.0", 2645 | "process-nextick-args": "~2.0.0", 2646 | "safe-buffer": "~5.1.1", 2647 | "string_decoder": "~1.1.1", 2648 | "util-deprecate": "~1.0.1" 2649 | } 2650 | }, 2651 | "string_decoder": { 2652 | "version": "1.1.1", 2653 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 2654 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 2655 | "requires": { 2656 | "safe-buffer": "~5.1.0" 2657 | } 2658 | } 2659 | } 2660 | }, 2661 | "word-wrap": { 2662 | "version": "1.2.3", 2663 | "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", 2664 | "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", 2665 | "dev": true 2666 | }, 2667 | "wrappy": { 2668 | "version": "1.0.2", 2669 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 2670 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 2671 | "dev": true 2672 | }, 2673 | "xtend": { 2674 | "version": "4.0.2", 2675 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", 2676 | "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==" 2677 | }, 2678 | "yallist": { 2679 | "version": "4.0.0", 2680 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", 2681 | "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", 2682 | "dev": true 2683 | } 2684 | } 2685 | } 2686 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "matrix-appservice-email", 3 | "version": "0.0.1", 4 | "description": "An Email bridge for Matrix", 5 | "main": "app.js", 6 | "scripts": { 7 | "test": "echo \"Error: no test specified\" && exit 1" 8 | }, 9 | "repository": { 10 | "type": "git", 11 | "url": "git+https://github.com/abbyck/matrix-appservice-email.git" 12 | }, 13 | "author": "", 14 | "license": "Apache-2.0", 15 | "bugs": { 16 | "url": "https://github.com/abbyck/matrix-appservice-email/issues" 17 | }, 18 | "homepage": "https://github.com/abbyck/matrix-appservice-email#readme", 19 | "dependencies": { 20 | "chalk": "^4.1.2", 21 | "dkim-signer": "^0.2.2", 22 | "email-addresses": "^4.0.0", 23 | "mailparser": "^3.3.0", 24 | "matrix-appservice-bridge": "^2.7.0", 25 | "nodemailer": "^6.6.3", 26 | "smtp-server": "^3.9.0", 27 | "winston": "^3.3.3", 28 | "winston-daily-rotate-file": "^2.0.0-rc.6" 29 | }, 30 | "devDependencies": { 31 | "eslint": "^7.32.0" 32 | } 33 | } 34 | -------------------------------------------------------------------------------- /src/bridge.js: -------------------------------------------------------------------------------- 1 | const fs = require("fs"); 2 | const { Bridge, MatrixUser } = require('matrix-appservice-bridge'); 3 | const { startSMTP } = require('./email'); 4 | const { Logging } = require('./log'); 5 | 6 | const log = Logging.get("bridge"); 7 | 8 | exports.bridge = async function(port, config, registration) { 9 | bridge = new Bridge({ 10 | homeserverUrl: config.bridge.homeserverUrl, 11 | domain: config.bridge.domain, 12 | registration, 13 | 14 | controller: { 15 | onUserQuery: function(queriedUser) { 16 | return {}; // auto-provision users with no additional data 17 | }, 18 | 19 | onEvent: function(request, context) { 20 | // events from matrix 21 | const event = request.getData(); 22 | if (event.type === "m.room.member" && event.state_key) { 23 | // Check DM leave 24 | if (event.content.membership === "leave") { 25 | checkMappingsAndLeaveDM(event.state_key, event.room_id) 26 | .then(() => { 27 | log.info(`Removed the mapping`); 28 | }) 29 | .catch(ex => { 30 | log.error(`Could not remove the mapping: ${ex}`); 31 | }); 32 | } 33 | } 34 | if (event.type !== "m.room.message" || !event.content) { 35 | return; 36 | } 37 | log.info(`Matrix-side: ${event.sender}: RoomID: ${event.room_id}, EventID: ${event.event_id}`); 38 | sendMessageViaEmail(event.room_id, event) 39 | .then(() => { 40 | log.info(`Mail sent`); 41 | }) 42 | .catch(ex => { 43 | log.error(`sendMessageViaEmail failed, ${ex}`); 44 | }); 45 | } 46 | } 47 | }); 48 | 49 | const sendMail = require('./email/outbound')({ 50 | startTLS: config.email.tls.enabled, 51 | tls: { 52 | key: fs.existsSync(config.email.tls.tlsKey) ? fs.readFileSync(config.email.tlsKey) : '', 53 | cert: fs.existsSync(config.email.tls.tlsCert) ? fs.readFileSync(config.email.tls.tlsCert) : '', 54 | }, 55 | dkimEnabled: config.email.dkim.enabled, 56 | dkim: { 57 | privateKey: fs.existsSync(config.email.dkim.dkimKey) ? 58 | fs.readFileSync(config.email.dkim.dkimKey, 'utf8') : '', 59 | keySelector: config.email.dkim.selector, 60 | }, 61 | smtpPort: config.email.outboundPort, 62 | smtpHost: config.email.smtpHost 63 | }); 64 | 65 | async function sendMessageViaEmail(roomid, event) { 66 | const botClient = bridge.getIntent().getClient(); 67 | let roomEmail, roomMembers, dmMappings; 68 | try { 69 | dmMappings = await botClient.getAccountDataFromServer("me.abhy.email-bridge"); 70 | } 71 | catch (ex) { 72 | throw Error(`Could not fetch the DM Mappings from account data: ${ex}`); 73 | } 74 | // Check if the user ID is in DM mappings. 75 | if (event.user_id in dmMappings) { 76 | log.info('Sender is in DM mappings'); 77 | // Check if the roomID mapping 78 | if (dmMappings[event.user_id].roomId === event.room_id) { 79 | log.info('Message is from a DM'); 80 | roomEmail = getRoomMailIdFromUserId(event.user_id, config.email.mxDomain); 81 | sendMail({ 82 | from: roomEmail, 83 | to: getMailIdFromUserId(dmMappings[event.user_id].emailUser), 84 | subject: `You have a message from ${event.sender}`, 85 | html: `${event.content.body}`, 86 | }).then(() => { 87 | log.info(`Message sent from ${roomEmail} to ${roomEmail}`); 88 | }).catch(ex => { 89 | throw Error(`Could not sent email from ${roomEmail} to ${roomEmail}: ${ex}`); 90 | }); 91 | return; 92 | } 93 | } 94 | const ASBot = bridge.getBot(); 95 | try { 96 | roomMembers = await ASBot.getJoinedMembers(roomid).catch(); 97 | } 98 | catch (ex) { 99 | log.error(`Could not get room member list`, ex); 100 | return; 101 | } 102 | for (let member in roomMembers) { 103 | if (ASBot.isRemoteUser(member)) { 104 | log.info("Remote email userId", member); 105 | // Query for `m.room.canonical_alias` only if roomEmail is undefined(first occurrence). 106 | let roomAlias; 107 | if (!roomEmail) { 108 | const intent = bridge.getIntent(member); 109 | try { 110 | roomAlias = await intent.getStateEvent(roomid, 'm.room.canonical_alias'); 111 | log.info("Room alias:", roomAlias.alias); 112 | } 113 | catch (ex) { 114 | log.error(`Could not get roomAlias`, ex); 115 | return; 116 | } 117 | roomEmail = getRoomMailIdFromRoomAlias(roomAlias.alias, config.email.mxDomain); 118 | log.info("Room email id:", roomEmail); 119 | } 120 | const emailIdOfMember = getMailIdFromUserId(member, config.bridge.domain); 121 | sendMail({ 122 | from: roomEmail, 123 | to: emailIdOfMember, 124 | subject: `You have a message from ${roomAlias.alias}`, 125 | html: `${event.content.body}`, 126 | }).then(() => { 127 | log.info(`Message sent from ${roomEmail} to ${roomEmail}`); 128 | }).catch(ex => { 129 | log.error(`Could not sent email from ${roomEmail} to ${roomEmail}: ${ex}`); 130 | }); 131 | } 132 | } 133 | } 134 | 135 | async function checkMappingsAndLeaveDM(sender, roomId) { 136 | let dmMappings; 137 | const botClient = bridge.getIntent().getClient(); 138 | try { 139 | dmMappings = await botClient.getAccountDataFromServer('me.abhy.email-bridge'); 140 | } 141 | catch (ex) { 142 | throw Error(`Failed to get DM mappings from HS: ${ex}`); 143 | } 144 | if (sender in dmMappings && dmMappings[sender].roomId === roomId) { 145 | log.info(`${sender} left from DM room ${roomId}`); 146 | const intent = bridge.getIntent(dmMappings[sender].emailUser); 147 | try { 148 | await intent.leave(roomId, "Empty room"); 149 | } 150 | catch (ex) { 151 | throw Error(`${sender} could not leave the empty room: ${ex}`); 152 | } 153 | try { 154 | delete dmMappings[sender]; 155 | await botClient.setAccountData('me.abhy.email-bridge', dmMappings); 156 | } 157 | catch (ex) { 158 | throw Error(`Could not update the DM Mappings: ${ex}`); 159 | } 160 | } 161 | } 162 | 163 | process.on('SIGINT', async () => { 164 | // Handle Ctrl-C 165 | log.info(`Closing bridge due to SIGINT`); 166 | try { 167 | await bridge.appService.close(); 168 | process.exit(0); 169 | } 170 | catch (ex) { 171 | log.error(`Ungraceful shutdown:`, ex); 172 | process.exit(1); 173 | } 174 | }); 175 | 176 | // Check if the homeserver is up yet. 177 | let ready = false; 178 | await bridge.initalise(); 179 | do { 180 | try { 181 | log.info(`Checking connection to the HS..`); 182 | // Simple call. 183 | await bridge.botClient.getVersions(); 184 | log.info(`HS connection ready`); 185 | break; 186 | } 187 | catch (ex) { 188 | log.error('Could not verify HS connection, retrying in 5s.'); 189 | await new Promise(res => setTimeout(res, 5000)); // Wait 5s before reattempting 190 | } 191 | } while (!ready); 192 | 193 | startSMTP(config); 194 | bridge.listen(port, config); 195 | log.info("Matrix-side listening on port:", port); 196 | }; 197 | 198 | /** 199 | * Returns the email address obtained from Matrix UserId 200 | * @param userId userId in the format `@_email__:localhost`. 201 | * @param homeServer homeServer address. 202 | * @returns {string} Email-ID obtained from bridge userID `localPart@domain.tld`. 203 | */ 204 | function getMailIdFromUserId(userId, homeServer) { 205 | const mUserId = new MatrixUser(userId); 206 | const emailPart = mUserId.localpart.slice('@email_'.length); 207 | const localPart = emailPart.slice(0, emailPart.lastIndexOf('_')); 208 | const domain = emailPart.slice(emailPart.lastIndexOf('_')+1); 209 | return `${localPart}@${domain}`; 210 | } 211 | 212 | /** 213 | * Returns the email address for a provided roomAlias 214 | * @param roomAlias Corresponding Room alias. 215 | * @param mxDomain SMTP listening domain. 216 | * @returns {string} Email address in the format `room+_@matrix.org` 217 | */ 218 | function getRoomMailIdFromRoomAlias(roomAlias, mxDomain) { 219 | const alias = roomAlias.slice(1, roomAlias.lastIndexOf(":")); 220 | const homeServer = roomAlias.slice(roomAlias.lastIndexOf(":")+1); 221 | return `room+${alias}_${homeServer}@${mxDomain}`; 222 | } 223 | 224 | /** 225 | * Returns the email address for a userID 226 | * @param userId Corresponding user ID. 227 | * @param mxDomain SMTP listening domain. 228 | * @returns {string} Email address in the format `room+_@matrix.org` 229 | */ 230 | function getRoomMailIdFromUserId(userId, mxDomain) { 231 | const localPart = userId.slice(1, userId.lastIndexOf(":")); 232 | const homeServer = userId.slice(userId.lastIndexOf(":")+1); 233 | return `user+${localPart}_${homeServer}@${mxDomain}`; 234 | } 235 | -------------------------------------------------------------------------------- /src/email/email.js: -------------------------------------------------------------------------------- 1 | const { SMTPServer } = require("smtp-server"); 2 | const { MailParser } = require('mailparser'); 3 | const ParseEmailAddress = require("email-addresses"); 4 | const { Logging } = require('../log'); 5 | 6 | const log = Logging.get("email"); 7 | 8 | 9 | /*** 10 | * The complete event MUST NOT be larger than 65535 bytes. 11 | * https://spec.matrix.org/unstable/client-server-api/#size-limits 12 | * Using 63k as the maximum text size. 13 | * @type {number} 14 | */ 15 | const MAX_MATRIX_MESSAGE_SIZE = 63000; 16 | 17 | 18 | /*** 19 | * Returns an array containing Room Alias and HomeServer. 20 | * @param {string} rcptTo The `To address` from the received email. 21 | * @param {string} mxDomain The domain name in which the SMTP server is listening. 22 | * @returns {Error|[string, string]} An array containing the alias or userID and homeserver. 23 | */ 24 | const getRoomAliasFromEmailTo = function(rcptTo, mxDomain) { 25 | let localPartRcptTo; 26 | for (let i = 0; i < rcptTo.length; i++) { 27 | localPartRcptTo = ParseEmailAddress.parseOneAddress(rcptTo[i].address).local; 28 | if (localPartRcptTo.endsWith(mxDomain)) { 29 | log.info("Message destination address:", localPartRcptTo); 30 | return getUserIdOrAlias(localPartRcptTo); 31 | } 32 | } 33 | throw Error("To address does not contain a valid recipient"); 34 | }; 35 | 36 | 37 | /*** 38 | * Split a string at given index into an array containing the two parts. 39 | * @param index string index at which the string has to be split. 40 | * @returns {function(string): [string, string]} 41 | */ 42 | const splitAt = index => x => [x.slice(0, index), x.slice(index+1)]; 43 | 44 | 45 | /*** 46 | * Get userID or Room alias based on the localPart of the email address. 47 | * @param {string} localPart 48 | * @returns {Error|[string, string]} 49 | */ 50 | const getUserIdOrAlias = function(localPart) { 51 | // Received room+_hs 52 | if (localPart.startsWith('room+')) { 53 | log.info("Message destination is a room"); 54 | let alias = localPart.substring(localPart.indexOf('+')+1); 55 | if (alias.lastIndexOf('_') >= 1) { 56 | let res = splitAt(alias.lastIndexOf('_'))(alias); 57 | res.unshift("room"); 58 | return res; 59 | } 60 | } 61 | // Received user+_ 62 | else if (localPart.startsWith('user+')) { 63 | log.info("Message destination is a user"); 64 | let uname = localPart.substring(localPart.indexOf('+')+1); 65 | if (uname.lastIndexOf('_') >= 1) { 66 | let res = splitAt(uname.lastIndexOf('_'))(uname); 67 | res.unshift("user"); 68 | return res; 69 | } 70 | } 71 | throw Error("Could not resolve UserID or Alias from the received localPart"); 72 | }; 73 | 74 | 75 | /*** 76 | * Send the inbound mail's contents to the corresponding rooms. 77 | * @param {string} text The text content of the email. 78 | * @param {string} toAdd 79 | * @param {ParsedMailbox} fromAdd Email address of the sender. 80 | * @param {address} from The from address object from email header. 81 | * @param {object} config Bridge configurations. 82 | * @returns {Promise} 83 | */ 84 | async function handleMail(text, toAdd, fromAdd, from, config) { 85 | let alias = "", matrixId = "", roomID = "", receivedAddress; 86 | try { 87 | receivedAddress = getRoomAliasFromEmailTo(toAdd, config.bridge.domain); 88 | if (receivedAddress[0] === "room") { 89 | alias = `#${receivedAddress[1]}:${receivedAddress[2]}`; 90 | } 91 | else if (receivedAddress[0] === "user") { 92 | matrixId = `@${receivedAddress[1]}:${receivedAddress[2]}`; 93 | } 94 | } 95 | catch (ex) { 96 | log.error(`Error resolving room address: ${ex}`); 97 | //TODO: Send a reply to the sender indicating `unable to resolve room` 98 | throw Error("Unable to resolve the room"); 99 | } 100 | if (!text.trim().length) { 101 | // text only contains whitespace (ie. spaces, tabs or line breaks) 102 | log.warn("Inbound email contains whitespace only"); 103 | throw Error(`Inbound email contains whitespace only, ignoring this empty message from ${fromAdd.address}`); 104 | } 105 | 106 | log.info("Inbound email contents: "+ text.substring(0, 10) + "... "); 107 | log.info("Email contents from", fromAdd.address, "will be sent to", 108 | `${receivedAddress[1]}:${receivedAddress[2]}`); 109 | 110 | const fromId = `@_email_${fromAdd.local }_${fromAdd.domain}:${config.bridge.domain}`; 111 | const intent = bridge.getIntent(fromId); 112 | const displayName = from.value[0].name !== "" ? `${from.value[0].name}` : `${fromAdd.address}`; 113 | await intent.setDisplayName(displayName); 114 | let message = Buffer.from(text, "utf-8"); 115 | 116 | if (alias) { 117 | // Destination is a public room. 118 | roomID = await intent.resolveRoom(alias); 119 | } 120 | else { 121 | let dmMappings; 122 | const botClient = bridge.getIntent().getClient(); 123 | // check the DM recipient exists 124 | try { 125 | await botClient.getProfileInfo(matrixId); 126 | } 127 | catch (ex) { 128 | throw Error(`Could not find the recipient user(DM) ${ex}`); 129 | } 130 | // Check the bridge bot's account_data for existing DM relations. 131 | try { 132 | dmMappings = await botClient.getAccountDataFromServer("me.abhy.email-bridge"); 133 | } 134 | catch (ex) { 135 | throw Error(`Could not get DM room mappings from bot's account data ${ex}`); 136 | } 137 | if (matrixId in dmMappings) { 138 | roomID = dmMappings[matrixId].roomId; 139 | } 140 | else { 141 | try { 142 | // Create a new DM and invite the [m] user 143 | roomID = (await intent.createRoom({ 144 | createAsClient: true, 145 | options: { 146 | name: (displayName + " (PM via email)"), 147 | visibility: "private", 148 | creation_content: { 149 | "m.federate": true 150 | }, 151 | // preset: "trusted_private_chat", 152 | is_direct: true, 153 | invite: [matrixId], 154 | initial_state: [{ 155 | content: { 156 | users: { 157 | [matrixId]: 10, 158 | [fromId]: 100, 159 | }, 160 | events: { 161 | "m.room.avatar": 10, 162 | "m.room.name": 10, 163 | "m.room.canonical_alias": 100, 164 | "m.room.history_visibility": 100, 165 | "m.room.power_levels": 100, 166 | "m.room.encryption": 100 167 | }, 168 | invite: 100, 169 | }, 170 | type: "m.room.power_levels", 171 | state_key: "", 172 | }], 173 | } 174 | })).room_id; 175 | } 176 | catch (ex) { 177 | throw Error(`Could not create a new DM with ${fromAdd.address} & ${matrixId}: ${ex}`); 178 | } 179 | try { 180 | // Store the mappings to the bot's account data. 181 | await botClient.setAccountData("me.abhy.email-bridge", { 182 | ...dmMappings, 183 | [matrixId]: { 184 | "roomId": roomID, 185 | "emailUser": fromId, 186 | } 187 | }); 188 | } 189 | catch (ex) { 190 | // (Leave the created room -> inform the sender?) 191 | throw Error("Could not store update the DM mappings"); 192 | } 193 | } 194 | } 195 | if (!roomID.startsWith("!")) { 196 | throw Error("Could not resolve roomID from given alias"); 197 | } 198 | 199 | if (message.byteLength > MAX_MATRIX_MESSAGE_SIZE) { 200 | // split text to under `MAX_MATRIX_MESSAGE_SIZE` and send as separate events. 201 | log.info("Mail contents greater than 63k"); 202 | for (let i = 0; i { 235 | subject = headers.get('subject'); 236 | from = headers.get('from'); 237 | }); 238 | 239 | mailparser.on('data', data => { 240 | if (data.type === 'text') { 241 | text = data.text; 242 | } 243 | }); 244 | 245 | mailparser.on('end', () => { 246 | handleMail(text, toAddress, fromAddress, from, config) 247 | .then(() => log.info("Message sent to the room")) 248 | .catch((err) => log.error(`Could not handle mail:`, err)); 249 | }); 250 | 251 | stream.pipe(mailparser); 252 | stream.on('end', callback); 253 | } 254 | }); 255 | SMTP.listen(config.email.inboundPort); 256 | }; 257 | -------------------------------------------------------------------------------- /src/email/index.js: -------------------------------------------------------------------------------- 1 | const { startSMTP } = require('./email'); 2 | 3 | exports.startSMTP = startSMTP; 4 | -------------------------------------------------------------------------------- /src/email/outbound.js: -------------------------------------------------------------------------------- 1 | const { createConnection } = require('net'); 2 | const { Resolver } = require('dns').promises; 3 | const { DKIMSign } = require('dkim-signer'); 4 | const { connect, createSecureContext } = require('tls'); 5 | const MailComposer = require("nodemailer/lib/mail-composer"); 6 | const ParseEmailAddress = require("email-addresses"); 7 | 8 | const { Logging } = require('../log'); 9 | const resolver = new Resolver(); 10 | const log = Logging.get("outbound"); 11 | const CRLF = '\r\n'; 12 | 13 | const smtpCodes = { 14 | ServiceReady: 220, 15 | Bye: 221, 16 | AuthSuccess: 235, 17 | OperationOK: 250, 18 | ForwardNonLocalUser: 251, 19 | StartMailBody: 354, 20 | ServerChallenge: 334, 21 | NegativeCompletion:400, 22 | }; 23 | 24 | module.exports = function (options) { 25 | options = options || {}; 26 | const dkimEnabled = options.dkimEnabled || false; 27 | const dkimPrivateKey = (options.dkim || {}).privateKey; 28 | const dkimKeySelector = (options.dkim || {}).keySelector || 'dkim'; 29 | const smtpPort = options.smtpPort || 25; 30 | const smtpHost = options.smtpHost; 31 | const rejectUnauthorized = options.rejectUnauthorized; 32 | const startTLS = options.startTLS || false; 33 | const tls = options.tls || {}; 34 | 35 | // group recipients by domain(to limit the number of connections per 'To' domain). 36 | function groupRecipients(recipients) { 37 | const groups = {}; 38 | for (const recipient of recipients) { 39 | const host = ParseEmailAddress.parseOneAddress(recipient).domain; 40 | (groups[host] || (groups[host] = [])).push(recipient); 41 | } 42 | return groups; 43 | } 44 | 45 | /*** 46 | * Get email addresses from address string. 47 | * @param addresses 48 | * @returns {*[]} 49 | */ 50 | function getAddresses(addresses) { 51 | const results = []; 52 | if (!Array.isArray(addresses)) { 53 | addresses = addresses.split(','); 54 | } 55 | const addressesLength = addresses.length; 56 | for (let i = 0; i < addressesLength; i++) { 57 | results.push(ParseEmailAddress.parseOneAddress(addresses[i]).address); 58 | } 59 | return results; 60 | } 61 | 62 | /** 63 | * Resolve MX records by domain. 64 | * @param {string} domain 65 | * @returns {Promise} 66 | */ 67 | async function resolveMX(domain) { 68 | let resolvedMX = []; 69 | if (smtpHost !== '') { 70 | resolvedMX.push({ exchange: smtpHost }); 71 | return resolvedMX; 72 | } 73 | try { 74 | resolvedMX = await resolver.resolveMx(domain); 75 | resolvedMX.sort(function (a, b) { return a.priority - b.priority; }); 76 | return resolvedMX; 77 | } 78 | catch (ex) { 79 | throw Error(`Failed to resolve MX for ${domain}: ${ex}`); 80 | } 81 | } 82 | 83 | /*** 84 | * Send email using SMTP. 85 | * @param domain The to address. 86 | * @param srcHost Source hostname. 87 | * @param from Source from address. 88 | * @param recipients Recipients list. 89 | * @param body Email body 90 | * @returns {Promise} 91 | */ 92 | async function sendToSMTP(domain, srcHost, from, recipients, body) { 93 | const resolvedMX = await resolveMX(domain); 94 | log.info("Resolved mx list:", resolvedMX); 95 | let sock, msg = ''; 96 | function tryConnect(i) { 97 | if (i >= resolvedMX.length) { 98 | throw Error(`Could not connect to any SMTP server for ${domain}`); 99 | } 100 | 101 | sock = createConnection(smtpPort, resolvedMX[i].exchange); 102 | 103 | sock.on('error', function (err) { 104 | log.error('Error on connectMx for: ', resolvedMX[i], err); 105 | tryConnect(++i); 106 | }); 107 | 108 | sock.on('connect', function () { 109 | log.debug('MX connection created: ', resolvedMX[i].exchange); 110 | sock.removeAllListeners('error'); 111 | return sock; 112 | }); 113 | } 114 | tryConnect(0); 115 | 116 | function onLine(line) { 117 | log.debug('RECV ' + domain + '>' + line); 118 | 119 | msg += (line + CRLF); 120 | 121 | if (line[3] === ' ') { 122 | // 250-information dash is not complete. 123 | // 250 OK. space is complete. 124 | let lineNumber = parseInt(line.substr(0, 3)); 125 | response(lineNumber, msg); 126 | msg = ''; 127 | } 128 | } 129 | 130 | function writeToSocket(s) { 131 | log.debug(`SEND ${domain}> ${s}`); 132 | sock.write(s + CRLF); 133 | } 134 | 135 | sock.setEncoding('utf8'); 136 | 137 | sock.on('data', function (chunk) { 138 | data += chunk; 139 | parts = data.split(CRLF); 140 | const partsLength = parts.length - 1; 141 | for (let i = 0, len = partsLength; i < len; i++) { 142 | onLine(parts[i]); 143 | } 144 | data = parts[parts.length - 1]; 145 | }); 146 | 147 | sock.on('error', function (err) { 148 | throw Error(`Failed to connect to ${domain}: ${err}`); 149 | }); 150 | 151 | let data = ''; 152 | let step = 0; 153 | let loginStep = 0; 154 | const queue = []; 155 | // const login = []; 156 | let parts; 157 | let cmd; 158 | let upgraded = false; 159 | let isUpgradeInProgress = false; 160 | 161 | queue.push('MAIL FROM:<' + from + '>'); 162 | const recipientsLength = recipients.length; 163 | for (let i = 0; i < recipientsLength; i++) { 164 | queue.push('RCPT TO:<' + recipients[i] + '>'); 165 | } 166 | queue.push('DATA'); 167 | queue.push('QUIT'); 168 | queue.push(''); 169 | 170 | function response(code, msg) { 171 | switch (code) { 172 | case smtpCodes.ServiceReady: 173 | //220 on server ready 174 | if (isUpgradeInProgress === true) { 175 | sock.removeAllListeners('data'); 176 | let original = sock; 177 | original.pause(); 178 | let opts = { 179 | socket: sock, 180 | host: sock._host, 181 | rejectUnauthorized, 182 | }; 183 | if (startTLS) { 184 | opts.secureContext = createSecureContext({ cert: tls.cert, key: tls.key }); 185 | } 186 | sock = connect( 187 | opts, 188 | () => { 189 | sock.on('data', function (chunk) { 190 | data += chunk; 191 | parts = data.split(CRLF); 192 | const partsLength = parts.length - 1; 193 | for (let i = 0, len = partsLength; i < len; i++) { 194 | onLine(parts[i]); 195 | } 196 | data = parts[parts.length - 1]; 197 | }); 198 | sock.removeAllListeners('close'); 199 | sock.removeAllListeners('end'); 200 | } 201 | ); 202 | sock.on('error', function (err) { 203 | log.warn("Could not upgrade to TLS:", err, "Falling back to plaintext"); 204 | }); 205 | // Resume plaintext connection 206 | original.resume(); 207 | upgraded = true; 208 | writeToSocket("EHLO " + srcHost); 209 | break; 210 | } 211 | else { 212 | // check for ESMTP/ignore-case 213 | if (/\besmtp\b/i.test(msg)) { 214 | // TODO: determine AUTH type; auth login, auth crm-md5, auth plain 215 | cmd = 'EHLO'; 216 | } 217 | else { 218 | upgraded = true; 219 | cmd = 'HELO'; 220 | } 221 | writeToSocket(`${cmd} ${srcHost}`); 222 | break; 223 | } 224 | case smtpCodes.Bye: 225 | // BYE 226 | sock.end(); 227 | log.info("message sent successfully", msg); 228 | break; 229 | case smtpCodes.AuthSuccess: // Verify OK 230 | case smtpCodes.OperationOK: // Operation OK 231 | if (upgraded !== true) { 232 | // check for STARTTLS/ignore-case 233 | if (/\bSTARTTLS\b/i.test(msg) && options.startTLS) { 234 | log.debug("Server supports STARTTLS, continuing"); 235 | writeToSocket('STARTTLS'); 236 | isUpgradeInProgress = true; 237 | break; 238 | } 239 | else { 240 | upgraded = true; 241 | log.debug("No STARTTLS support or ignored, continuing"); 242 | } 243 | } 244 | writeToSocket(queue[step]); 245 | step++; 246 | break; 247 | 248 | case smtpCodes.ForwardNonLocalUser: 249 | // User not local; will forward. 250 | if (step === queue.length - 1) { 251 | log.info('OK:', code, msg); 252 | return; 253 | } 254 | writeToSocket(queue[step]); 255 | step++; 256 | break; 257 | 258 | case smtpCodes.StartMailBody: 259 | // Start mail input 260 | // Inform end by `.` 261 | log.info('Sending mail body', body); 262 | writeToSocket(body); 263 | writeToSocket(''); 264 | writeToSocket('.'); 265 | break; 266 | 267 | case smtpCodes.ServerChallenge: 268 | // Send login details [for relay] 269 | // TODO: support login. 270 | writeToSocket(login[loginStep]); 271 | loginStep++; 272 | break; 273 | 274 | default: 275 | if (code >= smtpCodes.NegativeCompletion) { 276 | log.error('SMTP server responds with error code', code); 277 | sock.end(); 278 | throw Error(`SMTP server responded with code: ${code} + ${msg}`); 279 | } 280 | } 281 | } 282 | } 283 | 284 | /** 285 | * Send Mail directly 286 | * @param mail Mail object containing message, to/from etc. 287 | * Complete attributes reference: https://nodemailer.com/extras/mailcomposer/#e-mail-message-fields 288 | * @returns {Promise} 289 | */ 290 | async function sendmail(mail) { 291 | const mailMe = new MailComposer(mail); 292 | let recipients = []; 293 | if (mail.to) { 294 | recipients = recipients.concat(getAddresses(mail.to)); 295 | } 296 | 297 | if (mail.cc) { 298 | recipients = recipients.concat(getAddresses(mail.cc)); 299 | } 300 | 301 | if (mail.bcc) { 302 | recipients = recipients.concat(getAddresses(mail.bcc)); 303 | } 304 | 305 | const groups = groupRecipients(recipients); 306 | const from = ParseEmailAddress.parseOneAddress(mail.from).address; 307 | const srcHost = ParseEmailAddress.parseOneAddress(from).domain; 308 | 309 | await mailMe.compile().build(async function (err, message) { 310 | if (err) { 311 | throw Error(`Error on building the message: ${err}`); 312 | } 313 | if (dkimEnabled) { 314 | // eslint-disable-next-line new-cap 315 | const signature = DKIMSign(message, { 316 | privateKey: dkimPrivateKey, 317 | keySelector: dkimKeySelector, 318 | domainName: srcHost 319 | }); 320 | message = signature + CRLF + message; 321 | } 322 | // eslint-disable-next-line guard-for-in 323 | for (let domain in groups) { 324 | try { 325 | await sendToSMTP(domain, srcHost, from, groups[domain], message); 326 | } 327 | catch (ex) { 328 | log.error(`Could not send email to ${domain}: ${ex}`); 329 | } 330 | } 331 | }); 332 | } 333 | return sendmail; 334 | }; 335 | -------------------------------------------------------------------------------- /src/log.js: -------------------------------------------------------------------------------- 1 | const { Logging } = require("matrix-appservice-bridge"); 2 | 3 | Logging.configure({ 4 | // A level to set the console reporter to. 5 | console: "debug", 6 | 7 | // // Format to append to log files. 8 | // fileDatePattern: "DD-MM-YYYY", 9 | 10 | // // Format of the timestamp in log files. 11 | // timestampFormat: "MMM-D HH:mm:ss.SSS", 12 | 13 | // // Log files to emit to, keyed of the minimum level they report. 14 | // // You can leave this out, or set it to false to disable files. 15 | // files: { 16 | // // File paths can be relative or absolute, the date is appended onto the end. 17 | // "info.log": "info", 18 | // }, 19 | 20 | // // The maximum number of files per level before old files get cleaned 21 | // // up. Use 0 to disable. 22 | // maxFiles: 5, 23 | }) 24 | 25 | exports.Logging = Logging; 26 | --------------------------------------------------------------------------------