├── .gitignore ├── LICENSE ├── README.md ├── index.d.ts ├── package-lock.json ├── package.json ├── src ├── api │ ├── index.js │ ├── sendFile.js │ └── sendWebhook.js ├── classes │ ├── messageBuilder.js │ └── webhook.js ├── index.js ├── index.mjs └── utils │ └── index.js └── test └── test.js /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules/ 2 | .env 3 | .DS_Store 4 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2019 matthew1232 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Discord Webhook sending 2 | ![version](https://img.shields.io/npm/v/discord-webhook-node "Version") 3 | ![npm](https://img.shields.io/npm/dt/discord-webhook-node.svg "Total Downloads") 4 | [![Total alerts](https://img.shields.io/lgtm/alerts/g/matthew1232/discord-webhook-node.svg?logo=lgtm&logoWidth=18)](https://lgtm.com/projects/g/matthew1232/discord-webhook-node/alerts/) 5 | [![Language grade: JavaScript](https://img.shields.io/lgtm/grade/javascript/g/matthew1232/discord-webhook-node.svg?logo=lgtm&logoWidth=18)](https://lgtm.com/projects/g/matthew1232/discord-webhook-node/context:javascript) 6 | 7 | - [Installation](#installation) 8 | - [Examples](#examples) 9 | - [Basic use](#basic-use) 10 | - [Custom embeds](#custom-embeds) 11 | - [Sending files](#sending-files) 12 | - [Preset messages](#preset-messages) 13 | - [Custom settings](#custom-settings) 14 | - [Notes](#notes) 15 | - [API](#api) 16 | - [Webhook](#webhook---class) 17 | - [MessageBuilder](#messagebuilder---class) 18 | - [License](#license) 19 | 20 | # Installation 21 | ```npm install discord-webhook-node``` or ```yarn add discord-webhook-node``` 22 | 23 | # Examples 24 | 25 | ## Basic use 26 | ```js 27 | const { Webhook } = require('discord-webhook-node'); 28 | const hook = new Webhook("YOUR WEBHOOK URL"); 29 | 30 | const IMAGE_URL = 'https://homepages.cae.wisc.edu/~ece533/images/airplane.png'; 31 | hook.setUsername('Discord Webhook Node Name'); 32 | hook.setAvatar(IMAGE_URL); 33 | 34 | hook.send("Hello there!"); 35 | ``` 36 | 37 | ## Custom embeds 38 | ```js 39 | const { Webhook, MessageBuilder } = require('discord-webhook-node'); 40 | const hook = new Webhook("YOUR WEBHOOK URL"); 41 | 42 | const embed = new MessageBuilder() 43 | .setTitle('My title here') 44 | .setAuthor('Author here', 'https://cdn.discordapp.com/embed/avatars/0.png', 'https://www.google.com') 45 | .setURL('https://www.google.com') 46 | .addField('First field', 'this is inline', true) 47 | .addField('Second field', 'this is not inline') 48 | .setColor('#00b0f4') 49 | .setThumbnail('https://cdn.discordapp.com/embed/avatars/0.png') 50 | .setDescription('Oh look a description :)') 51 | .setImage('https://cdn.discordapp.com/embed/avatars/0.png') 52 | .setFooter('Hey its a footer', 'https://cdn.discordapp.com/embed/avatars/0.png') 53 | .setTimestamp(); 54 | 55 | hook.send(embed); 56 | ``` 57 | 58 | Keep in mind that the custom embed method `setColor` takes in a decimal color/a hex color (pure black and white hex colors will be innacurate). You can convert hex colors to decimal using this website here: [https://convertingcolors.com](https://convertingcolors.com) 59 | 60 | ## Sending files 61 | ```js 62 | const { Webhook } = require('discord-webhook-node'); 63 | const hook = new Webhook('YOUR WEBHOOK URL'); 64 | 65 | hook.sendFile('../yourfilename.png'); 66 | ``` 67 | 68 | ## Preset messages 69 | ```js 70 | const { Webhook } = require('discord-webhook-node'); 71 | const hook = new Webhook('YOUR WEBHOOK URL'); 72 | 73 | //Sends an information message 74 | hook.info('**Information hook**', 'Information field title here', 'Information field value here'); 75 | 76 | //Sends a success message 77 | hook.success('**Success hook**', 'Success field title here', 'Success field value here'); 78 | 79 | //Sends an warning message 80 | hook.warning('**Warning hook**', 'Warning field title here', 'Warning field value here'); 81 | 82 | //Sends an error message 83 | hook.error('**Error hook**', 'Error field title here', 'Error field value here'); 84 | ``` 85 | 86 | ## Custom settings 87 | ```js 88 | const { Webhook } = require('discord-webhook-node'); 89 | const hook = new Webhook({ 90 | url: "YOUR WEBHOOK URL", 91 | //If throwErrors is set to false, no errors will be thrown if there is an error sending 92 | throwErrors: false, 93 | //retryOnLimit gives you the option to not attempt to send the message again if rate limited 94 | retryOnLimit: false 95 | }); 96 | 97 | hook.setUsername('Username'); //Overrides the default webhook username 98 | hook.setAvatar('YOUR_AVATAR_URL'); //Overrides the default webhook avatar 99 | ``` 100 | 101 | # Notes 102 | discord-webhook-node is a promise based library, which means you can use `.catch`, `.then`, and `await`, although if successful will not return any values. For example: 103 | 104 | ```js 105 | const { Webhook } = require('discord-webhook-node'); 106 | const hook = new Webhook("YOUR WEBHOOK URL"); 107 | 108 | hook.send("Hello there!") 109 | .then(() => console.log('Sent webhook successfully!')) 110 | .catch(err => console.log(err.message)); 111 | ``` 112 | 113 | or using async: 114 | ```js 115 | const { Webhook } = require('discord-webhook-node'); 116 | const hook = new Webhook("YOUR WEBHOOK URL"); 117 | 118 | (async () => { 119 | try { 120 | await hook.send('Hello there!'); 121 | console.log('Successfully sent webhook!'); 122 | } 123 | catch(e){ 124 | console.log(e.message); 125 | }; 126 | })(); 127 | ``` 128 | 129 | By default, it will handle Discord's rate limiting, and if there is an error sending the message (other than rate limiting), an error will be thrown. You can change these options with the custom settings options below. 130 | 131 | # API 132 | ## Webhook - class 133 | Constructor 134 | - options (optional) : object 135 | - throwErrors (optional) : boolean 136 | - retryOnLimit (optional) : boolean 137 | 138 | Methods 139 | - setUsername(username : string) returns this 140 | - setAvatar(avatarURL : string (image url)) returns this 141 | - async sendFile(filePath : string) 142 | - async send(payload : string/MessageBuilder) 143 | - async info(title : string, fieldName (optional) : string, fieldValue (optional) : string, inline (optional) : boolean) 144 | - async success(title : string, fieldName (optional) : string, fieldValue (optional) : string, inline (optional) : boolean) 145 | - async warning(title : string, fieldName (optional) : string, fieldValue (optional) : string, inline (optional) : boolean) 146 | - async error(title : string, fieldName (optional) : string, fieldValue (optional) : string, inline (optional) : boolean) 147 | - 148 | ## MessageBuilder - class 149 | Methods 150 | - setText(text: string) 151 | - setAuthor(author: string (text), authorImage (optional) : string (image url), authorUrl (optional) : string (link)) 152 | - setTitle(title: string) 153 | - setURL(url: string) 154 | - setThumbnail(thumbnail : string (image url)) 155 | - setImage(image : string (image url)) 156 | - setTimestamp(date (optional) number/date object) 157 | - setColor(color : string/number (hex or decimal color)) 158 | - setDescription(description : string) 159 | - addField(fieldName : string, fieldValue: string, inline (optional) : boolean) 160 | - setFooter(footer : string, footerImage (optional) : string (image url)) 161 | 162 | # License 163 | 164 | MIT -------------------------------------------------------------------------------- /index.d.ts: -------------------------------------------------------------------------------- 1 | export type WebhookOptions = { 2 | url: string; 3 | throwErrors?: boolean; 4 | retryOnLimit?: boolean; 5 | }; 6 | 7 | export class Webhook { 8 | constructor(options: string | WebhookOptions); 9 | setUsername(username: string): void; 10 | setAvatar(avatar: string): void; 11 | sendFile(filePath: string): Promise; 12 | send(message: MessageBuilder | string): Promise; 13 | info( 14 | title: string, 15 | fieldName?: string, 16 | fieldValue?: string, 17 | inline?: boolean 18 | ): Promise; 19 | success( 20 | title: string, 21 | fieldName?: string, 22 | fieldValue?: string, 23 | inline?: boolean 24 | ): Promise; 25 | warning( 26 | title: string, 27 | fieldName?: string, 28 | fieldValue?: string, 29 | inline?: boolean 30 | ): Promise; 31 | error( 32 | title: string, 33 | fieldName?: string, 34 | fieldValue?: string, 35 | inline?: boolean 36 | ): Promise; 37 | } 38 | 39 | export type WebhookField = { 40 | name: string; 41 | value: string; 42 | inline?: boolean; 43 | }; 44 | 45 | export type WebhookEmbed = { 46 | author?: { 47 | name?: string; 48 | url?: string; 49 | icon_url?: string; 50 | }; 51 | title?: string; 52 | url?: string; 53 | thumbnail?: { 54 | url?: string; 55 | }; 56 | image?: { 57 | url?: string; 58 | }; 59 | timestamp?: Date; 60 | color?: number; 61 | description?: string; 62 | fields: WebhookField[]; 63 | footer?: { 64 | text: string; 65 | icon_url?: string; 66 | }; 67 | }; 68 | 69 | export type WebhookPayload = { 70 | embeds: WebhookEmbed[]; 71 | }; 72 | 73 | export class MessageBuilder { 74 | constructor(); 75 | getJSON(): WebhookPayload; 76 | setText(text: string): this; 77 | setAuthor(author?: string, authorImage?: string, authorUrl?: string): this; 78 | setTitle(title: string): this; 79 | setUrl(url: string): this; 80 | setThumbnail(thumbnailUrl: string): this; 81 | setImage(image: string): this; 82 | setTimestamp(): this; 83 | setColor(color: number): this; 84 | setDescription(description: string): this; 85 | addField(fieldName: string, fieldValue: string, inline?: boolean): this; 86 | setFooter(footer: string, footerImage?: string): this; 87 | } -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "discord-webhook-node", 3 | "version": "1.1.1", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "ansi-colors": { 8 | "version": "4.1.1", 9 | "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", 10 | "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", 11 | "dev": true 12 | }, 13 | "ansi-regex": { 14 | "version": "3.0.0", 15 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", 16 | "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", 17 | "dev": true 18 | }, 19 | "ansi-styles": { 20 | "version": "3.2.1", 21 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 22 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 23 | "dev": true, 24 | "requires": { 25 | "color-convert": "^1.9.0" 26 | } 27 | }, 28 | "anymatch": { 29 | "version": "3.1.1", 30 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", 31 | "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", 32 | "dev": true, 33 | "requires": { 34 | "normalize-path": "^3.0.0", 35 | "picomatch": "^2.0.4" 36 | } 37 | }, 38 | "argparse": { 39 | "version": "1.0.10", 40 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 41 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 42 | "dev": true, 43 | "requires": { 44 | "sprintf-js": "~1.0.2" 45 | } 46 | }, 47 | "array.prototype.map": { 48 | "version": "1.0.2", 49 | "resolved": "https://registry.npmjs.org/array.prototype.map/-/array.prototype.map-1.0.2.tgz", 50 | "integrity": "sha512-Az3OYxgsa1g7xDYp86l0nnN4bcmuEITGe1rbdEBVkrqkzMgDcbdQ2R7r41pNzti+4NMces3H8gMmuioZUilLgw==", 51 | "dev": true, 52 | "requires": { 53 | "define-properties": "^1.1.3", 54 | "es-abstract": "^1.17.0-next.1", 55 | "es-array-method-boxes-properly": "^1.0.0", 56 | "is-string": "^1.0.4" 57 | } 58 | }, 59 | "asynckit": { 60 | "version": "0.4.0", 61 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 62 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" 63 | }, 64 | "balanced-match": { 65 | "version": "1.0.0", 66 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 67 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 68 | "dev": true 69 | }, 70 | "binary-extensions": { 71 | "version": "2.1.0", 72 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz", 73 | "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==", 74 | "dev": true 75 | }, 76 | "brace-expansion": { 77 | "version": "1.1.11", 78 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 79 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 80 | "dev": true, 81 | "requires": { 82 | "balanced-match": "^1.0.0", 83 | "concat-map": "0.0.1" 84 | } 85 | }, 86 | "braces": { 87 | "version": "3.0.2", 88 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 89 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 90 | "dev": true, 91 | "requires": { 92 | "fill-range": "^7.0.1" 93 | } 94 | }, 95 | "browser-stdout": { 96 | "version": "1.3.1", 97 | "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz", 98 | "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", 99 | "dev": true 100 | }, 101 | "camelcase": { 102 | "version": "5.3.1", 103 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 104 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 105 | "dev": true 106 | }, 107 | "chalk": { 108 | "version": "2.4.2", 109 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 110 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 111 | "dev": true, 112 | "requires": { 113 | "ansi-styles": "^3.2.1", 114 | "escape-string-regexp": "^1.0.5", 115 | "supports-color": "^5.3.0" 116 | }, 117 | "dependencies": { 118 | "supports-color": { 119 | "version": "5.5.0", 120 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 121 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 122 | "dev": true, 123 | "requires": { 124 | "has-flag": "^3.0.0" 125 | } 126 | } 127 | } 128 | }, 129 | "chokidar": { 130 | "version": "3.3.1", 131 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.3.1.tgz", 132 | "integrity": "sha512-4QYCEWOcK3OJrxwvyyAOxFuhpvOVCYkr33LPfFNBjAD/w3sEzWsp2BUOkI4l9bHvWioAd0rc6NlHUOEaWkTeqg==", 133 | "dev": true, 134 | "requires": { 135 | "anymatch": "~3.1.1", 136 | "braces": "~3.0.2", 137 | "fsevents": "~2.1.2", 138 | "glob-parent": "~5.1.0", 139 | "is-binary-path": "~2.1.0", 140 | "is-glob": "~4.0.1", 141 | "normalize-path": "~3.0.0", 142 | "readdirp": "~3.3.0" 143 | } 144 | }, 145 | "cliui": { 146 | "version": "5.0.0", 147 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", 148 | "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", 149 | "dev": true, 150 | "requires": { 151 | "string-width": "^3.1.0", 152 | "strip-ansi": "^5.2.0", 153 | "wrap-ansi": "^5.1.0" 154 | }, 155 | "dependencies": { 156 | "ansi-regex": { 157 | "version": "4.1.0", 158 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 159 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 160 | "dev": true 161 | }, 162 | "string-width": { 163 | "version": "3.1.0", 164 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 165 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 166 | "dev": true, 167 | "requires": { 168 | "emoji-regex": "^7.0.1", 169 | "is-fullwidth-code-point": "^2.0.0", 170 | "strip-ansi": "^5.1.0" 171 | } 172 | }, 173 | "strip-ansi": { 174 | "version": "5.2.0", 175 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 176 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 177 | "dev": true, 178 | "requires": { 179 | "ansi-regex": "^4.1.0" 180 | } 181 | } 182 | } 183 | }, 184 | "color-convert": { 185 | "version": "1.9.3", 186 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 187 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 188 | "dev": true, 189 | "requires": { 190 | "color-name": "1.1.3" 191 | } 192 | }, 193 | "color-name": { 194 | "version": "1.1.3", 195 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 196 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 197 | "dev": true 198 | }, 199 | "combined-stream": { 200 | "version": "1.0.8", 201 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 202 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 203 | "requires": { 204 | "delayed-stream": "~1.0.0" 205 | } 206 | }, 207 | "concat-map": { 208 | "version": "0.0.1", 209 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 210 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 211 | "dev": true 212 | }, 213 | "debug": { 214 | "version": "3.2.6", 215 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", 216 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", 217 | "dev": true, 218 | "requires": { 219 | "ms": "^2.1.1" 220 | } 221 | }, 222 | "decamelize": { 223 | "version": "1.2.0", 224 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", 225 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", 226 | "dev": true 227 | }, 228 | "define-properties": { 229 | "version": "1.1.3", 230 | "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", 231 | "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", 232 | "dev": true, 233 | "requires": { 234 | "object-keys": "^1.0.12" 235 | } 236 | }, 237 | "delayed-stream": { 238 | "version": "1.0.0", 239 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 240 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=" 241 | }, 242 | "diff": { 243 | "version": "4.0.2", 244 | "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", 245 | "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", 246 | "dev": true 247 | }, 248 | "dotenv": { 249 | "version": "8.2.0", 250 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz", 251 | "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw==", 252 | "dev": true 253 | }, 254 | "emoji-regex": { 255 | "version": "7.0.3", 256 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 257 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 258 | "dev": true 259 | }, 260 | "es-abstract": { 261 | "version": "1.17.6", 262 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.6.tgz", 263 | "integrity": "sha512-Fr89bON3WFyUi5EvAeI48QTWX0AyekGgLA8H+c+7fbfCkJwRWRMLd8CQedNEyJuoYYhmtEqY92pgte1FAhBlhw==", 264 | "dev": true, 265 | "requires": { 266 | "es-to-primitive": "^1.2.1", 267 | "function-bind": "^1.1.1", 268 | "has": "^1.0.3", 269 | "has-symbols": "^1.0.1", 270 | "is-callable": "^1.2.0", 271 | "is-regex": "^1.1.0", 272 | "object-inspect": "^1.7.0", 273 | "object-keys": "^1.1.1", 274 | "object.assign": "^4.1.0", 275 | "string.prototype.trimend": "^1.0.1", 276 | "string.prototype.trimstart": "^1.0.1" 277 | } 278 | }, 279 | "es-array-method-boxes-properly": { 280 | "version": "1.0.0", 281 | "resolved": "https://registry.npmjs.org/es-array-method-boxes-properly/-/es-array-method-boxes-properly-1.0.0.tgz", 282 | "integrity": "sha512-wd6JXUmyHmt8T5a2xreUwKcGPq6f1f+WwIJkijUqiGcJz1qqnZgP6XIK+QyIWU5lT7imeNxUll48bziG+TSYcA==", 283 | "dev": true 284 | }, 285 | "es-get-iterator": { 286 | "version": "1.1.0", 287 | "resolved": "https://registry.npmjs.org/es-get-iterator/-/es-get-iterator-1.1.0.tgz", 288 | "integrity": "sha512-UfrmHuWQlNMTs35e1ypnvikg6jCz3SK8v8ImvmDsh36fCVUR1MqoFDiyn0/k52C8NqO3YsO8Oe0azeesNuqSsQ==", 289 | "dev": true, 290 | "requires": { 291 | "es-abstract": "^1.17.4", 292 | "has-symbols": "^1.0.1", 293 | "is-arguments": "^1.0.4", 294 | "is-map": "^2.0.1", 295 | "is-set": "^2.0.1", 296 | "is-string": "^1.0.5", 297 | "isarray": "^2.0.5" 298 | } 299 | }, 300 | "es-to-primitive": { 301 | "version": "1.2.1", 302 | "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", 303 | "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", 304 | "dev": true, 305 | "requires": { 306 | "is-callable": "^1.1.4", 307 | "is-date-object": "^1.0.1", 308 | "is-symbol": "^1.0.2" 309 | } 310 | }, 311 | "escape-string-regexp": { 312 | "version": "1.0.5", 313 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 314 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 315 | "dev": true 316 | }, 317 | "esprima": { 318 | "version": "4.0.1", 319 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 320 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", 321 | "dev": true 322 | }, 323 | "fill-range": { 324 | "version": "7.0.1", 325 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 326 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 327 | "dev": true, 328 | "requires": { 329 | "to-regex-range": "^5.0.1" 330 | } 331 | }, 332 | "find-up": { 333 | "version": "4.1.0", 334 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", 335 | "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", 336 | "dev": true, 337 | "requires": { 338 | "locate-path": "^5.0.0", 339 | "path-exists": "^4.0.0" 340 | } 341 | }, 342 | "flat": { 343 | "version": "4.1.0", 344 | "resolved": "https://registry.npmjs.org/flat/-/flat-4.1.0.tgz", 345 | "integrity": "sha512-Px/TiLIznH7gEDlPXcUD4KnBusa6kR6ayRUVcnEAbreRIuhkqow/mun59BuRXwoYk7ZQOLW1ZM05ilIvK38hFw==", 346 | "dev": true, 347 | "requires": { 348 | "is-buffer": "~2.0.3" 349 | } 350 | }, 351 | "form-data": { 352 | "version": "3.0.0", 353 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.0.tgz", 354 | "integrity": "sha512-CKMFDglpbMi6PyN+brwB9Q/GOw0eAnsrEZDgcsH5Krhz5Od/haKHAX0NmQfha2zPPz0JpWzA7GJHGSnvCRLWsg==", 355 | "requires": { 356 | "asynckit": "^0.4.0", 357 | "combined-stream": "^1.0.8", 358 | "mime-types": "^2.1.12" 359 | } 360 | }, 361 | "fs.realpath": { 362 | "version": "1.0.0", 363 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 364 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", 365 | "dev": true 366 | }, 367 | "fsevents": { 368 | "version": "2.1.3", 369 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", 370 | "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", 371 | "dev": true, 372 | "optional": true 373 | }, 374 | "function-bind": { 375 | "version": "1.1.1", 376 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 377 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", 378 | "dev": true 379 | }, 380 | "get-caller-file": { 381 | "version": "2.0.5", 382 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 383 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 384 | "dev": true 385 | }, 386 | "glob": { 387 | "version": "7.1.6", 388 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", 389 | "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", 390 | "dev": true, 391 | "requires": { 392 | "fs.realpath": "^1.0.0", 393 | "inflight": "^1.0.4", 394 | "inherits": "2", 395 | "minimatch": "^3.0.4", 396 | "once": "^1.3.0", 397 | "path-is-absolute": "^1.0.0" 398 | } 399 | }, 400 | "glob-parent": { 401 | "version": "5.1.1", 402 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", 403 | "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", 404 | "dev": true, 405 | "requires": { 406 | "is-glob": "^4.0.1" 407 | } 408 | }, 409 | "growl": { 410 | "version": "1.10.5", 411 | "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz", 412 | "integrity": "sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==", 413 | "dev": true 414 | }, 415 | "has": { 416 | "version": "1.0.3", 417 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 418 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 419 | "dev": true, 420 | "requires": { 421 | "function-bind": "^1.1.1" 422 | } 423 | }, 424 | "has-flag": { 425 | "version": "3.0.0", 426 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 427 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 428 | "dev": true 429 | }, 430 | "has-symbols": { 431 | "version": "1.0.1", 432 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", 433 | "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==", 434 | "dev": true 435 | }, 436 | "he": { 437 | "version": "1.2.0", 438 | "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", 439 | "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", 440 | "dev": true 441 | }, 442 | "inflight": { 443 | "version": "1.0.6", 444 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 445 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 446 | "dev": true, 447 | "requires": { 448 | "once": "^1.3.0", 449 | "wrappy": "1" 450 | } 451 | }, 452 | "inherits": { 453 | "version": "2.0.4", 454 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 455 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 456 | "dev": true 457 | }, 458 | "is-arguments": { 459 | "version": "1.0.4", 460 | "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.0.4.tgz", 461 | "integrity": "sha512-xPh0Rmt8NE65sNzvyUmWgI1tz3mKq74lGA0mL8LYZcoIzKOzDh6HmrYm3d18k60nHerC8A9Km8kYu87zfSFnLA==", 462 | "dev": true 463 | }, 464 | "is-binary-path": { 465 | "version": "2.1.0", 466 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 467 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 468 | "dev": true, 469 | "requires": { 470 | "binary-extensions": "^2.0.0" 471 | } 472 | }, 473 | "is-buffer": { 474 | "version": "2.0.4", 475 | "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.4.tgz", 476 | "integrity": "sha512-Kq1rokWXOPXWuaMAqZiJW4XxsmD9zGx9q4aePabbn3qCRGedtH7Cm+zV8WETitMfu1wdh+Rvd6w5egwSngUX2A==", 477 | "dev": true 478 | }, 479 | "is-callable": { 480 | "version": "1.2.0", 481 | "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.0.tgz", 482 | "integrity": "sha512-pyVD9AaGLxtg6srb2Ng6ynWJqkHU9bEM087AKck0w8QwDarTfNcpIYoU8x8Hv2Icm8u6kFJM18Dag8lyqGkviw==", 483 | "dev": true 484 | }, 485 | "is-date-object": { 486 | "version": "1.0.2", 487 | "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", 488 | "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==", 489 | "dev": true 490 | }, 491 | "is-extglob": { 492 | "version": "2.1.1", 493 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 494 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 495 | "dev": true 496 | }, 497 | "is-fullwidth-code-point": { 498 | "version": "2.0.0", 499 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 500 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 501 | "dev": true 502 | }, 503 | "is-glob": { 504 | "version": "4.0.1", 505 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 506 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 507 | "dev": true, 508 | "requires": { 509 | "is-extglob": "^2.1.1" 510 | } 511 | }, 512 | "is-map": { 513 | "version": "2.0.1", 514 | "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.1.tgz", 515 | "integrity": "sha512-T/S49scO8plUiAOA2DBTBG3JHpn1yiw0kRp6dgiZ0v2/6twi5eiB0rHtHFH9ZIrvlWc6+4O+m4zg5+Z833aXgw==", 516 | "dev": true 517 | }, 518 | "is-number": { 519 | "version": "7.0.0", 520 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 521 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 522 | "dev": true 523 | }, 524 | "is-regex": { 525 | "version": "1.1.0", 526 | "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.0.tgz", 527 | "integrity": "sha512-iI97M8KTWID2la5uYXlkbSDQIg4F6o1sYboZKKTDpnDQMLtUL86zxhgDet3Q2SriaYsyGqZ6Mn2SjbRKeLHdqw==", 528 | "dev": true, 529 | "requires": { 530 | "has-symbols": "^1.0.1" 531 | } 532 | }, 533 | "is-set": { 534 | "version": "2.0.1", 535 | "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.1.tgz", 536 | "integrity": "sha512-eJEzOtVyenDs1TMzSQ3kU3K+E0GUS9sno+F0OBT97xsgcJsF9nXMBtkT9/kut5JEpM7oL7X/0qxR17K3mcwIAA==", 537 | "dev": true 538 | }, 539 | "is-string": { 540 | "version": "1.0.5", 541 | "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.5.tgz", 542 | "integrity": "sha512-buY6VNRjhQMiF1qWDouloZlQbRhDPCebwxSjxMjxgemYT46YMd2NR0/H+fBhEfWX4A/w9TBJ+ol+okqJKFE6vQ==", 543 | "dev": true 544 | }, 545 | "is-symbol": { 546 | "version": "1.0.3", 547 | "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", 548 | "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", 549 | "dev": true, 550 | "requires": { 551 | "has-symbols": "^1.0.1" 552 | } 553 | }, 554 | "isarray": { 555 | "version": "2.0.5", 556 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", 557 | "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", 558 | "dev": true 559 | }, 560 | "isexe": { 561 | "version": "2.0.0", 562 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 563 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 564 | "dev": true 565 | }, 566 | "iterate-iterator": { 567 | "version": "1.0.1", 568 | "resolved": "https://registry.npmjs.org/iterate-iterator/-/iterate-iterator-1.0.1.tgz", 569 | "integrity": "sha512-3Q6tudGN05kbkDQDI4CqjaBf4qf85w6W6GnuZDtUVYwKgtC1q8yxYX7CZed7N+tLzQqS6roujWvszf13T+n9aw==", 570 | "dev": true 571 | }, 572 | "iterate-value": { 573 | "version": "1.0.2", 574 | "resolved": "https://registry.npmjs.org/iterate-value/-/iterate-value-1.0.2.tgz", 575 | "integrity": "sha512-A6fMAio4D2ot2r/TYzr4yUWrmwNdsN5xL7+HUiyACE4DXm+q8HtPcnFTp+NnW3k4N05tZ7FVYFFb2CR13NxyHQ==", 576 | "dev": true, 577 | "requires": { 578 | "es-get-iterator": "^1.0.2", 579 | "iterate-iterator": "^1.0.1" 580 | } 581 | }, 582 | "js-yaml": { 583 | "version": "3.13.1", 584 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", 585 | "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", 586 | "dev": true, 587 | "requires": { 588 | "argparse": "^1.0.7", 589 | "esprima": "^4.0.0" 590 | } 591 | }, 592 | "locate-path": { 593 | "version": "5.0.0", 594 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", 595 | "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", 596 | "dev": true, 597 | "requires": { 598 | "p-locate": "^4.1.0" 599 | } 600 | }, 601 | "lodash": { 602 | "version": "4.17.19", 603 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.19.tgz", 604 | "integrity": "sha512-JNvd8XER9GQX0v2qJgsaN/mzFCNA5BRe/j8JN9d+tWyGLSodKQHKFicdwNYzWwI3wjRnaKPsGj1XkBjx/F96DQ==", 605 | "dev": true 606 | }, 607 | "log-symbols": { 608 | "version": "3.0.0", 609 | "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-3.0.0.tgz", 610 | "integrity": "sha512-dSkNGuI7iG3mfvDzUuYZyvk5dD9ocYCYzNU6CYDE6+Xqd+gwme6Z00NS3dUh8mq/73HaEtT7m6W+yUPtU6BZnQ==", 611 | "dev": true, 612 | "requires": { 613 | "chalk": "^2.4.2" 614 | } 615 | }, 616 | "mime-db": { 617 | "version": "1.42.0", 618 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.42.0.tgz", 619 | "integrity": "sha512-UbfJCR4UAVRNgMpfImz05smAXK7+c+ZntjaA26ANtkXLlOe947Aag5zdIcKQULAiF9Cq4WxBi9jUs5zkA84bYQ==" 620 | }, 621 | "mime-types": { 622 | "version": "2.1.25", 623 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.25.tgz", 624 | "integrity": "sha512-5KhStqB5xpTAeGqKBAMgwaYMnQik7teQN4IAzC7npDv6kzeU6prfkR67bc87J1kWMPGkoaZSq1npmexMgkmEVg==", 625 | "requires": { 626 | "mime-db": "1.42.0" 627 | } 628 | }, 629 | "minimatch": { 630 | "version": "3.0.4", 631 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 632 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 633 | "dev": true, 634 | "requires": { 635 | "brace-expansion": "^1.1.7" 636 | } 637 | }, 638 | "mocha": { 639 | "version": "8.0.1", 640 | "resolved": "https://registry.npmjs.org/mocha/-/mocha-8.0.1.tgz", 641 | "integrity": "sha512-vefaXfdYI8+Yo8nPZQQi0QO2o+5q9UIMX1jZ1XMmK3+4+CQjc7+B0hPdUeglXiTlr8IHMVRo63IhO9Mzt6fxOg==", 642 | "dev": true, 643 | "requires": { 644 | "ansi-colors": "4.1.1", 645 | "browser-stdout": "1.3.1", 646 | "chokidar": "3.3.1", 647 | "debug": "3.2.6", 648 | "diff": "4.0.2", 649 | "escape-string-regexp": "1.0.5", 650 | "find-up": "4.1.0", 651 | "glob": "7.1.6", 652 | "growl": "1.10.5", 653 | "he": "1.2.0", 654 | "js-yaml": "3.13.1", 655 | "log-symbols": "3.0.0", 656 | "minimatch": "3.0.4", 657 | "ms": "2.1.2", 658 | "object.assign": "4.1.0", 659 | "promise.allsettled": "1.0.2", 660 | "serialize-javascript": "3.0.0", 661 | "strip-json-comments": "3.0.1", 662 | "supports-color": "7.1.0", 663 | "which": "2.0.2", 664 | "wide-align": "1.1.3", 665 | "workerpool": "6.0.0", 666 | "yargs": "13.3.2", 667 | "yargs-parser": "13.1.2", 668 | "yargs-unparser": "1.6.0" 669 | } 670 | }, 671 | "ms": { 672 | "version": "2.1.2", 673 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 674 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 675 | "dev": true 676 | }, 677 | "node-fetch": { 678 | "version": "2.6.0", 679 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.0.tgz", 680 | "integrity": "sha512-8dG4H5ujfvFiqDmVu9fQ5bOHUC15JMjMY/Zumv26oOvvVJjM67KF8koCWIabKQ1GJIa9r2mMZscBq/TbdOcmNA==" 681 | }, 682 | "normalize-path": { 683 | "version": "3.0.0", 684 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 685 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 686 | "dev": true 687 | }, 688 | "object-inspect": { 689 | "version": "1.8.0", 690 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.8.0.tgz", 691 | "integrity": "sha512-jLdtEOB112fORuypAyl/50VRVIBIdVQOSUUGQHzJ4xBSbit81zRarz7GThkEFZy1RceYrWYcPcBFPQwHyAc1gA==", 692 | "dev": true 693 | }, 694 | "object-keys": { 695 | "version": "1.1.1", 696 | "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", 697 | "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", 698 | "dev": true 699 | }, 700 | "object.assign": { 701 | "version": "4.1.0", 702 | "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.0.tgz", 703 | "integrity": "sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==", 704 | "dev": true, 705 | "requires": { 706 | "define-properties": "^1.1.2", 707 | "function-bind": "^1.1.1", 708 | "has-symbols": "^1.0.0", 709 | "object-keys": "^1.0.11" 710 | } 711 | }, 712 | "once": { 713 | "version": "1.4.0", 714 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 715 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 716 | "dev": true, 717 | "requires": { 718 | "wrappy": "1" 719 | } 720 | }, 721 | "p-limit": { 722 | "version": "2.3.0", 723 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", 724 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", 725 | "dev": true, 726 | "requires": { 727 | "p-try": "^2.0.0" 728 | } 729 | }, 730 | "p-locate": { 731 | "version": "4.1.0", 732 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", 733 | "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", 734 | "dev": true, 735 | "requires": { 736 | "p-limit": "^2.2.0" 737 | } 738 | }, 739 | "p-try": { 740 | "version": "2.2.0", 741 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", 742 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", 743 | "dev": true 744 | }, 745 | "path-exists": { 746 | "version": "4.0.0", 747 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 748 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 749 | "dev": true 750 | }, 751 | "path-is-absolute": { 752 | "version": "1.0.1", 753 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 754 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 755 | "dev": true 756 | }, 757 | "picomatch": { 758 | "version": "2.2.2", 759 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", 760 | "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", 761 | "dev": true 762 | }, 763 | "promise.allsettled": { 764 | "version": "1.0.2", 765 | "resolved": "https://registry.npmjs.org/promise.allsettled/-/promise.allsettled-1.0.2.tgz", 766 | "integrity": "sha512-UpcYW5S1RaNKT6pd+s9jp9K9rlQge1UXKskec0j6Mmuq7UJCvlS2J2/s/yuPN8ehftf9HXMxWlKiPbGGUzpoRg==", 767 | "dev": true, 768 | "requires": { 769 | "array.prototype.map": "^1.0.1", 770 | "define-properties": "^1.1.3", 771 | "es-abstract": "^1.17.0-next.1", 772 | "function-bind": "^1.1.1", 773 | "iterate-value": "^1.0.0" 774 | } 775 | }, 776 | "readdirp": { 777 | "version": "3.3.0", 778 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.3.0.tgz", 779 | "integrity": "sha512-zz0pAkSPOXXm1viEwygWIPSPkcBYjW1xU5j/JBh5t9bGCJwa6f9+BJa6VaB2g+b55yVrmXzqkyLf4xaWYM0IkQ==", 780 | "dev": true, 781 | "requires": { 782 | "picomatch": "^2.0.7" 783 | } 784 | }, 785 | "require-directory": { 786 | "version": "2.1.1", 787 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 788 | "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", 789 | "dev": true 790 | }, 791 | "require-main-filename": { 792 | "version": "2.0.0", 793 | "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", 794 | "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", 795 | "dev": true 796 | }, 797 | "serialize-javascript": { 798 | "version": "3.0.0", 799 | "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-3.0.0.tgz", 800 | "integrity": "sha512-skZcHYw2vEX4bw90nAr2iTTsz6x2SrHEnfxgKYmZlvJYBEZrvbKtobJWlQ20zczKb3bsHHXXTYt48zBA7ni9cw==", 801 | "dev": true 802 | }, 803 | "set-blocking": { 804 | "version": "2.0.0", 805 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", 806 | "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", 807 | "dev": true 808 | }, 809 | "sprintf-js": { 810 | "version": "1.0.3", 811 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 812 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", 813 | "dev": true 814 | }, 815 | "string-width": { 816 | "version": "2.1.1", 817 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", 818 | "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", 819 | "dev": true, 820 | "requires": { 821 | "is-fullwidth-code-point": "^2.0.0", 822 | "strip-ansi": "^4.0.0" 823 | } 824 | }, 825 | "string.prototype.trimend": { 826 | "version": "1.0.1", 827 | "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.1.tgz", 828 | "integrity": "sha512-LRPxFUaTtpqYsTeNKaFOw3R4bxIzWOnbQ837QfBylo8jIxtcbK/A/sMV7Q+OAV/vWo+7s25pOE10KYSjaSO06g==", 829 | "dev": true, 830 | "requires": { 831 | "define-properties": "^1.1.3", 832 | "es-abstract": "^1.17.5" 833 | } 834 | }, 835 | "string.prototype.trimstart": { 836 | "version": "1.0.1", 837 | "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.1.tgz", 838 | "integrity": "sha512-XxZn+QpvrBI1FOcg6dIpxUPgWCPuNXvMD72aaRaUQv1eD4e/Qy8i/hFTe0BUmD60p/QA6bh1avmuPTfNjqVWRw==", 839 | "dev": true, 840 | "requires": { 841 | "define-properties": "^1.1.3", 842 | "es-abstract": "^1.17.5" 843 | } 844 | }, 845 | "strip-ansi": { 846 | "version": "4.0.0", 847 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", 848 | "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", 849 | "dev": true, 850 | "requires": { 851 | "ansi-regex": "^3.0.0" 852 | } 853 | }, 854 | "strip-json-comments": { 855 | "version": "3.0.1", 856 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.0.1.tgz", 857 | "integrity": "sha512-VTyMAUfdm047mwKl+u79WIdrZxtFtn+nBxHeb844XBQ9uMNTuTHdx2hc5RiAJYqwTj3wc/xe5HLSdJSkJ+WfZw==", 858 | "dev": true 859 | }, 860 | "supports-color": { 861 | "version": "7.1.0", 862 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz", 863 | "integrity": "sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==", 864 | "dev": true, 865 | "requires": { 866 | "has-flag": "^4.0.0" 867 | }, 868 | "dependencies": { 869 | "has-flag": { 870 | "version": "4.0.0", 871 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 872 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 873 | "dev": true 874 | } 875 | } 876 | }, 877 | "to-regex-range": { 878 | "version": "5.0.1", 879 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 880 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 881 | "dev": true, 882 | "requires": { 883 | "is-number": "^7.0.0" 884 | } 885 | }, 886 | "which": { 887 | "version": "2.0.2", 888 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 889 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 890 | "dev": true, 891 | "requires": { 892 | "isexe": "^2.0.0" 893 | } 894 | }, 895 | "which-module": { 896 | "version": "2.0.0", 897 | "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", 898 | "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", 899 | "dev": true 900 | }, 901 | "wide-align": { 902 | "version": "1.1.3", 903 | "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", 904 | "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", 905 | "dev": true, 906 | "requires": { 907 | "string-width": "^1.0.2 || 2" 908 | } 909 | }, 910 | "workerpool": { 911 | "version": "6.0.0", 912 | "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.0.0.tgz", 913 | "integrity": "sha512-fU2OcNA/GVAJLLyKUoHkAgIhKb0JoCpSjLC/G2vYKxUjVmQwGbRVeoPJ1a8U4pnVofz4AQV5Y/NEw8oKqxEBtA==", 914 | "dev": true 915 | }, 916 | "wrap-ansi": { 917 | "version": "5.1.0", 918 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", 919 | "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", 920 | "dev": true, 921 | "requires": { 922 | "ansi-styles": "^3.2.0", 923 | "string-width": "^3.0.0", 924 | "strip-ansi": "^5.0.0" 925 | }, 926 | "dependencies": { 927 | "ansi-regex": { 928 | "version": "4.1.0", 929 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 930 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 931 | "dev": true 932 | }, 933 | "string-width": { 934 | "version": "3.1.0", 935 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 936 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 937 | "dev": true, 938 | "requires": { 939 | "emoji-regex": "^7.0.1", 940 | "is-fullwidth-code-point": "^2.0.0", 941 | "strip-ansi": "^5.1.0" 942 | } 943 | }, 944 | "strip-ansi": { 945 | "version": "5.2.0", 946 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 947 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 948 | "dev": true, 949 | "requires": { 950 | "ansi-regex": "^4.1.0" 951 | } 952 | } 953 | } 954 | }, 955 | "wrappy": { 956 | "version": "1.0.2", 957 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 958 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 959 | "dev": true 960 | }, 961 | "y18n": { 962 | "version": "4.0.0", 963 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz", 964 | "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==", 965 | "dev": true 966 | }, 967 | "yargs": { 968 | "version": "13.3.2", 969 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz", 970 | "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==", 971 | "dev": true, 972 | "requires": { 973 | "cliui": "^5.0.0", 974 | "find-up": "^3.0.0", 975 | "get-caller-file": "^2.0.1", 976 | "require-directory": "^2.1.1", 977 | "require-main-filename": "^2.0.0", 978 | "set-blocking": "^2.0.0", 979 | "string-width": "^3.0.0", 980 | "which-module": "^2.0.0", 981 | "y18n": "^4.0.0", 982 | "yargs-parser": "^13.1.2" 983 | }, 984 | "dependencies": { 985 | "ansi-regex": { 986 | "version": "4.1.0", 987 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 988 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 989 | "dev": true 990 | }, 991 | "find-up": { 992 | "version": "3.0.0", 993 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", 994 | "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", 995 | "dev": true, 996 | "requires": { 997 | "locate-path": "^3.0.0" 998 | } 999 | }, 1000 | "locate-path": { 1001 | "version": "3.0.0", 1002 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", 1003 | "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", 1004 | "dev": true, 1005 | "requires": { 1006 | "p-locate": "^3.0.0", 1007 | "path-exists": "^3.0.0" 1008 | } 1009 | }, 1010 | "p-locate": { 1011 | "version": "3.0.0", 1012 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", 1013 | "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", 1014 | "dev": true, 1015 | "requires": { 1016 | "p-limit": "^2.0.0" 1017 | } 1018 | }, 1019 | "path-exists": { 1020 | "version": "3.0.0", 1021 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", 1022 | "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", 1023 | "dev": true 1024 | }, 1025 | "string-width": { 1026 | "version": "3.1.0", 1027 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 1028 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 1029 | "dev": true, 1030 | "requires": { 1031 | "emoji-regex": "^7.0.1", 1032 | "is-fullwidth-code-point": "^2.0.0", 1033 | "strip-ansi": "^5.1.0" 1034 | } 1035 | }, 1036 | "strip-ansi": { 1037 | "version": "5.2.0", 1038 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 1039 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 1040 | "dev": true, 1041 | "requires": { 1042 | "ansi-regex": "^4.1.0" 1043 | } 1044 | } 1045 | } 1046 | }, 1047 | "yargs-parser": { 1048 | "version": "13.1.2", 1049 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz", 1050 | "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==", 1051 | "dev": true, 1052 | "requires": { 1053 | "camelcase": "^5.0.0", 1054 | "decamelize": "^1.2.0" 1055 | } 1056 | }, 1057 | "yargs-unparser": { 1058 | "version": "1.6.0", 1059 | "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-1.6.0.tgz", 1060 | "integrity": "sha512-W9tKgmSn0DpSatfri0nx52Joq5hVXgeLiqR/5G0sZNDoLZFOr/xjBUDcShCOGNsBnEMNo1KAMBkTej1Hm62HTw==", 1061 | "dev": true, 1062 | "requires": { 1063 | "flat": "^4.1.0", 1064 | "lodash": "^4.17.15", 1065 | "yargs": "^13.3.0" 1066 | } 1067 | } 1068 | } 1069 | } 1070 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "discord-webhook-node", 3 | "version": "1.1.8", 4 | "description": "Allows for easy webhook sending through discord's webhook API", 5 | "main": "./src/index.js", 6 | "exports": { 7 | "import": "./src/index.mjs", 8 | "require": "./src/index.js" 9 | }, 10 | "typings": "index.d.ts", 11 | "scripts": { 12 | "test": "mocha --timeout 50000" 13 | }, 14 | "author": "Matthew Mulfinger", 15 | "license": "MIT", 16 | "keywords": [ 17 | "Webhook", 18 | "Discord", 19 | "Bot" 20 | ], 21 | "homepage": "https://github.com/matthew1232/discord-webhook-node#readme", 22 | "dependencies": { 23 | "form-data": "^3.0.0", 24 | "node-fetch": "^2.6.0" 25 | }, 26 | "devDependencies": { 27 | "dotenv": "^8.2.0", 28 | "mocha": "^8.0.1" 29 | } 30 | } 31 | -------------------------------------------------------------------------------- /src/api/index.js: -------------------------------------------------------------------------------- 1 | const sendFile = require('./sendFile'); 2 | const sendWebhook = require('./sendWebhook'); 3 | 4 | module.exports = { 5 | sendFile, 6 | sendWebhook 7 | }; -------------------------------------------------------------------------------- /src/api/sendFile.js: -------------------------------------------------------------------------------- 1 | const FormData = require('form-data'); 2 | const fs = require('fs'); 3 | 4 | module.exports = (hookURL, file, { username, avatar_url }) => new Promise((resolve, reject) => { 5 | const form = new FormData(); 6 | 7 | if (username){ 8 | form.append('username', username); 9 | }; 10 | 11 | if (avatar_url){ 12 | form.append('avatar_url', avatar_url); 13 | }; 14 | 15 | form.append('file', fs.createReadStream(file)); 16 | 17 | form.submit(hookURL, (error, response) => { 18 | if (error) reject(error); 19 | else resolve(response); 20 | }); 21 | }); 22 | -------------------------------------------------------------------------------- /src/api/sendWebhook.js: -------------------------------------------------------------------------------- 1 | const fetch = require('node-fetch'); 2 | 3 | module.exports = (hookURL, payload) => new Promise((resolve, reject) => { 4 | fetch(hookURL, { 5 | method: 'POST', 6 | headers: { 7 | 'Content-Type': 'application/json' 8 | }, 9 | body: JSON.stringify(payload) 10 | }) 11 | .then(res => resolve(res)) 12 | .catch(err => reject(err)); 13 | }); 14 | -------------------------------------------------------------------------------- /src/classes/messageBuilder.js: -------------------------------------------------------------------------------- 1 | const { formatColor } = require('../utils'); 2 | 3 | module.exports = class MessageBuilder { 4 | constructor(){ 5 | this.payload = { 6 | embeds: [{fields: []}] 7 | }; 8 | }; 9 | 10 | getJSON(){ 11 | return this.payload; 12 | }; 13 | 14 | setText(text){ 15 | this.payload.content = text; 16 | 17 | return this; 18 | } 19 | 20 | setAuthor(author, authorImage, authorUrl){ 21 | this.payload.embeds[0].author = {}; 22 | this.payload.embeds[0].author.name = author; 23 | this.payload.embeds[0].author.url = authorUrl; 24 | this.payload.embeds[0].author.icon_url = authorImage; 25 | 26 | return this; 27 | }; 28 | 29 | setTitle(title){ 30 | this.payload.embeds[0].title = title; 31 | 32 | return this; 33 | }; 34 | 35 | setURL(url){ 36 | this.payload.embeds[0].url = url; 37 | 38 | return this; 39 | }; 40 | 41 | setThumbnail(thumbnail){ 42 | this.payload.embeds[0].thumbnail = {}; 43 | this.payload.embeds[0].thumbnail.url = thumbnail; 44 | 45 | return this; 46 | }; 47 | 48 | setImage(image){ 49 | this.payload.embeds[0].image = {}; 50 | this.payload.embeds[0].image.url = image; 51 | 52 | return this; 53 | }; 54 | 55 | setTimestamp(date){ 56 | if (date){ 57 | this.payload.embeds[0].timestamp = date; 58 | } 59 | else { 60 | this.payload.embeds[0].timestamp = new Date(); 61 | }; 62 | 63 | return this; 64 | }; 65 | 66 | setColor(color){ 67 | this.payload.embeds[0].color = formatColor(color); 68 | 69 | return this; 70 | }; 71 | 72 | setDescription(description){ 73 | this.payload.embeds[0].description = description; 74 | 75 | return this; 76 | }; 77 | 78 | addField(fieldName, fieldValue, inline){ 79 | this.payload.embeds[0].fields.push({ 80 | name: fieldName, 81 | value: fieldValue, 82 | inline: inline 83 | }); 84 | 85 | return this; 86 | }; 87 | 88 | setFooter(footer, footerImage){ 89 | this.payload.embeds[0].footer = {}; 90 | this.payload.embeds[0].footer.icon_url = footerImage; 91 | this.payload.embeds[0].footer.text = footer; 92 | 93 | return this; 94 | }; 95 | }; -------------------------------------------------------------------------------- /src/classes/webhook.js: -------------------------------------------------------------------------------- 1 | const { sendWebhook, sendFile } = require('../api'); 2 | const MessageBuilder = require('./messageBuilder'); 3 | 4 | module.exports = class Webhook { 5 | constructor(options){ 6 | this.payload = {}; 7 | 8 | if (typeof options == 'string'){ 9 | this.hookURL = options; 10 | this.throwErrors = true; 11 | this.retryOnLimit = true; 12 | } 13 | else { 14 | this.hookURL = options.url; 15 | this.throwErrors = options.throwErrors == undefined ? true : options.throwErrors; 16 | this.retryOnLimit = options.retryOnLimit == undefined ? true : options.retryOnLimit; 17 | }; 18 | }; 19 | 20 | setUsername(username){ 21 | this.payload.username = username; 22 | 23 | return this; 24 | } 25 | 26 | setAvatar(avatarURL){ 27 | this.payload.avatar_url = avatarURL; 28 | 29 | return this; 30 | } 31 | 32 | async sendFile(filePath){ 33 | try { 34 | const res = await sendFile(this.hookURL, filePath, this.payload); 35 | 36 | if (res.statusCode != 200){ 37 | throw new Error(`Error sending webhook: ${res.statusCode} status code.`); 38 | }; 39 | } 40 | catch(err){ 41 | if (this.throwErrors) throw new Error(err.message); 42 | }; 43 | } 44 | 45 | async send(payload){ 46 | let endPayload = { 47 | ...this.payload 48 | }; 49 | 50 | if (typeof payload === 'string'){ 51 | endPayload.content = payload; 52 | } 53 | else { 54 | endPayload = { 55 | ...endPayload, 56 | ...payload.getJSON() 57 | }; 58 | }; 59 | 60 | try { 61 | const res = await sendWebhook(this.hookURL, endPayload); 62 | 63 | if (res.status == 429 && this.retryOnLimit){ 64 | const body = await res.json(); 65 | const waitUntil = body["retry_after"]; 66 | 67 | setTimeout(() => sendWebhook(this.hookURL, endPayload), waitUntil); 68 | } 69 | else if (res.status != 204){ 70 | throw new Error(`Error sending webhook: ${res.status} status code. Response: ${await res.text()}`); 71 | }; 72 | } 73 | catch(err){ 74 | if (this.throwErrors) throw new Error(err.message); 75 | }; 76 | }; 77 | 78 | async info(title, fieldName, fieldValue, inline){ 79 | const embed = new MessageBuilder() 80 | .setTitle(title) 81 | .setTimestamp() 82 | .setColor(4037805); 83 | 84 | if (fieldName != undefined && fieldValue != undefined){ 85 | embed.addField(fieldName, fieldValue, inline) 86 | }; 87 | 88 | await this.send(embed); 89 | }; 90 | 91 | async success(title, fieldName, fieldValue, inline){ 92 | const embed = new MessageBuilder() 93 | .setTitle(title) 94 | .setTimestamp() 95 | .setColor(65340); 96 | 97 | if (fieldName != undefined && fieldValue != undefined){ 98 | embed.addField(fieldName, fieldValue, inline) 99 | }; 100 | 101 | await this.send(embed); 102 | } 103 | 104 | async warning(title, fieldName, fieldValue, inline){ 105 | const embed = new MessageBuilder() 106 | .setTitle(title) 107 | .setTimestamp() 108 | .setColor(16763904); 109 | 110 | if (fieldName != undefined && fieldValue != undefined){ 111 | embed.addField(fieldName, fieldValue, inline) 112 | }; 113 | 114 | await this.send(embed); 115 | } 116 | 117 | 118 | async error(title, fieldName, fieldValue, inline){ 119 | const embed = new MessageBuilder() 120 | .setTitle(title) 121 | .setTimestamp() 122 | .setColor(16729149); 123 | 124 | if (fieldName != undefined && fieldValue != undefined){ 125 | embed.addField(fieldName, fieldValue, inline) 126 | }; 127 | 128 | await this.send(embed); 129 | } 130 | }; 131 | -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | const Webhook = require('./classes/webhook'); 2 | const MessageBuilder = require('./classes/messageBuilder'); 3 | 4 | module.exports = { 5 | Webhook, 6 | MessageBuilder 7 | }; 8 | -------------------------------------------------------------------------------- /src/index.mjs: -------------------------------------------------------------------------------- 1 | import discord_webhook from "./index.js"; 2 | 3 | const Webhook = discord_webhook.Webhook; 4 | const MessageBuilder = discord_webhook.MessageBuilder; 5 | 6 | export default discord_webhook; 7 | export {Webhook, MessageBuilder}; 8 | -------------------------------------------------------------------------------- /src/utils/index.js: -------------------------------------------------------------------------------- 1 | exports.formatColor = (color) => { 2 | if (typeof color === 'string' && color.startsWith("#")){ 3 | const rawHex = color.split('#')[1]; 4 | 5 | return parseInt(rawHex, 16); 6 | } 7 | else { 8 | return Number(color); 9 | }; 10 | }; -------------------------------------------------------------------------------- /test/test.js: -------------------------------------------------------------------------------- 1 | require('dotenv').config(); 2 | const fs = require('fs').promises; 3 | const { Webhook, MessageBuilder } = require('../src'); 4 | 5 | const { WEBHOOK_URL } = process.env; 6 | 7 | const hook = new Webhook(WEBHOOK_URL); 8 | 9 | const IMAGE_URL = 'https://homepages.cae.wisc.edu/~ece533/images/airplane.png'; 10 | 11 | hook.setUsername('Discord Webhook Node Name'); 12 | hook.setAvatar(IMAGE_URL); 13 | 14 | describe('Custom hooks', function(){ 15 | it('Sends embed', function(done){ 16 | const embed = new MessageBuilder(); 17 | 18 | embed.setText('Text') 19 | .setAuthor('Discord Webhook Node Author', IMAGE_URL, 'https://npmjs.org/package/discord-webhook-node') 20 | .setTitle('Title') 21 | .setURL('https://npmjs.org/package/discord-webhook-node') 22 | .setImage(IMAGE_URL) 23 | .setThumbnail(IMAGE_URL) 24 | .setColor('#00b0f4') 25 | .addField('Field #1', 'Not inline') 26 | .setDescription('Description') 27 | .setFooter('Footer', IMAGE_URL) 28 | .setTimestamp(); 29 | 30 | hook.send(embed).then(() => { 31 | done(); 32 | }) 33 | .catch(err => done(err)); 34 | }); 35 | 36 | it('Sends file', function(done){ 37 | fs.writeFile('./test/customfile.txt', 'Message from discord-webhook-node').then(() => { 38 | hook.sendFile('./test/customfile.txt').then(() => { 39 | fs.unlink('./test/customfile.txt'); 40 | 41 | done(); 42 | }) 43 | .catch(err => done(err)); 44 | }) 45 | .catch(err => done(err)); 46 | }); 47 | 48 | it('Sends text to webhook', function(done){ 49 | hook.send('Plain text').then(() => { 50 | done(); 51 | }) 52 | .catch(err => done(err)); 53 | }); 54 | }); 55 | 56 | describe('Preset hooks', function(){ 57 | it('info', function(done){ 58 | hook.info('Info', 'Field name', 'Field value').then(() => { 59 | done(); 60 | }) 61 | .catch(err => done(err)); 62 | }); 63 | 64 | it('success', function(done){ 65 | hook.success('Success', 'Field name', 'Field value').then(() => { 66 | done(); 67 | }) 68 | .catch(err => done(err)); 69 | }); 70 | 71 | it('warning', function(done){ 72 | hook.warning('Warning', 'Field name', 'Field value').then(() => { 73 | done(); 74 | }) 75 | .catch(err => done(err)); 76 | }); 77 | 78 | it('error', function(done){ 79 | hook.error('error', 'Field name', 'Field value').then(() => { 80 | done(); 81 | }) 82 | .catch(err => done(err)); 83 | }); 84 | }); --------------------------------------------------------------------------------