├── maxcoin ├── .eslintrc.json ├── .gitignore ├── main.js ├── package-lock.json ├── package.json └── services │ ├── CoinAPI.js │ └── backend │ ├── MongoBackend.js │ ├── MySQLBackend.js │ └── RedisBackend.js └── shopper ├── .eslintrc.json ├── .gitignore ├── client └── css │ └── site.css ├── package-lock.json ├── package.json └── server ├── app.js ├── bin └── start.js ├── config └── index.js ├── models └── mongoose │ ├── Item.js │ └── User.js ├── routes ├── admin │ ├── item │ │ └── index.js │ ├── orders │ │ └── index.js │ └── user │ │ └── index.js ├── basket │ └── index.js ├── index.js └── shop │ └── index.js ├── services ├── BasketService.js ├── ItemService.js ├── OrderService.js └── UserService.js └── views ├── admin ├── item.pug ├── orders.pug └── user.pug ├── basket.pug ├── index.pug ├── layout.pug └── shop.pug /maxcoin/.eslintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "env": { 3 | "commonjs": true, 4 | "es6": true, 5 | "node": true 6 | }, 7 | "extends": ["airbnb-base", "prettier"], 8 | "plugins": ["prettier"], 9 | "globals": { 10 | "Atomics": "readonly", 11 | "SharedArrayBuffer": "readonly" 12 | }, 13 | "parserOptions": { 14 | "ecmaVersion": 2018 15 | }, 16 | "rules": { 17 | "no-console": 0 18 | } 19 | } 20 | -------------------------------------------------------------------------------- /maxcoin/.gitignore: -------------------------------------------------------------------------------- 1 | /node_modules 2 | .DS_STORE -------------------------------------------------------------------------------- /maxcoin/main.js: -------------------------------------------------------------------------------- 1 | const CoinAPI = require("./services/CoinAPI"); 2 | 3 | async function run() { 4 | const coinAPI = new CoinAPI(); 5 | return coinAPI.fetch(); 6 | } 7 | 8 | run() 9 | .then((result) => { 10 | console.log(result); 11 | }) 12 | .catch((err) => console.error(err)); 13 | -------------------------------------------------------------------------------- /maxcoin/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "maxcoin", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@babel/code-frame": { 8 | "version": "7.10.4", 9 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.10.4.tgz", 10 | "integrity": "sha512-vG6SvB6oYEhvgisZNFRmRCUkLz11c7rp+tbNTynGqc6mS1d5ATd/sGyV6W0KZZnXRKMTzZDRgQT3Ou9jhpAfUg==", 11 | "dev": true, 12 | "requires": { 13 | "@babel/highlight": "^7.10.4" 14 | } 15 | }, 16 | "@babel/helper-validator-identifier": { 17 | "version": "7.10.4", 18 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.10.4.tgz", 19 | "integrity": "sha512-3U9y+43hz7ZM+rzG24Qe2mufW5KhvFg/NhnNph+i9mgCtdTCtMJuI1TMkrIUiK7Ix4PYlRF9I5dhqaLYA/ADXw==", 20 | "dev": true 21 | }, 22 | "@babel/highlight": { 23 | "version": "7.10.4", 24 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz", 25 | "integrity": "sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==", 26 | "dev": true, 27 | "requires": { 28 | "@babel/helper-validator-identifier": "^7.10.4", 29 | "chalk": "^2.0.0", 30 | "js-tokens": "^4.0.0" 31 | }, 32 | "dependencies": { 33 | "chalk": { 34 | "version": "2.4.2", 35 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 36 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 37 | "dev": true, 38 | "requires": { 39 | "ansi-styles": "^3.2.1", 40 | "escape-string-regexp": "^1.0.5", 41 | "supports-color": "^5.3.0" 42 | } 43 | } 44 | } 45 | }, 46 | "@eslint/eslintrc": { 47 | "version": "0.1.3", 48 | "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-0.1.3.tgz", 49 | "integrity": "sha512-4YVwPkANLeNtRjMekzux1ci8hIaH5eGKktGqR0d3LWsKNn5B2X/1Z6Trxy7jQXl9EBGE6Yj02O+t09FMeRllaA==", 50 | "dev": true, 51 | "requires": { 52 | "ajv": "^6.12.4", 53 | "debug": "^4.1.1", 54 | "espree": "^7.3.0", 55 | "globals": "^12.1.0", 56 | "ignore": "^4.0.6", 57 | "import-fresh": "^3.2.1", 58 | "js-yaml": "^3.13.1", 59 | "lodash": "^4.17.19", 60 | "minimatch": "^3.0.4", 61 | "strip-json-comments": "^3.1.1" 62 | } 63 | }, 64 | "@types/color-name": { 65 | "version": "1.1.1", 66 | "resolved": "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz", 67 | "integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==", 68 | "dev": true 69 | }, 70 | "@types/json5": { 71 | "version": "0.0.29", 72 | "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", 73 | "integrity": "sha1-7ihweulOEdK4J7y+UnC86n8+ce4=", 74 | "dev": true 75 | }, 76 | "acorn": { 77 | "version": "7.4.0", 78 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.0.tgz", 79 | "integrity": "sha512-+G7P8jJmCHr+S+cLfQxygbWhXy+8YTVGzAkpEbcLo2mLoL7tij/VG41QSHACSf5QgYRhMZYHuNc6drJaO0Da+w==", 80 | "dev": true 81 | }, 82 | "acorn-jsx": { 83 | "version": "5.3.1", 84 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.1.tgz", 85 | "integrity": "sha512-K0Ptm/47OKfQRpNQ2J/oIN/3QYiK6FwW+eJbILhsdxh2WTLdl+30o8aGdTbm5JbffpFFAg/g+zi1E+jvJha5ng==", 86 | "dev": true 87 | }, 88 | "ajv": { 89 | "version": "6.12.5", 90 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.5.tgz", 91 | "integrity": "sha512-lRF8RORchjpKG50/WFf8xmg7sgCLFiYNNnqdKflk63whMQcWR5ngGjiSXkL9bjxy6B2npOK2HSMN49jEBMSkag==", 92 | "dev": true, 93 | "requires": { 94 | "fast-deep-equal": "^3.1.1", 95 | "fast-json-stable-stringify": "^2.0.0", 96 | "json-schema-traverse": "^0.4.1", 97 | "uri-js": "^4.2.2" 98 | } 99 | }, 100 | "ansi-colors": { 101 | "version": "4.1.1", 102 | "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", 103 | "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", 104 | "dev": true 105 | }, 106 | "ansi-regex": { 107 | "version": "5.0.0", 108 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 109 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 110 | "dev": true 111 | }, 112 | "ansi-styles": { 113 | "version": "3.2.1", 114 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 115 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 116 | "dev": true, 117 | "requires": { 118 | "color-convert": "^1.9.0" 119 | } 120 | }, 121 | "argparse": { 122 | "version": "1.0.10", 123 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 124 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 125 | "dev": true, 126 | "requires": { 127 | "sprintf-js": "~1.0.2" 128 | } 129 | }, 130 | "array-includes": { 131 | "version": "3.1.1", 132 | "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.1.tgz", 133 | "integrity": "sha512-c2VXaCHl7zPsvpkFsw4nxvFie4fh1ur9bpcgsVkIjqn0H/Xwdg+7fv3n2r/isyS8EBj5b06M9kHyZuIr4El6WQ==", 134 | "dev": true, 135 | "requires": { 136 | "define-properties": "^1.1.3", 137 | "es-abstract": "^1.17.0", 138 | "is-string": "^1.0.5" 139 | }, 140 | "dependencies": { 141 | "es-abstract": { 142 | "version": "1.17.7", 143 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", 144 | "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", 145 | "dev": true, 146 | "requires": { 147 | "es-to-primitive": "^1.2.1", 148 | "function-bind": "^1.1.1", 149 | "has": "^1.0.3", 150 | "has-symbols": "^1.0.1", 151 | "is-callable": "^1.2.2", 152 | "is-regex": "^1.1.1", 153 | "object-inspect": "^1.8.0", 154 | "object-keys": "^1.1.1", 155 | "object.assign": "^4.1.1", 156 | "string.prototype.trimend": "^1.0.1", 157 | "string.prototype.trimstart": "^1.0.1" 158 | } 159 | } 160 | } 161 | }, 162 | "array.prototype.flat": { 163 | "version": "1.2.3", 164 | "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.2.3.tgz", 165 | "integrity": "sha512-gBlRZV0VSmfPIeWfuuy56XZMvbVfbEUnOXUvt3F/eUUUSyzlgLxhEX4YAEpxNAogRGehPSnfXyPtYyKAhkzQhQ==", 166 | "dev": true, 167 | "requires": { 168 | "define-properties": "^1.1.3", 169 | "es-abstract": "^1.17.0-next.1" 170 | }, 171 | "dependencies": { 172 | "es-abstract": { 173 | "version": "1.17.7", 174 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", 175 | "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", 176 | "dev": true, 177 | "requires": { 178 | "es-to-primitive": "^1.2.1", 179 | "function-bind": "^1.1.1", 180 | "has": "^1.0.3", 181 | "has-symbols": "^1.0.1", 182 | "is-callable": "^1.2.2", 183 | "is-regex": "^1.1.1", 184 | "object-inspect": "^1.8.0", 185 | "object-keys": "^1.1.1", 186 | "object.assign": "^4.1.1", 187 | "string.prototype.trimend": "^1.0.1", 188 | "string.prototype.trimstart": "^1.0.1" 189 | } 190 | } 191 | } 192 | }, 193 | "astral-regex": { 194 | "version": "1.0.0", 195 | "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", 196 | "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", 197 | "dev": true 198 | }, 199 | "axios": { 200 | "version": "0.20.0", 201 | "resolved": "https://registry.npmjs.org/axios/-/axios-0.20.0.tgz", 202 | "integrity": "sha512-ANA4rr2BDcmmAQLOKft2fufrtuvlqR+cXNNinUmvfeSNCOF98PZL+7M/v1zIdGo7OLjEA9J2gXJL+j4zGsl0bA==", 203 | "requires": { 204 | "follow-redirects": "^1.10.0" 205 | } 206 | }, 207 | "balanced-match": { 208 | "version": "1.0.0", 209 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 210 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 211 | "dev": true 212 | }, 213 | "brace-expansion": { 214 | "version": "1.1.11", 215 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 216 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 217 | "dev": true, 218 | "requires": { 219 | "balanced-match": "^1.0.0", 220 | "concat-map": "0.0.1" 221 | } 222 | }, 223 | "callsites": { 224 | "version": "3.1.0", 225 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", 226 | "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", 227 | "dev": true 228 | }, 229 | "chalk": { 230 | "version": "4.1.0", 231 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", 232 | "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", 233 | "dev": true, 234 | "requires": { 235 | "ansi-styles": "^4.1.0", 236 | "supports-color": "^7.1.0" 237 | }, 238 | "dependencies": { 239 | "ansi-styles": { 240 | "version": "4.2.1", 241 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz", 242 | "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==", 243 | "dev": true, 244 | "requires": { 245 | "@types/color-name": "^1.1.1", 246 | "color-convert": "^2.0.1" 247 | } 248 | }, 249 | "color-convert": { 250 | "version": "2.0.1", 251 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 252 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 253 | "dev": true, 254 | "requires": { 255 | "color-name": "~1.1.4" 256 | } 257 | }, 258 | "color-name": { 259 | "version": "1.1.4", 260 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 261 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 262 | "dev": true 263 | }, 264 | "has-flag": { 265 | "version": "4.0.0", 266 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 267 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 268 | "dev": true 269 | }, 270 | "supports-color": { 271 | "version": "7.2.0", 272 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 273 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 274 | "dev": true, 275 | "requires": { 276 | "has-flag": "^4.0.0" 277 | } 278 | } 279 | } 280 | }, 281 | "color-convert": { 282 | "version": "1.9.3", 283 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 284 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 285 | "dev": true, 286 | "requires": { 287 | "color-name": "1.1.3" 288 | } 289 | }, 290 | "color-name": { 291 | "version": "1.1.3", 292 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 293 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 294 | "dev": true 295 | }, 296 | "concat-map": { 297 | "version": "0.0.1", 298 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 299 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 300 | "dev": true 301 | }, 302 | "confusing-browser-globals": { 303 | "version": "1.0.9", 304 | "resolved": "https://registry.npmjs.org/confusing-browser-globals/-/confusing-browser-globals-1.0.9.tgz", 305 | "integrity": "sha512-KbS1Y0jMtyPgIxjO7ZzMAuUpAKMt1SzCL9fsrKsX6b0zJPTaT0SiSPmewwVZg9UAO83HVIlEhZF84LIjZ0lmAw==", 306 | "dev": true 307 | }, 308 | "contains-path": { 309 | "version": "0.1.0", 310 | "resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz", 311 | "integrity": "sha1-/ozxhP9mcLa67wGp1IYaXL7EEgo=", 312 | "dev": true 313 | }, 314 | "cross-spawn": { 315 | "version": "7.0.3", 316 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", 317 | "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", 318 | "dev": true, 319 | "requires": { 320 | "path-key": "^3.1.0", 321 | "shebang-command": "^2.0.0", 322 | "which": "^2.0.1" 323 | } 324 | }, 325 | "debug": { 326 | "version": "4.2.0", 327 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.2.0.tgz", 328 | "integrity": "sha512-IX2ncY78vDTjZMFUdmsvIRFY2Cf4FnD0wRs+nQwJU8Lu99/tPFdb0VybiiMTPe3I6rQmwsqQqRBvxU+bZ/I8sg==", 329 | "dev": true, 330 | "requires": { 331 | "ms": "2.1.2" 332 | } 333 | }, 334 | "deep-is": { 335 | "version": "0.1.3", 336 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", 337 | "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", 338 | "dev": true 339 | }, 340 | "define-properties": { 341 | "version": "1.1.3", 342 | "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", 343 | "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", 344 | "dev": true, 345 | "requires": { 346 | "object-keys": "^1.0.12" 347 | } 348 | }, 349 | "doctrine": { 350 | "version": "3.0.0", 351 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", 352 | "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", 353 | "dev": true, 354 | "requires": { 355 | "esutils": "^2.0.2" 356 | } 357 | }, 358 | "emoji-regex": { 359 | "version": "7.0.3", 360 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 361 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 362 | "dev": true 363 | }, 364 | "enquirer": { 365 | "version": "2.3.6", 366 | "resolved": "https://registry.npmjs.org/enquirer/-/enquirer-2.3.6.tgz", 367 | "integrity": "sha512-yjNnPr315/FjS4zIsUxYguYUPP2e1NK4d7E7ZOLiyYCcbFBiTMyID+2wvm2w6+pZ/odMA7cRkjhsPbltwBOrLg==", 368 | "dev": true, 369 | "requires": { 370 | "ansi-colors": "^4.1.1" 371 | } 372 | }, 373 | "error-ex": { 374 | "version": "1.3.2", 375 | "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", 376 | "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", 377 | "dev": true, 378 | "requires": { 379 | "is-arrayish": "^0.2.1" 380 | } 381 | }, 382 | "es-abstract": { 383 | "version": "1.18.0-next.1", 384 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", 385 | "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", 386 | "dev": true, 387 | "requires": { 388 | "es-to-primitive": "^1.2.1", 389 | "function-bind": "^1.1.1", 390 | "has": "^1.0.3", 391 | "has-symbols": "^1.0.1", 392 | "is-callable": "^1.2.2", 393 | "is-negative-zero": "^2.0.0", 394 | "is-regex": "^1.1.1", 395 | "object-inspect": "^1.8.0", 396 | "object-keys": "^1.1.1", 397 | "object.assign": "^4.1.1", 398 | "string.prototype.trimend": "^1.0.1", 399 | "string.prototype.trimstart": "^1.0.1" 400 | } 401 | }, 402 | "es-to-primitive": { 403 | "version": "1.2.1", 404 | "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", 405 | "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", 406 | "dev": true, 407 | "requires": { 408 | "is-callable": "^1.1.4", 409 | "is-date-object": "^1.0.1", 410 | "is-symbol": "^1.0.2" 411 | } 412 | }, 413 | "escape-string-regexp": { 414 | "version": "1.0.5", 415 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 416 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 417 | "dev": true 418 | }, 419 | "eslint": { 420 | "version": "7.10.0", 421 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-7.10.0.tgz", 422 | "integrity": "sha512-BDVffmqWl7JJXqCjAK6lWtcQThZB/aP1HXSH1JKwGwv0LQEdvpR7qzNrUT487RM39B5goWuboFad5ovMBmD8yA==", 423 | "dev": true, 424 | "requires": { 425 | "@babel/code-frame": "^7.0.0", 426 | "@eslint/eslintrc": "^0.1.3", 427 | "ajv": "^6.10.0", 428 | "chalk": "^4.0.0", 429 | "cross-spawn": "^7.0.2", 430 | "debug": "^4.0.1", 431 | "doctrine": "^3.0.0", 432 | "enquirer": "^2.3.5", 433 | "eslint-scope": "^5.1.1", 434 | "eslint-utils": "^2.1.0", 435 | "eslint-visitor-keys": "^1.3.0", 436 | "espree": "^7.3.0", 437 | "esquery": "^1.2.0", 438 | "esutils": "^2.0.2", 439 | "file-entry-cache": "^5.0.1", 440 | "functional-red-black-tree": "^1.0.1", 441 | "glob-parent": "^5.0.0", 442 | "globals": "^12.1.0", 443 | "ignore": "^4.0.6", 444 | "import-fresh": "^3.0.0", 445 | "imurmurhash": "^0.1.4", 446 | "is-glob": "^4.0.0", 447 | "js-yaml": "^3.13.1", 448 | "json-stable-stringify-without-jsonify": "^1.0.1", 449 | "levn": "^0.4.1", 450 | "lodash": "^4.17.19", 451 | "minimatch": "^3.0.4", 452 | "natural-compare": "^1.4.0", 453 | "optionator": "^0.9.1", 454 | "progress": "^2.0.0", 455 | "regexpp": "^3.1.0", 456 | "semver": "^7.2.1", 457 | "strip-ansi": "^6.0.0", 458 | "strip-json-comments": "^3.1.0", 459 | "table": "^5.2.3", 460 | "text-table": "^0.2.0", 461 | "v8-compile-cache": "^2.0.3" 462 | } 463 | }, 464 | "eslint-config-airbnb-base": { 465 | "version": "14.2.0", 466 | "resolved": "https://registry.npmjs.org/eslint-config-airbnb-base/-/eslint-config-airbnb-base-14.2.0.tgz", 467 | "integrity": "sha512-Snswd5oC6nJaevs3nZoLSTvGJBvzTfnBqOIArkf3cbyTyq9UD79wOk8s+RiL6bhca0p/eRO6veczhf6A/7Jy8Q==", 468 | "dev": true, 469 | "requires": { 470 | "confusing-browser-globals": "^1.0.9", 471 | "object.assign": "^4.1.0", 472 | "object.entries": "^1.1.2" 473 | } 474 | }, 475 | "eslint-config-prettier": { 476 | "version": "6.12.0", 477 | "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-6.12.0.tgz", 478 | "integrity": "sha512-9jWPlFlgNwRUYVoujvWTQ1aMO8o6648r+K7qU7K5Jmkbyqav1fuEZC0COYpGBxyiAJb65Ra9hrmFx19xRGwXWw==", 479 | "dev": true, 480 | "requires": { 481 | "get-stdin": "^6.0.0" 482 | } 483 | }, 484 | "eslint-import-resolver-node": { 485 | "version": "0.3.4", 486 | "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.4.tgz", 487 | "integrity": "sha512-ogtf+5AB/O+nM6DIeBUNr2fuT7ot9Qg/1harBfBtaP13ekEWFQEEMP94BCB7zaNW3gyY+8SHYF00rnqYwXKWOA==", 488 | "dev": true, 489 | "requires": { 490 | "debug": "^2.6.9", 491 | "resolve": "^1.13.1" 492 | }, 493 | "dependencies": { 494 | "debug": { 495 | "version": "2.6.9", 496 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 497 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 498 | "dev": true, 499 | "requires": { 500 | "ms": "2.0.0" 501 | } 502 | }, 503 | "ms": { 504 | "version": "2.0.0", 505 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 506 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", 507 | "dev": true 508 | } 509 | } 510 | }, 511 | "eslint-module-utils": { 512 | "version": "2.6.0", 513 | "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.6.0.tgz", 514 | "integrity": "sha512-6j9xxegbqe8/kZY8cYpcp0xhbK0EgJlg3g9mib3/miLaExuuwc3n5UEfSnU6hWMbT0FAYVvDbL9RrRgpUeQIvA==", 515 | "dev": true, 516 | "requires": { 517 | "debug": "^2.6.9", 518 | "pkg-dir": "^2.0.0" 519 | }, 520 | "dependencies": { 521 | "debug": { 522 | "version": "2.6.9", 523 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 524 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 525 | "dev": true, 526 | "requires": { 527 | "ms": "2.0.0" 528 | } 529 | }, 530 | "ms": { 531 | "version": "2.0.0", 532 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 533 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", 534 | "dev": true 535 | } 536 | } 537 | }, 538 | "eslint-plugin-import": { 539 | "version": "2.22.1", 540 | "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.22.1.tgz", 541 | "integrity": "sha512-8K7JjINHOpH64ozkAhpT3sd+FswIZTfMZTjdx052pnWrgRCVfp8op9tbjpAk3DdUeI/Ba4C8OjdC0r90erHEOw==", 542 | "dev": true, 543 | "requires": { 544 | "array-includes": "^3.1.1", 545 | "array.prototype.flat": "^1.2.3", 546 | "contains-path": "^0.1.0", 547 | "debug": "^2.6.9", 548 | "doctrine": "1.5.0", 549 | "eslint-import-resolver-node": "^0.3.4", 550 | "eslint-module-utils": "^2.6.0", 551 | "has": "^1.0.3", 552 | "minimatch": "^3.0.4", 553 | "object.values": "^1.1.1", 554 | "read-pkg-up": "^2.0.0", 555 | "resolve": "^1.17.0", 556 | "tsconfig-paths": "^3.9.0" 557 | }, 558 | "dependencies": { 559 | "debug": { 560 | "version": "2.6.9", 561 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 562 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 563 | "dev": true, 564 | "requires": { 565 | "ms": "2.0.0" 566 | } 567 | }, 568 | "doctrine": { 569 | "version": "1.5.0", 570 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz", 571 | "integrity": "sha1-N53Ocw9hZvds76TmcHoVmwLFpvo=", 572 | "dev": true, 573 | "requires": { 574 | "esutils": "^2.0.2", 575 | "isarray": "^1.0.0" 576 | } 577 | }, 578 | "ms": { 579 | "version": "2.0.0", 580 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 581 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", 582 | "dev": true 583 | } 584 | } 585 | }, 586 | "eslint-plugin-prettier": { 587 | "version": "3.1.4", 588 | "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-3.1.4.tgz", 589 | "integrity": "sha512-jZDa8z76klRqo+TdGDTFJSavwbnWK2ZpqGKNZ+VvweMW516pDUMmQ2koXvxEE4JhzNvTv+radye/bWGBmA6jmg==", 590 | "dev": true, 591 | "requires": { 592 | "prettier-linter-helpers": "^1.0.0" 593 | } 594 | }, 595 | "eslint-scope": { 596 | "version": "5.1.1", 597 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", 598 | "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", 599 | "dev": true, 600 | "requires": { 601 | "esrecurse": "^4.3.0", 602 | "estraverse": "^4.1.1" 603 | } 604 | }, 605 | "eslint-utils": { 606 | "version": "2.1.0", 607 | "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", 608 | "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", 609 | "dev": true, 610 | "requires": { 611 | "eslint-visitor-keys": "^1.1.0" 612 | } 613 | }, 614 | "eslint-visitor-keys": { 615 | "version": "1.3.0", 616 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", 617 | "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", 618 | "dev": true 619 | }, 620 | "espree": { 621 | "version": "7.3.0", 622 | "resolved": "https://registry.npmjs.org/espree/-/espree-7.3.0.tgz", 623 | "integrity": "sha512-dksIWsvKCixn1yrEXO8UosNSxaDoSYpq9reEjZSbHLpT5hpaCAKTLBwq0RHtLrIr+c0ByiYzWT8KTMRzoRCNlw==", 624 | "dev": true, 625 | "requires": { 626 | "acorn": "^7.4.0", 627 | "acorn-jsx": "^5.2.0", 628 | "eslint-visitor-keys": "^1.3.0" 629 | } 630 | }, 631 | "esprima": { 632 | "version": "4.0.1", 633 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 634 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", 635 | "dev": true 636 | }, 637 | "esquery": { 638 | "version": "1.3.1", 639 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.3.1.tgz", 640 | "integrity": "sha512-olpvt9QG0vniUBZspVRN6lwB7hOZoTRtT+jzR+tS4ffYx2mzbw+z0XCOk44aaLYKApNX5nMm+E+P6o25ip/DHQ==", 641 | "dev": true, 642 | "requires": { 643 | "estraverse": "^5.1.0" 644 | }, 645 | "dependencies": { 646 | "estraverse": { 647 | "version": "5.2.0", 648 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", 649 | "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", 650 | "dev": true 651 | } 652 | } 653 | }, 654 | "esrecurse": { 655 | "version": "4.3.0", 656 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", 657 | "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", 658 | "dev": true, 659 | "requires": { 660 | "estraverse": "^5.2.0" 661 | }, 662 | "dependencies": { 663 | "estraverse": { 664 | "version": "5.2.0", 665 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", 666 | "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", 667 | "dev": true 668 | } 669 | } 670 | }, 671 | "estraverse": { 672 | "version": "4.3.0", 673 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", 674 | "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", 675 | "dev": true 676 | }, 677 | "esutils": { 678 | "version": "2.0.3", 679 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", 680 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", 681 | "dev": true 682 | }, 683 | "fast-deep-equal": { 684 | "version": "3.1.3", 685 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", 686 | "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", 687 | "dev": true 688 | }, 689 | "fast-diff": { 690 | "version": "1.2.0", 691 | "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.2.0.tgz", 692 | "integrity": "sha512-xJuoT5+L99XlZ8twedaRf6Ax2TgQVxvgZOYoPKqZufmJib0tL2tegPBOZb1pVNgIhlqDlA0eO0c3wBvQcmzx4w==", 693 | "dev": true 694 | }, 695 | "fast-json-stable-stringify": { 696 | "version": "2.1.0", 697 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", 698 | "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", 699 | "dev": true 700 | }, 701 | "fast-levenshtein": { 702 | "version": "2.0.6", 703 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", 704 | "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", 705 | "dev": true 706 | }, 707 | "file-entry-cache": { 708 | "version": "5.0.1", 709 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz", 710 | "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", 711 | "dev": true, 712 | "requires": { 713 | "flat-cache": "^2.0.1" 714 | } 715 | }, 716 | "find-up": { 717 | "version": "2.1.0", 718 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", 719 | "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", 720 | "dev": true, 721 | "requires": { 722 | "locate-path": "^2.0.0" 723 | } 724 | }, 725 | "flat-cache": { 726 | "version": "2.0.1", 727 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz", 728 | "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", 729 | "dev": true, 730 | "requires": { 731 | "flatted": "^2.0.0", 732 | "rimraf": "2.6.3", 733 | "write": "1.0.3" 734 | } 735 | }, 736 | "flatted": { 737 | "version": "2.0.2", 738 | "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", 739 | "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", 740 | "dev": true 741 | }, 742 | "follow-redirects": { 743 | "version": "1.13.0", 744 | "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.13.0.tgz", 745 | "integrity": "sha512-aq6gF1BEKje4a9i9+5jimNFIpq4Q1WiwBToeRK5NvZBd/TRsmW8BsJfOEGkr76TbOyPVD3OVDN910EcUNtRYEA==" 746 | }, 747 | "fs.realpath": { 748 | "version": "1.0.0", 749 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 750 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", 751 | "dev": true 752 | }, 753 | "function-bind": { 754 | "version": "1.1.1", 755 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 756 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", 757 | "dev": true 758 | }, 759 | "functional-red-black-tree": { 760 | "version": "1.0.1", 761 | "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", 762 | "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", 763 | "dev": true 764 | }, 765 | "get-stdin": { 766 | "version": "6.0.0", 767 | "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-6.0.0.tgz", 768 | "integrity": "sha512-jp4tHawyV7+fkkSKyvjuLZswblUtz+SQKzSWnBbii16BuZksJlU1wuBYXY75r+duh/llF1ur6oNwi+2ZzjKZ7g==", 769 | "dev": true 770 | }, 771 | "glob": { 772 | "version": "7.1.6", 773 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", 774 | "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", 775 | "dev": true, 776 | "requires": { 777 | "fs.realpath": "^1.0.0", 778 | "inflight": "^1.0.4", 779 | "inherits": "2", 780 | "minimatch": "^3.0.4", 781 | "once": "^1.3.0", 782 | "path-is-absolute": "^1.0.0" 783 | } 784 | }, 785 | "glob-parent": { 786 | "version": "5.1.1", 787 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", 788 | "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", 789 | "dev": true, 790 | "requires": { 791 | "is-glob": "^4.0.1" 792 | } 793 | }, 794 | "globals": { 795 | "version": "12.4.0", 796 | "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", 797 | "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", 798 | "dev": true, 799 | "requires": { 800 | "type-fest": "^0.8.1" 801 | } 802 | }, 803 | "graceful-fs": { 804 | "version": "4.2.4", 805 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz", 806 | "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==", 807 | "dev": true 808 | }, 809 | "has": { 810 | "version": "1.0.3", 811 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 812 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 813 | "dev": true, 814 | "requires": { 815 | "function-bind": "^1.1.1" 816 | } 817 | }, 818 | "has-flag": { 819 | "version": "3.0.0", 820 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 821 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 822 | "dev": true 823 | }, 824 | "has-symbols": { 825 | "version": "1.0.1", 826 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", 827 | "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==", 828 | "dev": true 829 | }, 830 | "hosted-git-info": { 831 | "version": "2.8.8", 832 | "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", 833 | "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", 834 | "dev": true 835 | }, 836 | "ignore": { 837 | "version": "4.0.6", 838 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", 839 | "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", 840 | "dev": true 841 | }, 842 | "import-fresh": { 843 | "version": "3.2.1", 844 | "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.2.1.tgz", 845 | "integrity": "sha512-6e1q1cnWP2RXD9/keSkxHScg508CdXqXWgWBaETNhyuBFz+kUZlKboh+ISK+bU++DmbHimVBrOz/zzPe0sZ3sQ==", 846 | "dev": true, 847 | "requires": { 848 | "parent-module": "^1.0.0", 849 | "resolve-from": "^4.0.0" 850 | } 851 | }, 852 | "imurmurhash": { 853 | "version": "0.1.4", 854 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 855 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 856 | "dev": true 857 | }, 858 | "inflight": { 859 | "version": "1.0.6", 860 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 861 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 862 | "dev": true, 863 | "requires": { 864 | "once": "^1.3.0", 865 | "wrappy": "1" 866 | } 867 | }, 868 | "inherits": { 869 | "version": "2.0.4", 870 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 871 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 872 | "dev": true 873 | }, 874 | "is-arrayish": { 875 | "version": "0.2.1", 876 | "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", 877 | "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", 878 | "dev": true 879 | }, 880 | "is-callable": { 881 | "version": "1.2.2", 882 | "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.2.tgz", 883 | "integrity": "sha512-dnMqspv5nU3LoewK2N/y7KLtxtakvTuaCsU9FU50/QDmdbHNy/4/JuRtMHqRU22o3q+W89YQndQEeCVwK+3qrA==", 884 | "dev": true 885 | }, 886 | "is-date-object": { 887 | "version": "1.0.2", 888 | "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", 889 | "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==", 890 | "dev": true 891 | }, 892 | "is-extglob": { 893 | "version": "2.1.1", 894 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 895 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 896 | "dev": true 897 | }, 898 | "is-fullwidth-code-point": { 899 | "version": "2.0.0", 900 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 901 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 902 | "dev": true 903 | }, 904 | "is-glob": { 905 | "version": "4.0.1", 906 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 907 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 908 | "dev": true, 909 | "requires": { 910 | "is-extglob": "^2.1.1" 911 | } 912 | }, 913 | "is-negative-zero": { 914 | "version": "2.0.0", 915 | "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.0.tgz", 916 | "integrity": "sha1-lVOxIbD6wohp2p7UWeIMdUN4hGE=", 917 | "dev": true 918 | }, 919 | "is-regex": { 920 | "version": "1.1.1", 921 | "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.1.tgz", 922 | "integrity": "sha512-1+QkEcxiLlB7VEyFtyBg94e08OAsvq7FUBgApTq/w2ymCLyKJgDPsybBENVtA7XCQEgEXxKPonG+mvYRxh/LIg==", 923 | "dev": true, 924 | "requires": { 925 | "has-symbols": "^1.0.1" 926 | } 927 | }, 928 | "is-string": { 929 | "version": "1.0.5", 930 | "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.5.tgz", 931 | "integrity": "sha512-buY6VNRjhQMiF1qWDouloZlQbRhDPCebwxSjxMjxgemYT46YMd2NR0/H+fBhEfWX4A/w9TBJ+ol+okqJKFE6vQ==", 932 | "dev": true 933 | }, 934 | "is-symbol": { 935 | "version": "1.0.3", 936 | "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", 937 | "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", 938 | "dev": true, 939 | "requires": { 940 | "has-symbols": "^1.0.1" 941 | } 942 | }, 943 | "isarray": { 944 | "version": "1.0.0", 945 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 946 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", 947 | "dev": true 948 | }, 949 | "isexe": { 950 | "version": "2.0.0", 951 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 952 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 953 | "dev": true 954 | }, 955 | "js-tokens": { 956 | "version": "4.0.0", 957 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 958 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 959 | "dev": true 960 | }, 961 | "js-yaml": { 962 | "version": "3.14.0", 963 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.0.tgz", 964 | "integrity": "sha512-/4IbIeHcD9VMHFqDR/gQ7EdZdLimOvW2DdcxFjdyyZ9NsbS+ccrXqVWDtab/lRl5AlUqmpBx8EhPaWR+OtY17A==", 965 | "dev": true, 966 | "requires": { 967 | "argparse": "^1.0.7", 968 | "esprima": "^4.0.0" 969 | } 970 | }, 971 | "json-schema-traverse": { 972 | "version": "0.4.1", 973 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", 974 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", 975 | "dev": true 976 | }, 977 | "json-stable-stringify-without-jsonify": { 978 | "version": "1.0.1", 979 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", 980 | "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", 981 | "dev": true 982 | }, 983 | "json5": { 984 | "version": "1.0.1", 985 | "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", 986 | "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", 987 | "dev": true, 988 | "requires": { 989 | "minimist": "^1.2.0" 990 | } 991 | }, 992 | "levn": { 993 | "version": "0.4.1", 994 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", 995 | "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", 996 | "dev": true, 997 | "requires": { 998 | "prelude-ls": "^1.2.1", 999 | "type-check": "~0.4.0" 1000 | } 1001 | }, 1002 | "load-json-file": { 1003 | "version": "2.0.0", 1004 | "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz", 1005 | "integrity": "sha1-eUfkIUmvgNaWy/eXvKq8/h/inKg=", 1006 | "dev": true, 1007 | "requires": { 1008 | "graceful-fs": "^4.1.2", 1009 | "parse-json": "^2.2.0", 1010 | "pify": "^2.0.0", 1011 | "strip-bom": "^3.0.0" 1012 | } 1013 | }, 1014 | "locate-path": { 1015 | "version": "2.0.0", 1016 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", 1017 | "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", 1018 | "dev": true, 1019 | "requires": { 1020 | "p-locate": "^2.0.0", 1021 | "path-exists": "^3.0.0" 1022 | } 1023 | }, 1024 | "lodash": { 1025 | "version": "4.17.20", 1026 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.20.tgz", 1027 | "integrity": "sha512-PlhdFcillOINfeV7Ni6oF1TAEayyZBoZ8bcshTHqOYJYlrqzRK5hagpagky5o4HfCzzd1TRkXPMFq6cKk9rGmA==", 1028 | "dev": true 1029 | }, 1030 | "minimatch": { 1031 | "version": "3.0.4", 1032 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 1033 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 1034 | "dev": true, 1035 | "requires": { 1036 | "brace-expansion": "^1.1.7" 1037 | } 1038 | }, 1039 | "minimist": { 1040 | "version": "1.2.5", 1041 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 1042 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", 1043 | "dev": true 1044 | }, 1045 | "mkdirp": { 1046 | "version": "0.5.5", 1047 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", 1048 | "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", 1049 | "dev": true, 1050 | "requires": { 1051 | "minimist": "^1.2.5" 1052 | } 1053 | }, 1054 | "ms": { 1055 | "version": "2.1.2", 1056 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1057 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 1058 | "dev": true 1059 | }, 1060 | "natural-compare": { 1061 | "version": "1.4.0", 1062 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", 1063 | "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", 1064 | "dev": true 1065 | }, 1066 | "normalize-package-data": { 1067 | "version": "2.5.0", 1068 | "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", 1069 | "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", 1070 | "dev": true, 1071 | "requires": { 1072 | "hosted-git-info": "^2.1.4", 1073 | "resolve": "^1.10.0", 1074 | "semver": "2 || 3 || 4 || 5", 1075 | "validate-npm-package-license": "^3.0.1" 1076 | }, 1077 | "dependencies": { 1078 | "semver": { 1079 | "version": "5.7.1", 1080 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1081 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1082 | "dev": true 1083 | } 1084 | } 1085 | }, 1086 | "object-inspect": { 1087 | "version": "1.8.0", 1088 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.8.0.tgz", 1089 | "integrity": "sha512-jLdtEOB112fORuypAyl/50VRVIBIdVQOSUUGQHzJ4xBSbit81zRarz7GThkEFZy1RceYrWYcPcBFPQwHyAc1gA==", 1090 | "dev": true 1091 | }, 1092 | "object-keys": { 1093 | "version": "1.1.1", 1094 | "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", 1095 | "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", 1096 | "dev": true 1097 | }, 1098 | "object.assign": { 1099 | "version": "4.1.1", 1100 | "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.1.tgz", 1101 | "integrity": "sha512-VT/cxmx5yaoHSOTSyrCygIDFco+RsibY2NM0a4RdEeY/4KgqezwFtK1yr3U67xYhqJSlASm2pKhLVzPj2lr4bA==", 1102 | "dev": true, 1103 | "requires": { 1104 | "define-properties": "^1.1.3", 1105 | "es-abstract": "^1.18.0-next.0", 1106 | "has-symbols": "^1.0.1", 1107 | "object-keys": "^1.1.1" 1108 | } 1109 | }, 1110 | "object.entries": { 1111 | "version": "1.1.2", 1112 | "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.2.tgz", 1113 | "integrity": "sha512-BQdB9qKmb/HyNdMNWVr7O3+z5MUIx3aiegEIJqjMBbBf0YT9RRxTJSim4mzFqtyr7PDAHigq0N9dO0m0tRakQA==", 1114 | "dev": true, 1115 | "requires": { 1116 | "define-properties": "^1.1.3", 1117 | "es-abstract": "^1.17.5", 1118 | "has": "^1.0.3" 1119 | }, 1120 | "dependencies": { 1121 | "es-abstract": { 1122 | "version": "1.17.7", 1123 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", 1124 | "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", 1125 | "dev": true, 1126 | "requires": { 1127 | "es-to-primitive": "^1.2.1", 1128 | "function-bind": "^1.1.1", 1129 | "has": "^1.0.3", 1130 | "has-symbols": "^1.0.1", 1131 | "is-callable": "^1.2.2", 1132 | "is-regex": "^1.1.1", 1133 | "object-inspect": "^1.8.0", 1134 | "object-keys": "^1.1.1", 1135 | "object.assign": "^4.1.1", 1136 | "string.prototype.trimend": "^1.0.1", 1137 | "string.prototype.trimstart": "^1.0.1" 1138 | } 1139 | } 1140 | } 1141 | }, 1142 | "object.values": { 1143 | "version": "1.1.1", 1144 | "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.1.tgz", 1145 | "integrity": "sha512-WTa54g2K8iu0kmS/us18jEmdv1a4Wi//BZ/DTVYEcH0XhLM5NYdpDHja3gt57VrZLcNAO2WGA+KpWsDBaHt6eA==", 1146 | "dev": true, 1147 | "requires": { 1148 | "define-properties": "^1.1.3", 1149 | "es-abstract": "^1.17.0-next.1", 1150 | "function-bind": "^1.1.1", 1151 | "has": "^1.0.3" 1152 | }, 1153 | "dependencies": { 1154 | "es-abstract": { 1155 | "version": "1.17.7", 1156 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", 1157 | "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", 1158 | "dev": true, 1159 | "requires": { 1160 | "es-to-primitive": "^1.2.1", 1161 | "function-bind": "^1.1.1", 1162 | "has": "^1.0.3", 1163 | "has-symbols": "^1.0.1", 1164 | "is-callable": "^1.2.2", 1165 | "is-regex": "^1.1.1", 1166 | "object-inspect": "^1.8.0", 1167 | "object-keys": "^1.1.1", 1168 | "object.assign": "^4.1.1", 1169 | "string.prototype.trimend": "^1.0.1", 1170 | "string.prototype.trimstart": "^1.0.1" 1171 | } 1172 | } 1173 | } 1174 | }, 1175 | "once": { 1176 | "version": "1.4.0", 1177 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1178 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1179 | "dev": true, 1180 | "requires": { 1181 | "wrappy": "1" 1182 | } 1183 | }, 1184 | "optionator": { 1185 | "version": "0.9.1", 1186 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.1.tgz", 1187 | "integrity": "sha512-74RlY5FCnhq4jRxVUPKDaRwrVNXMqsGsiW6AJw4XK8hmtm10wC0ypZBLw5IIp85NZMr91+qd1RvvENwg7jjRFw==", 1188 | "dev": true, 1189 | "requires": { 1190 | "deep-is": "^0.1.3", 1191 | "fast-levenshtein": "^2.0.6", 1192 | "levn": "^0.4.1", 1193 | "prelude-ls": "^1.2.1", 1194 | "type-check": "^0.4.0", 1195 | "word-wrap": "^1.2.3" 1196 | } 1197 | }, 1198 | "p-limit": { 1199 | "version": "1.3.0", 1200 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", 1201 | "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", 1202 | "dev": true, 1203 | "requires": { 1204 | "p-try": "^1.0.0" 1205 | } 1206 | }, 1207 | "p-locate": { 1208 | "version": "2.0.0", 1209 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", 1210 | "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", 1211 | "dev": true, 1212 | "requires": { 1213 | "p-limit": "^1.1.0" 1214 | } 1215 | }, 1216 | "p-try": { 1217 | "version": "1.0.0", 1218 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", 1219 | "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", 1220 | "dev": true 1221 | }, 1222 | "parent-module": { 1223 | "version": "1.0.1", 1224 | "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", 1225 | "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", 1226 | "dev": true, 1227 | "requires": { 1228 | "callsites": "^3.0.0" 1229 | } 1230 | }, 1231 | "parse-json": { 1232 | "version": "2.2.0", 1233 | "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", 1234 | "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=", 1235 | "dev": true, 1236 | "requires": { 1237 | "error-ex": "^1.2.0" 1238 | } 1239 | }, 1240 | "path-exists": { 1241 | "version": "3.0.0", 1242 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", 1243 | "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", 1244 | "dev": true 1245 | }, 1246 | "path-is-absolute": { 1247 | "version": "1.0.1", 1248 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1249 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 1250 | "dev": true 1251 | }, 1252 | "path-key": { 1253 | "version": "3.1.1", 1254 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", 1255 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", 1256 | "dev": true 1257 | }, 1258 | "path-parse": { 1259 | "version": "1.0.6", 1260 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", 1261 | "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", 1262 | "dev": true 1263 | }, 1264 | "path-type": { 1265 | "version": "2.0.0", 1266 | "resolved": "https://registry.npmjs.org/path-type/-/path-type-2.0.0.tgz", 1267 | "integrity": "sha1-8BLMuEFbcJb8LaoQVMPXI4lZTHM=", 1268 | "dev": true, 1269 | "requires": { 1270 | "pify": "^2.0.0" 1271 | } 1272 | }, 1273 | "pify": { 1274 | "version": "2.3.0", 1275 | "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", 1276 | "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", 1277 | "dev": true 1278 | }, 1279 | "pkg-dir": { 1280 | "version": "2.0.0", 1281 | "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-2.0.0.tgz", 1282 | "integrity": "sha1-9tXREJ4Z1j7fQo4L1X4Sd3YVM0s=", 1283 | "dev": true, 1284 | "requires": { 1285 | "find-up": "^2.1.0" 1286 | } 1287 | }, 1288 | "prelude-ls": { 1289 | "version": "1.2.1", 1290 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", 1291 | "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", 1292 | "dev": true 1293 | }, 1294 | "prettier": { 1295 | "version": "2.1.2", 1296 | "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.1.2.tgz", 1297 | "integrity": "sha512-16c7K+x4qVlJg9rEbXl7HEGmQyZlG4R9AgP+oHKRMsMsuk8s+ATStlf1NpDqyBI1HpVyfjLOeMhH2LvuNvV5Vg==", 1298 | "dev": true 1299 | }, 1300 | "prettier-linter-helpers": { 1301 | "version": "1.0.0", 1302 | "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz", 1303 | "integrity": "sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==", 1304 | "dev": true, 1305 | "requires": { 1306 | "fast-diff": "^1.1.2" 1307 | } 1308 | }, 1309 | "progress": { 1310 | "version": "2.0.3", 1311 | "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", 1312 | "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", 1313 | "dev": true 1314 | }, 1315 | "punycode": { 1316 | "version": "2.1.1", 1317 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 1318 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", 1319 | "dev": true 1320 | }, 1321 | "read-pkg": { 1322 | "version": "2.0.0", 1323 | "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-2.0.0.tgz", 1324 | "integrity": "sha1-jvHAYjxqbbDcZxPEv6xGMysjaPg=", 1325 | "dev": true, 1326 | "requires": { 1327 | "load-json-file": "^2.0.0", 1328 | "normalize-package-data": "^2.3.2", 1329 | "path-type": "^2.0.0" 1330 | } 1331 | }, 1332 | "read-pkg-up": { 1333 | "version": "2.0.0", 1334 | "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-2.0.0.tgz", 1335 | "integrity": "sha1-a3KoBImE4MQeeVEP1en6mbO1Sb4=", 1336 | "dev": true, 1337 | "requires": { 1338 | "find-up": "^2.0.0", 1339 | "read-pkg": "^2.0.0" 1340 | } 1341 | }, 1342 | "regexpp": { 1343 | "version": "3.1.0", 1344 | "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.1.0.tgz", 1345 | "integrity": "sha512-ZOIzd8yVsQQA7j8GCSlPGXwg5PfmA1mrq0JP4nGhh54LaKN3xdai/vHUDu74pKwV8OxseMS65u2NImosQcSD0Q==", 1346 | "dev": true 1347 | }, 1348 | "resolve": { 1349 | "version": "1.17.0", 1350 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.17.0.tgz", 1351 | "integrity": "sha512-ic+7JYiV8Vi2yzQGFWOkiZD5Z9z7O2Zhm9XMaTxdJExKasieFCr+yXZ/WmXsckHiKl12ar0y6XiXDx3m4RHn1w==", 1352 | "dev": true, 1353 | "requires": { 1354 | "path-parse": "^1.0.6" 1355 | } 1356 | }, 1357 | "resolve-from": { 1358 | "version": "4.0.0", 1359 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", 1360 | "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", 1361 | "dev": true 1362 | }, 1363 | "rimraf": { 1364 | "version": "2.6.3", 1365 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", 1366 | "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", 1367 | "dev": true, 1368 | "requires": { 1369 | "glob": "^7.1.3" 1370 | } 1371 | }, 1372 | "semver": { 1373 | "version": "7.3.2", 1374 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", 1375 | "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", 1376 | "dev": true 1377 | }, 1378 | "shebang-command": { 1379 | "version": "2.0.0", 1380 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", 1381 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", 1382 | "dev": true, 1383 | "requires": { 1384 | "shebang-regex": "^3.0.0" 1385 | } 1386 | }, 1387 | "shebang-regex": { 1388 | "version": "3.0.0", 1389 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", 1390 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", 1391 | "dev": true 1392 | }, 1393 | "slice-ansi": { 1394 | "version": "2.1.0", 1395 | "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-2.1.0.tgz", 1396 | "integrity": "sha512-Qu+VC3EwYLldKa1fCxuuvULvSJOKEgk9pi8dZeCVK7TqBfUNTH4sFkk4joj8afVSfAYgJoSOetjx9QWOJ5mYoQ==", 1397 | "dev": true, 1398 | "requires": { 1399 | "ansi-styles": "^3.2.0", 1400 | "astral-regex": "^1.0.0", 1401 | "is-fullwidth-code-point": "^2.0.0" 1402 | } 1403 | }, 1404 | "spdx-correct": { 1405 | "version": "3.1.1", 1406 | "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.1.tgz", 1407 | "integrity": "sha512-cOYcUWwhCuHCXi49RhFRCyJEK3iPj1Ziz9DpViV3tbZOwXD49QzIN3MpOLJNxh2qwq2lJJZaKMVw9qNi4jTC0w==", 1408 | "dev": true, 1409 | "requires": { 1410 | "spdx-expression-parse": "^3.0.0", 1411 | "spdx-license-ids": "^3.0.0" 1412 | } 1413 | }, 1414 | "spdx-exceptions": { 1415 | "version": "2.3.0", 1416 | "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz", 1417 | "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==", 1418 | "dev": true 1419 | }, 1420 | "spdx-expression-parse": { 1421 | "version": "3.0.1", 1422 | "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz", 1423 | "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==", 1424 | "dev": true, 1425 | "requires": { 1426 | "spdx-exceptions": "^2.1.0", 1427 | "spdx-license-ids": "^3.0.0" 1428 | } 1429 | }, 1430 | "spdx-license-ids": { 1431 | "version": "3.0.6", 1432 | "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.6.tgz", 1433 | "integrity": "sha512-+orQK83kyMva3WyPf59k1+Y525csj5JejicWut55zeTWANuN17qSiSLUXWtzHeNWORSvT7GLDJ/E/XiIWoXBTw==", 1434 | "dev": true 1435 | }, 1436 | "sprintf-js": { 1437 | "version": "1.0.3", 1438 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 1439 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", 1440 | "dev": true 1441 | }, 1442 | "string-width": { 1443 | "version": "3.1.0", 1444 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 1445 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 1446 | "dev": true, 1447 | "requires": { 1448 | "emoji-regex": "^7.0.1", 1449 | "is-fullwidth-code-point": "^2.0.0", 1450 | "strip-ansi": "^5.1.0" 1451 | }, 1452 | "dependencies": { 1453 | "ansi-regex": { 1454 | "version": "4.1.0", 1455 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 1456 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 1457 | "dev": true 1458 | }, 1459 | "strip-ansi": { 1460 | "version": "5.2.0", 1461 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 1462 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 1463 | "dev": true, 1464 | "requires": { 1465 | "ansi-regex": "^4.1.0" 1466 | } 1467 | } 1468 | } 1469 | }, 1470 | "string.prototype.trimend": { 1471 | "version": "1.0.1", 1472 | "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.1.tgz", 1473 | "integrity": "sha512-LRPxFUaTtpqYsTeNKaFOw3R4bxIzWOnbQ837QfBylo8jIxtcbK/A/sMV7Q+OAV/vWo+7s25pOE10KYSjaSO06g==", 1474 | "dev": true, 1475 | "requires": { 1476 | "define-properties": "^1.1.3", 1477 | "es-abstract": "^1.17.5" 1478 | }, 1479 | "dependencies": { 1480 | "es-abstract": { 1481 | "version": "1.17.7", 1482 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", 1483 | "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", 1484 | "dev": true, 1485 | "requires": { 1486 | "es-to-primitive": "^1.2.1", 1487 | "function-bind": "^1.1.1", 1488 | "has": "^1.0.3", 1489 | "has-symbols": "^1.0.1", 1490 | "is-callable": "^1.2.2", 1491 | "is-regex": "^1.1.1", 1492 | "object-inspect": "^1.8.0", 1493 | "object-keys": "^1.1.1", 1494 | "object.assign": "^4.1.1", 1495 | "string.prototype.trimend": "^1.0.1", 1496 | "string.prototype.trimstart": "^1.0.1" 1497 | } 1498 | } 1499 | } 1500 | }, 1501 | "string.prototype.trimstart": { 1502 | "version": "1.0.1", 1503 | "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.1.tgz", 1504 | "integrity": "sha512-XxZn+QpvrBI1FOcg6dIpxUPgWCPuNXvMD72aaRaUQv1eD4e/Qy8i/hFTe0BUmD60p/QA6bh1avmuPTfNjqVWRw==", 1505 | "dev": true, 1506 | "requires": { 1507 | "define-properties": "^1.1.3", 1508 | "es-abstract": "^1.17.5" 1509 | }, 1510 | "dependencies": { 1511 | "es-abstract": { 1512 | "version": "1.17.7", 1513 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", 1514 | "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", 1515 | "dev": true, 1516 | "requires": { 1517 | "es-to-primitive": "^1.2.1", 1518 | "function-bind": "^1.1.1", 1519 | "has": "^1.0.3", 1520 | "has-symbols": "^1.0.1", 1521 | "is-callable": "^1.2.2", 1522 | "is-regex": "^1.1.1", 1523 | "object-inspect": "^1.8.0", 1524 | "object-keys": "^1.1.1", 1525 | "object.assign": "^4.1.1", 1526 | "string.prototype.trimend": "^1.0.1", 1527 | "string.prototype.trimstart": "^1.0.1" 1528 | } 1529 | } 1530 | } 1531 | }, 1532 | "strip-ansi": { 1533 | "version": "6.0.0", 1534 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 1535 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 1536 | "dev": true, 1537 | "requires": { 1538 | "ansi-regex": "^5.0.0" 1539 | } 1540 | }, 1541 | "strip-bom": { 1542 | "version": "3.0.0", 1543 | "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", 1544 | "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", 1545 | "dev": true 1546 | }, 1547 | "strip-json-comments": { 1548 | "version": "3.1.1", 1549 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", 1550 | "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", 1551 | "dev": true 1552 | }, 1553 | "supports-color": { 1554 | "version": "5.5.0", 1555 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1556 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1557 | "dev": true, 1558 | "requires": { 1559 | "has-flag": "^3.0.0" 1560 | } 1561 | }, 1562 | "table": { 1563 | "version": "5.4.6", 1564 | "resolved": "https://registry.npmjs.org/table/-/table-5.4.6.tgz", 1565 | "integrity": "sha512-wmEc8m4fjnob4gt5riFRtTu/6+4rSe12TpAELNSqHMfF3IqnA+CH37USM6/YR3qRZv7e56kAEAtd6nKZaxe0Ug==", 1566 | "dev": true, 1567 | "requires": { 1568 | "ajv": "^6.10.2", 1569 | "lodash": "^4.17.14", 1570 | "slice-ansi": "^2.1.0", 1571 | "string-width": "^3.0.0" 1572 | } 1573 | }, 1574 | "text-table": { 1575 | "version": "0.2.0", 1576 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", 1577 | "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", 1578 | "dev": true 1579 | }, 1580 | "tsconfig-paths": { 1581 | "version": "3.9.0", 1582 | "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.9.0.tgz", 1583 | "integrity": "sha512-dRcuzokWhajtZWkQsDVKbWyY+jgcLC5sqJhg2PSgf4ZkH2aHPvaOY8YWGhmjb68b5qqTfasSsDO9k7RUiEmZAw==", 1584 | "dev": true, 1585 | "requires": { 1586 | "@types/json5": "^0.0.29", 1587 | "json5": "^1.0.1", 1588 | "minimist": "^1.2.0", 1589 | "strip-bom": "^3.0.0" 1590 | } 1591 | }, 1592 | "type-check": { 1593 | "version": "0.4.0", 1594 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", 1595 | "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", 1596 | "dev": true, 1597 | "requires": { 1598 | "prelude-ls": "^1.2.1" 1599 | } 1600 | }, 1601 | "type-fest": { 1602 | "version": "0.8.1", 1603 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 1604 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", 1605 | "dev": true 1606 | }, 1607 | "uri-js": { 1608 | "version": "4.4.0", 1609 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.0.tgz", 1610 | "integrity": "sha512-B0yRTzYdUCCn9n+F4+Gh4yIDtMQcaJsmYBDsTSG8g/OejKBodLQ2IHfN3bM7jUsRXndopT7OIXWdYqc1fjmV6g==", 1611 | "dev": true, 1612 | "requires": { 1613 | "punycode": "^2.1.0" 1614 | } 1615 | }, 1616 | "v8-compile-cache": { 1617 | "version": "2.1.1", 1618 | "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.1.1.tgz", 1619 | "integrity": "sha512-8OQ9CL+VWyt3JStj7HX7/ciTL2V3Rl1Wf5OL+SNTm0yK1KvtReVulksyeRnCANHHuUxHlQig+JJDlUhBt1NQDQ==", 1620 | "dev": true 1621 | }, 1622 | "validate-npm-package-license": { 1623 | "version": "3.0.4", 1624 | "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", 1625 | "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", 1626 | "dev": true, 1627 | "requires": { 1628 | "spdx-correct": "^3.0.0", 1629 | "spdx-expression-parse": "^3.0.0" 1630 | } 1631 | }, 1632 | "which": { 1633 | "version": "2.0.2", 1634 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 1635 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 1636 | "dev": true, 1637 | "requires": { 1638 | "isexe": "^2.0.0" 1639 | } 1640 | }, 1641 | "word-wrap": { 1642 | "version": "1.2.3", 1643 | "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", 1644 | "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", 1645 | "dev": true 1646 | }, 1647 | "wrappy": { 1648 | "version": "1.0.2", 1649 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1650 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1651 | "dev": true 1652 | }, 1653 | "write": { 1654 | "version": "1.0.3", 1655 | "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz", 1656 | "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", 1657 | "dev": true, 1658 | "requires": { 1659 | "mkdirp": "^0.5.1" 1660 | } 1661 | } 1662 | } 1663 | } 1664 | -------------------------------------------------------------------------------- /maxcoin/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "maxcoin", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "test.js", 6 | "scripts": { 7 | "start": "node main.js" 8 | }, 9 | "author": "", 10 | "license": "ISC", 11 | "dependencies": { 12 | "axios": "^0.20.0" 13 | }, 14 | "devDependencies": { 15 | "eslint": "^7.10.0", 16 | "eslint-config-airbnb-base": "^14.2.0", 17 | "eslint-config-prettier": "^6.12.0", 18 | "eslint-plugin-import": "^2.22.0", 19 | "eslint-plugin-prettier": "^3.1.4", 20 | "prettier": "^2.1.2" 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /maxcoin/services/CoinAPI.js: -------------------------------------------------------------------------------- 1 | const axios = require("axios"); 2 | 3 | class CoinAPI { 4 | constructor() { 5 | this.apiUrl = "https://api.coindesk.com/v1/bpi/historical/close.json"; 6 | } 7 | 8 | // eslint-disable-next-line class-methods-use-this 9 | formatDate(date) { 10 | const d = new Date(date); 11 | let month = `${d.getMonth() + 1}`; 12 | let day = `${d.getDate()}`; 13 | const year = `${d.getFullYear()}`; 14 | 15 | if (month.length < 2) month = `0${month}`; 16 | if (day.length < 2) day = `0${day}`; 17 | 18 | return [year, month, day].join("-"); 19 | } 20 | 21 | async fetch() { 22 | const today = new Date(); 23 | const end = this.formatDate(today); 24 | const start = this.formatDate(today.setFullYear(today.getFullYear() - 5)); 25 | const url = `${this.apiUrl}?start=${start}&end=${end}`; 26 | const response = await axios.get(url); 27 | return response.data; 28 | } 29 | } 30 | 31 | module.exports = CoinAPI; 32 | -------------------------------------------------------------------------------- /maxcoin/services/backend/MongoBackend.js: -------------------------------------------------------------------------------- 1 | /* eslint-disable no-useless-constructor */ 2 | /* eslint-disable class-methods-use-this */ 3 | /* eslint-disable no-empty-function */ 4 | const CoinAPI = require('../CoinAPI'); 5 | 6 | class MongoBackend { 7 | 8 | constructor() { 9 | this.coinAPI = new CoinAPI(); 10 | } 11 | 12 | async connect() {} 13 | 14 | async disconnect() {} 15 | 16 | async insert() {} 17 | 18 | async getMax() {} 19 | 20 | async max() {} 21 | } 22 | 23 | module.exports = MongoBackend; -------------------------------------------------------------------------------- /maxcoin/services/backend/MySQLBackend.js: -------------------------------------------------------------------------------- 1 | /* eslint-disable no-useless-constructor */ 2 | /* eslint-disable class-methods-use-this */ 3 | /* eslint-disable no-empty-function */ 4 | const CoinAPI = require('../CoinAPI'); 5 | 6 | class MySQLBackend { 7 | 8 | constructor() { 9 | this.coinAPI = new CoinAPI(); 10 | } 11 | 12 | async connect() { 13 | 14 | } 15 | 16 | async disconnect() { 17 | 18 | } 19 | 20 | async insert() { 21 | 22 | } 23 | 24 | async getMax() { 25 | 26 | } 27 | 28 | async max() { 29 | 30 | } 31 | } 32 | 33 | module.exports = MySQLBackend; -------------------------------------------------------------------------------- /maxcoin/services/backend/RedisBackend.js: -------------------------------------------------------------------------------- 1 | /* eslint-disable no-useless-constructor */ 2 | /* eslint-disable class-methods-use-this */ 3 | /* eslint-disable no-empty-function */ 4 | const CoinAPI = require('../CoinAPI'); 5 | 6 | class RedisBackend { 7 | 8 | constructor() { 9 | this.coinAPI = new CoinAPI(); 10 | } 11 | 12 | async connect() { 13 | 14 | } 15 | 16 | async disconnect() { 17 | 18 | } 19 | 20 | async insert() { 21 | 22 | } 23 | 24 | async getMax() { 25 | 26 | } 27 | 28 | async max() { 29 | 30 | } 31 | } 32 | 33 | module.exports = RedisBackend; -------------------------------------------------------------------------------- /shopper/.eslintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "env": { 3 | "commonjs": true, 4 | "es6": true, 5 | "node": true 6 | }, 7 | "extends": ["airbnb-base", "prettier"], 8 | "plugins": ["prettier"], 9 | "globals": { 10 | "Atomics": "readonly", 11 | "SharedArrayBuffer": "readonly" 12 | }, 13 | "parserOptions": { 14 | "ecmaVersion": 2018 15 | }, 16 | "rules": { 17 | "no-console": 0 18 | } 19 | } 20 | -------------------------------------------------------------------------------- /shopper/.gitignore: -------------------------------------------------------------------------------- 1 | /node_modules 2 | .DS_STORE -------------------------------------------------------------------------------- /shopper/client/css/site.css: -------------------------------------------------------------------------------- 1 | body { 2 | padding-top: 5rem; 3 | } -------------------------------------------------------------------------------- /shopper/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "shopper", 3 | "version": "0.0.0", 4 | "private": true, 5 | "author": "Daniel Khan ", 6 | "scripts": { 7 | "start": "node ./server/bin/start", 8 | "dev": "nodemon ./server/bin/start" 9 | }, 10 | "dependencies": { 11 | "body-parser": "~1.19.0", 12 | "cookie-parser": "~1.4.5", 13 | "express": "~4.17.1", 14 | "express-session": "^1.17.1", 15 | "pug": "~3.0.0" 16 | }, 17 | "devDependencies": { 18 | "eslint": "^7.10.0", 19 | "eslint-config-airbnb-base": "^14.2.0", 20 | "eslint-config-prettier": "^6.12.0", 21 | "eslint-plugin-import": "^2.22.1", 22 | "eslint-plugin-prettier": "^3.1.4", 23 | "nodemon": "^2.0.4", 24 | "prettier": "^2.1.2" 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /shopper/server/app.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const path = require('path'); 3 | const bodyParser = require('body-parser'); 4 | const session = require('express-session'); 5 | const routeHandler = require('./routes'); 6 | 7 | module.exports = (config) => { 8 | const app = express(); 9 | 10 | // view engine setup 11 | app.set('views', path.join(__dirname, 'views')); 12 | app.set('view engine', 'pug'); 13 | 14 | app.use(bodyParser.json()); 15 | app.use(bodyParser.urlencoded({ extended: false })); 16 | 17 | app.set('trust proxy', 1); // trust first proxy 18 | app.use(session({ 19 | secret: 'very secret secret to encyrpt session', 20 | resave: false, 21 | saveUninitialized: false, 22 | })); 23 | 24 | app.use(express.static(path.join(__dirname, '../client'))); 25 | app.get('/favicon.ico', (req, res) => { 26 | res.status(204); 27 | }); 28 | app.get('/robots.txt', (req, res) => { 29 | res.status(204); 30 | }); 31 | 32 | // Define 'global' template variables here 33 | app.use(async (req, res, next) => { 34 | // To show the application name on the page 35 | res.locals.applicationName = config.applicationName; 36 | 37 | // Set up flash messaging 38 | if (!req.session.messages) { 39 | req.session.messages = []; 40 | } 41 | res.locals.messages = req.session.messages; 42 | return next(); 43 | }); 44 | 45 | app.use('/', routeHandler(config)); 46 | 47 | // catch 404 and forward to error handler 48 | app.use((req, res, next) => { 49 | const err = new Error(`Not Found (${req.url})`); 50 | err.status = 404; 51 | next(err); 52 | }); 53 | 54 | // error handler 55 | app.use((err, req, res) => { 56 | // set locals, only providing error in development 57 | res.locals.message = err.message; 58 | res.locals.error = req.app.get('env') === 'development' ? err : {}; 59 | 60 | // render the error page 61 | res.status(err.status || 500); 62 | res.render('error'); 63 | }); 64 | 65 | return app; 66 | }; 67 | -------------------------------------------------------------------------------- /shopper/server/bin/start.js: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | const http = require('http'); 4 | 5 | const config = require('../config'); 6 | const App = require('../app'); 7 | 8 | /* Logic to start the application */ 9 | const app = App(config); 10 | const port = process.env.PORT || '3000'; 11 | app.set('port', port); 12 | 13 | function onError(error) { 14 | if (error.syscall !== 'listen') { 15 | throw error; 16 | } 17 | const bind = typeof port === 'string' 18 | ? `Pipe ${port}` 19 | : `Port ${port}`; 20 | 21 | // handle specific listen errors with friendly messages 22 | switch (error.code) { 23 | case 'EACCES': 24 | console.error(`${bind} requires elevated privileges`); 25 | process.exit(1); 26 | break; 27 | case 'EADDRINUSE': 28 | console.error(`${bind} is already in use`); 29 | process.exit(1); 30 | break; 31 | default: 32 | throw error; 33 | } 34 | } 35 | 36 | const server = http.createServer(app); 37 | function onListening() { 38 | const addr = server.address(); 39 | const bind = typeof addr === 'string' 40 | ? `pipe ${addr}` 41 | : `port ${addr.port}`; 42 | 43 | console.info(`${config.applicationName} listening on ${bind}`); 44 | } 45 | server.on('error', onError); 46 | server.on('listening', onListening); 47 | 48 | server.listen(port); -------------------------------------------------------------------------------- /shopper/server/config/index.js: -------------------------------------------------------------------------------- 1 | const pkg = require('../../package.json'); 2 | 3 | module.exports = { 4 | applicationName: pkg.name, 5 | }; 6 | -------------------------------------------------------------------------------- /shopper/server/models/mongoose/Item.js: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/danielkhan/linkedin-node-databases/9a4f86890bcb10e1f63ee536fba711300443be3b/shopper/server/models/mongoose/Item.js -------------------------------------------------------------------------------- /shopper/server/models/mongoose/User.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose'); 2 | const bcrypt = require('bcrypt'); 3 | 4 | const UserSchema = mongoose.Schema({ 5 | email: { 6 | // Trim and lowercase 7 | type: String, required: true, index: { unique: true }, lowercase: true, trim: true, 8 | }, 9 | password: { 10 | type: String, required: true, trim: true, 11 | }, 12 | }, { timestamps: true }); 13 | 14 | async function generateHash(password) { 15 | const COST = 12; 16 | return bcrypt.hash(password, COST); 17 | } 18 | 19 | UserSchema.pre('save', function preSave(next) { 20 | const user = this; 21 | 22 | // Only create a new password hash if the field was updated 23 | if(user.isModified('password')) { 24 | return generateHash(user.password).then(hash => { 25 | user.password = hash; 26 | return next(); 27 | }).catch(error => { 28 | return next(error); 29 | }); 30 | } 31 | return next(); 32 | }); 33 | 34 | UserSchema.methods.comparePassword = async function comparePassword(candidatePassword) { 35 | return bcrypt.compare(candidatePassword, this.password); 36 | }; 37 | 38 | module.exports = mongoose.model('User', UserSchema); -------------------------------------------------------------------------------- /shopper/server/routes/admin/item/index.js: -------------------------------------------------------------------------------- 1 | const express = require("express"); 2 | 3 | module.exports = () => { 4 | const router = express.Router(); 5 | 6 | router.get("/:itemId?", async (req, res) => { 7 | return res.render("admin/item", {}); 8 | 9 | /* 10 | try { 11 | const items = await ItemService.getAll(); 12 | let item = null; 13 | 14 | // The optional param itemId is present 15 | if (req.params.itemId) { 16 | item = await ItemService.getOne(req.params.itemId); 17 | } 18 | 19 | return res.render("admin/item", { 20 | items, 21 | item, 22 | }); 23 | } catch (err) { 24 | return next(err); 25 | } 26 | */ 27 | }); 28 | 29 | // Save or update item 30 | router.post("/", async (req, res, next) => { 31 | return next("Not implemented"); 32 | 33 | /* 34 | // Massage the passed in form data a bit 35 | const sku = req.body.sku.trim(); 36 | const name = req.body.name.trim(); 37 | const price = req.body.price.trim(); 38 | 39 | // Make sure that the passed data is complete 40 | if (!sku || !name || !price) { 41 | req.session.messages.push({ 42 | type: "warning", 43 | text: "Please enter SKU, name and price!", 44 | }); 45 | return res.redirect("/admin/item"); 46 | } 47 | 48 | try { 49 | // If there was no existing item we now want to create a new item object 50 | if (!req.body.itemId) { 51 | await ItemService.create({ sku, name, price }); 52 | } else { 53 | const itemData = { 54 | sku, 55 | name, 56 | price, 57 | }; 58 | await ItemService.update(req.body.itemId, itemData); 59 | } 60 | req.session.messages.push({ 61 | type: "success", 62 | text: `The item was ${ 63 | req.body.itemId ? "updated" : "created" 64 | } successfully!`, 65 | }); 66 | return res.redirect("/admin/item"); 67 | } catch (err) { 68 | req.session.messages.push({ 69 | type: "danger", 70 | text: "There was an error while saving the item!", 71 | }); 72 | console.error(err); 73 | return res.redirect("/admin/item"); 74 | } 75 | */ 76 | }); 77 | 78 | // Delete item 79 | router.get("/delete/:itemId", async (req, res, next) => { 80 | return next("Not implemented"); 81 | 82 | /* 83 | try { 84 | await ItemService.remove(req.params.itemId); 85 | } catch (err) { 86 | // Error handling 87 | req.session.messages.push({ 88 | type: "danger", 89 | text: "There was an error while deleting the item!", 90 | }); 91 | console.error(err); 92 | return res.redirect("/admin/item"); 93 | } 94 | // Let the item knows that everything went fine 95 | req.session.messages.push({ 96 | type: "success", 97 | text: "The item was successfully deleted!", 98 | }); 99 | return res.redirect("/admin/item"); 100 | */ 101 | }); 102 | return router; 103 | }; 104 | -------------------------------------------------------------------------------- /shopper/server/routes/admin/orders/index.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | 3 | module.exports = () => { 4 | const router = express.Router(); 5 | 6 | router.get('/', async (req, res) => { 7 | return res.render('admin/orders', {}); 8 | 9 | /* 10 | try { 11 | // Get all orders 12 | const orderResult = await order.getAll(); 13 | // Run map on the data to convert it into nested arrays with orders and orderitems 14 | const orders = orderResult.map(item => item.get({ plain: true })); 15 | return res.render('admin/orders', { orders }); 16 | } catch (err) { 17 | req.session.messages.push({ 18 | type: 'danger', 19 | text: 'There was an error fetching the orders', 20 | }); 21 | console.error(err); 22 | return next(err); 23 | } 24 | */ 25 | }); 26 | 27 | router.get('/setshipped/:orderId', async (req, res, next) => { 28 | return next('Not Implemented'); 29 | 30 | /* 31 | try { 32 | await order.setStatus(req.params.orderId, 'Shipped'); 33 | req.session.messages.push({ 34 | type: 'success', 35 | text: 'Status updated', 36 | }); 37 | return res.redirect('/admin/orders'); 38 | } catch (err) { 39 | req.session.messages.push({ 40 | type: 'danger', 41 | text: 'There was an updaeting the order', 42 | }); 43 | console.error(err); 44 | return res.redirect('/admin/orders'); 45 | } 46 | */ 47 | }); 48 | 49 | return router; 50 | }; 51 | 52 | -------------------------------------------------------------------------------- /shopper/server/routes/admin/user/index.js: -------------------------------------------------------------------------------- 1 | const express = require("express"); 2 | 3 | module.exports = () => { 4 | const router = express.Router(); 5 | 6 | router.get("/:userId?", async (req, res) => { 7 | return res.render("admin/user"); 8 | 9 | /* 10 | try { 11 | const users = await UserService.getAll(); 12 | 13 | let user = null; 14 | 15 | // The optional userId param was passed 16 | if (req.params.userId) { 17 | user = await UserService.getOne(req.params.userId); 18 | } 19 | return res.render("admin/user", { 20 | users, 21 | user, 22 | }); 23 | } catch (err) { 24 | return next(err); 25 | } 26 | */ 27 | }); 28 | 29 | // Save or update user 30 | router.post("/", async (req, res, next) => { 31 | return next("Not implemented"); 32 | 33 | /* 34 | const email = req.body.email.trim(); 35 | const password = req.body.password.trim(); 36 | // Add this here because on update we might want to keep the password as it is 37 | if (!email || (!password && !req.body.userId)) { 38 | req.session.messages.push({ 39 | type: "warning", 40 | text: "Please enter email address and password!", 41 | }); 42 | return res.redirect("/admin/user"); 43 | } 44 | try { 45 | // If there was no existing user we now want to create a new user object 46 | if (!req.body.userId) { 47 | await UserService.create({ email, password }); 48 | } else { 49 | const userData = { 50 | email, 51 | }; 52 | // Add this if because password does not need to be changed on updated 53 | if (password) { 54 | userData.password = password; 55 | } 56 | await UserService.update(req.body.userId, userData); 57 | } 58 | req.session.messages.push({ 59 | type: "success", 60 | text: `The user was ${ 61 | req.body.userId ? "updated" : "created" 62 | } successfully!`, 63 | }); 64 | return res.redirect("/admin/user"); 65 | } catch (err) { 66 | req.session.messages.push({ 67 | type: "danger", 68 | text: "There was an error while saving the user!", 69 | }); 70 | console.error(err); 71 | return res.redirect("/admin/user"); 72 | } 73 | */ 74 | }); 75 | 76 | // Delete user 77 | router.get("/delete/:userId", async (req, res, next) => { 78 | return next("Not implemented"); 79 | 80 | /* 81 | try { 82 | await UserService.remove(req.params.userId); 83 | } catch (err) { 84 | // Error handling 85 | req.session.messages.push({ 86 | type: "danger", 87 | text: "There was an error while deleting the user!", 88 | }); 89 | console.error(err); 90 | return res.redirect("/admin/user"); 91 | } 92 | // Let the user knows that everything went fine 93 | req.session.messages.push({ 94 | type: "success", 95 | text: "The user was successfully deleted!", 96 | }); 97 | return res.redirect("/admin/user"); 98 | */ 99 | }); 100 | 101 | router.get("/impersonate/:userId", (req, res, next) => { 102 | return next("Not implemented"); 103 | 104 | /* 105 | req.session.userId = req.params.userId; 106 | req.session.messages.push({ 107 | type: "success", 108 | text: "User successfully switched", 109 | }); 110 | return res.redirect("/admin/user"); 111 | */ 112 | }); 113 | 114 | return router; 115 | }; 116 | -------------------------------------------------------------------------------- /shopper/server/routes/basket/index.js: -------------------------------------------------------------------------------- 1 | const express = require("express"); 2 | 3 | module.exports = () => { 4 | const router = express.Router(); 5 | 6 | router.get("/", async (req, res) => { 7 | return res.render("basket", {}); 8 | 9 | /* 10 | if (!res.locals.currentUser) { 11 | req.session.messages.push({ 12 | type: "warning", 13 | text: "Please log in first", 14 | }); 15 | return res.redirect("/shop"); 16 | } 17 | const basket = new BasketService( 18 | config.redis.client, 19 | res.locals.currentUser.id 20 | ); 21 | const basketItems = await basket.getAll(); 22 | let items = []; 23 | if (basketItems) { 24 | items = await Promise.all( 25 | Object.keys(basketItems).map(async (itemId) => { 26 | const item = await ItemService.getOne(itemId); 27 | item.quantity = basketItems[itemId]; 28 | return item; 29 | }) 30 | ); 31 | } 32 | return res.render("basket", { items }); 33 | */ 34 | }); 35 | 36 | router.get("/remove/:itemId", async (req, res, next) => { 37 | return next("Not implemented"); 38 | 39 | /* 40 | if (!res.locals.currentUser) { 41 | req.session.messages.push({ 42 | type: "warning", 43 | text: "Please log in first", 44 | }); 45 | return res.redirect("/shop"); 46 | } 47 | 48 | try { 49 | const basket = new BasketService( 50 | config.redis.client, 51 | res.locals.currentUser.id 52 | ); 53 | await basket.remove(req.params.itemId); 54 | req.session.messages.push({ 55 | type: "success", 56 | text: "The item was removed from the the basket", 57 | }); 58 | } catch (err) { 59 | req.session.messages.push({ 60 | type: "danger", 61 | text: "There was an error removing the item from the basket", 62 | }); 63 | console.error(err); 64 | return res.redirect("/basket"); 65 | } 66 | 67 | return res.redirect("/basket"); 68 | */ 69 | }); 70 | 71 | router.get("/buy", async (req, res, next) => { 72 | return next("Not implemented"); 73 | /* 74 | if (!res.locals.currentUser) { 75 | req.session.messages.push({ 76 | type: "warning", 77 | text: "Please log in first", 78 | }); 79 | return res.redirect("/shop"); 80 | } 81 | try { 82 | const userId = res.locals.currentUser.id; 83 | const user = res.locals.currentUser; 84 | 85 | // Get all basket items for a user 86 | const basket = new BasketService(config.redis.client, userId); 87 | const basketItems = await basket.getAll(); 88 | 89 | // be defensive 90 | if (!basketItems) { 91 | throw new Error("No items found in basket"); 92 | } 93 | 94 | // Find the item for each basket entry and add the quantity to it 95 | // Return a new array with items plus quantity as new field 96 | const items = await Promise.all( 97 | Object.keys(basketItems).map(async (key) => { 98 | const item = await ItemService.getOne(key); 99 | return { 100 | sku: item.sku, 101 | qty: basketItems[key], 102 | price: item.price, 103 | name: item.name, 104 | }; 105 | }) 106 | ); 107 | 108 | // Run this in a sequelize transaction 109 | await order.inTransaction(async (t) => { 110 | // Create a new order and add all items 111 | await order.create(user, items, t); 112 | // Clear the users basket 113 | await Promise.all( 114 | Object.keys(basketItems).map(async (itemId) => { 115 | await basket.remove(itemId); 116 | }) 117 | ); 118 | }); 119 | 120 | req.session.messages.push({ 121 | type: "success", 122 | text: "Thank you for your business", 123 | }); 124 | 125 | return res.redirect("/basket"); 126 | } catch (err) { 127 | req.session.messages.push({ 128 | type: "danger", 129 | text: "There was an error finishing your order", 130 | }); 131 | console.error(err); 132 | return res.redirect("/basket"); 133 | } 134 | */ 135 | }); 136 | 137 | return router; 138 | }; 139 | -------------------------------------------------------------------------------- /shopper/server/routes/index.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const userRoute = require('./admin/user'); 3 | const itemRoute = require('./admin/item'); 4 | const orderRoute = require('./admin/orders'); 5 | const shopRoute = require('./shop'); 6 | const basketRoute = require('./basket'); 7 | 8 | module.exports = (config) => { 9 | const router = express.Router(); 10 | router.get('/', (req, res) => { 11 | res.render('index'); 12 | }); 13 | 14 | router.use('/shop', shopRoute(config)); 15 | router.use('/basket', basketRoute(config)); 16 | 17 | // Secure that route in real life applications 18 | router.use('/admin/user', userRoute(config)); 19 | router.use('/admin/item', itemRoute(config)); 20 | router.use('/admin/orders', orderRoute(config)); 21 | 22 | return router; 23 | }; 24 | -------------------------------------------------------------------------------- /shopper/server/routes/shop/index.js: -------------------------------------------------------------------------------- 1 | const express = require("express"); 2 | 3 | module.exports = () => { 4 | const router = express.Router(); 5 | 6 | router.get("/", async (req, res) => { 7 | return res.render("shop", {}); 8 | /* 9 | const items = await ItemService.getAll(); 10 | return res.render('shop', { items }); 11 | */ 12 | }); 13 | 14 | router.get("/tobasket/:itemId", async (req, res, next) => { 15 | return next("Not implemented"); 16 | 17 | /* 18 | if (!res.locals.currentUser) { 19 | req.session.messages.push({ 20 | type: "warning", 21 | text: "Please log in first", 22 | }); 23 | return res.redirect("/shop"); 24 | } 25 | 26 | try { 27 | const basket = new BasketService( 28 | config.redis.client, 29 | res.locals.currentUser.id 30 | ); 31 | await basket.add(req.params.itemId); 32 | req.session.messages.push({ 33 | type: "success", 34 | text: "The item was added to the basket", 35 | }); 36 | } catch (err) { 37 | req.session.messages.push({ 38 | type: "danger", 39 | text: "There was an error adding the item to the basket", 40 | }); 41 | console.error(err); 42 | } 43 | 44 | return res.redirect("/shop"); 45 | */ 46 | }); 47 | 48 | return router; 49 | }; 50 | -------------------------------------------------------------------------------- /shopper/server/services/BasketService.js: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/danielkhan/linkedin-node-databases/9a4f86890bcb10e1f63ee536fba711300443be3b/shopper/server/services/BasketService.js -------------------------------------------------------------------------------- /shopper/server/services/ItemService.js: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/danielkhan/linkedin-node-databases/9a4f86890bcb10e1f63ee536fba711300443be3b/shopper/server/services/ItemService.js -------------------------------------------------------------------------------- /shopper/server/services/OrderService.js: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/danielkhan/linkedin-node-databases/9a4f86890bcb10e1f63ee536fba711300443be3b/shopper/server/services/OrderService.js -------------------------------------------------------------------------------- /shopper/server/services/UserService.js: -------------------------------------------------------------------------------- 1 | const UserModel = require('../models/mongoose/User'); 2 | 3 | class UserService { 4 | static async getAll() { 5 | return UserModel.find({}).sort({ createdAt: -1 }); 6 | } 7 | 8 | static async getOne(userId) { 9 | return UserModel.findById(userId).exec(); 10 | } 11 | 12 | static async create(data) { 13 | const user = new UserModel(data); 14 | return user.save(); 15 | } 16 | 17 | static async update(userId, data) { 18 | // Fetch the user first 19 | const user = await UserModel.findById(userId); 20 | user.email = data.email; 21 | 22 | // Only set the password if it was modified 23 | if(data.password) { 24 | user.password = data.password; 25 | } 26 | 27 | return user.save(); 28 | } 29 | 30 | static async remove(userId) { 31 | return UserModel.deleteOne({ _id: userId }).exec(); 32 | } 33 | } 34 | 35 | module.exports = UserService; -------------------------------------------------------------------------------- /shopper/server/views/admin/item.pug: -------------------------------------------------------------------------------- 1 | extends ../layout 2 | 3 | block content 4 | .row 5 | .col-md-12 6 | h1 7 | | Manage Items 8 | 9 | if items && items.length 10 | table.table.table-bordered.table-striped 11 | thead 12 | th 13 | | SKU 14 | 15 | th 16 | | Name 17 | 18 | th 19 | | Price 20 | 21 | th 22 | | Action 23 | 24 | tbody 25 | each item in items 26 | tr 27 | td=item.sku 28 | td=item.name 29 | td=item.price 30 | td 31 | a.m-1.btn.btn-primary(href='/admin/item/' + item.id, role='button') 32 | | Edit item 33 | a.m-1.btn.btn-danger(href='/admin/item/delete/' + item.id, role='button') 34 | | Delete item 35 | else 36 | p.lead 37 | | No items found 38 | .row 39 | .col-md-12 40 | .card 41 | .card-header 42 | if(!item) 43 | | Create item 44 | else 45 | | Edit item 46 | 47 | .card-block 48 | form(method='POST', action='/admin/item' autocomplete='off') 49 | input(type='hidden', name='itemId', value=item ? item.id : '') 50 | 51 | .form-group 52 | label.form-control-label(for='sku') 53 | | SKU: 54 | input#sku.form-control(type='number', name='sku', autocomplete='off', value=item ? item.sku : '') 55 | 56 | .form-group 57 | label.form-control-label(for='name') 58 | | Name: 59 | input#name.form-control(type='text', name='name', autocomplete='off', value=item ? item.name : '') 60 | 61 | .form-group 62 | label.form-control-label(for='price') 63 | | Price: 64 | input#price.form-control(type='text', name='price', autocomplete='off', value=item ? item.price : '') 65 | 66 | button.m-1.btn.btn-primary(type='submit') 67 | | Submit 68 | 69 | a.m-1.btn.btn-secondary(href='/admin/item', role='button') 70 | | Reset -------------------------------------------------------------------------------- /shopper/server/views/admin/orders.pug: -------------------------------------------------------------------------------- 1 | extends ../layout 2 | 3 | block content 4 | .row 5 | .col-md-12 6 | h1 7 | | All Orders 8 | 9 | 10 | if orders && orders.length > 0 11 | table.table.table-bordered 12 | thead 13 | th 14 | | # 15 | 16 | 17 | th 18 | | Status 19 | 20 | th 21 | | Email 22 | 23 | th 24 | | Date 25 | 26 | th 27 | | Items 28 | 29 | th 30 | | Action 31 | tbody 32 | each order in orders 33 | tr 34 | td=order.id 35 | td=order.status 36 | td=order.email 37 | td=order.createdAt 38 | 39 | td 40 | table.table.table-bordered 41 | thead 42 | th 43 | | SKU 44 | 45 | th 46 | | Quantity 47 | 48 | th 49 | | Price 50 | 51 | tbody 52 | each item in order.OrderItems 53 | tr 54 | td=item.sku 55 | td=item.qty 56 | td=item.price 57 | 58 | td 59 | if order.status !== 'Shipped' 60 | a.m-1.btn.btn-danger(href='/admin/orders/setshipped/' + order.id, role='button') 61 | | Set Shipped 62 | else 63 | a.m-1.btn.btn-danger.disabled(href='/admin/orders/setshipped/' + order.id, role='button') 64 | | Order Shipped 65 | 66 | 67 | else 68 | p.lead 69 | | Nothing in here. -------------------------------------------------------------------------------- /shopper/server/views/admin/user.pug: -------------------------------------------------------------------------------- 1 | extends ../layout 2 | 3 | block content 4 | .row 5 | .col-md-12 6 | h1 7 | | Manage Users 8 | 9 | if users && users.length 10 | 11 | table.table.table-bordered.table-striped 12 | thead 13 | th 14 | | Email 15 | 16 | th 17 | | Action 18 | 19 | tbody 20 | each user in users 21 | tr 22 | td 23 | a(href='/admin/user/impersonate/' + user.id)=user.email 24 | td 25 | a.m-1.btn.btn-primary(href='/admin/user/' + user.id, role='button') 26 | | Edit User 27 | a.m-1.btn.btn-danger(href='/admin/user/delete/' + user.id, role='button') 28 | | Delete User 29 | else 30 | p.lead 31 | | No users found 32 | 33 | .row 34 | .col-md-12 35 | .card 36 | .card-header 37 | if(!user) 38 | | Create User 39 | else 40 | | Edit User 41 | 42 | .card-block 43 | form(method='POST', action='/admin/user' autocomplete='off') 44 | input(type='hidden', name='userId', value=user ? user.id : '') 45 | .form-group 46 | label.form-control-label(for='email') 47 | | Email Address: 48 | input#email.form-control(type='email', name='email', autocomplete='off', value=user ? user.email : '') 49 | 50 | .form-group 51 | label.form-control-label(for='password') 52 | | Password: 53 | input#password.form-control(type='password', name='password', autocomplete='off') 54 | 55 | button.m-1.btn.btn-primary(type='submit') 56 | | Submit 57 | 58 | a.m-1.btn.btn-secondary(href='/admin/user', role='button') 59 | | Reset -------------------------------------------------------------------------------- /shopper/server/views/basket.pug: -------------------------------------------------------------------------------- 1 | extends ./layout 2 | 3 | block content 4 | .row 5 | .col-md-12 6 | h1 7 | | Your Basket 8 | 9 | 10 | if items && items.length > 0 11 | table.table.table-bordered.table-striped 12 | thead 13 | th 14 | | SKU 15 | 16 | th 17 | | Name 18 | 19 | th 20 | | Price 21 | 22 | th 23 | | Quantity 24 | 25 | th 26 | | Action 27 | tbody 28 | each item in items 29 | tr 30 | td=item.sku 31 | td=item.name 32 | td=item.price 33 | td=item.quantity 34 | td 35 | a.m-1.btn.btn-danger(href='/basket/remove/' + item.id, role='button') 36 | | Remove from Basket 37 | 38 | 39 | tfoot 40 | tr 41 | td(colspan=5) 42 | a.m-1.btn.btn-primary.btn-block(href='/basket/buy', role='button') 43 | | Buy Now 44 | else 45 | p.lead 46 | | Nothing in here. -------------------------------------------------------------------------------- /shopper/server/views/index.pug: -------------------------------------------------------------------------------- 1 | extends ./layout 2 | 3 | block content 4 | .row 5 | .col-md-12 6 | h1 7 | | Welcome to #{applicationName} 8 | p.lead 9 | | We will use this sample project to explore different database systems with Node.js. -------------------------------------------------------------------------------- /shopper/server/views/layout.pug: -------------------------------------------------------------------------------- 1 | doctype html 2 | html(lang='en') 3 | head 4 | meta(charset='utf-8') 5 | meta(name='viewport', content='width=device-width, initial-scale=1, shrink-to-fit=no') 6 | meta(name='description', content='') 7 | meta(name='author', content='') 8 | 9 | title #{applicationName} 10 | // Bootstrap core CSS 11 | link(rel='stylesheet', href='//maxcdn.bootstrapcdn.com/bootstrap/4.0.0-alpha.6/css/bootstrap.min.css', integrity='sha384-rwoIResjU2yc3z8GV/NPeZWAv56rSmLldC3R/AZzGRnGxQQKnKkoFVhFQhNUwEyJ', crossorigin='anonymous') 12 | link(rel='stylesheet', href='/css/site.css') 13 | 14 | body 15 | nav.navbar.navbar-toggleable-md.navbar-inverse.bg-inverse.fixed-top 16 | button.navbar-toggler.navbar-toggler-right(type='button', data-toggle='collapse', data-target='#navbarsExampleDefault', aria-controls='navbarsExampleDefault', aria-expanded='false', aria-label='Toggle navigation') 17 | span.navbar-toggler-icon 18 | 19 | .container 20 | a.navbar-brand(href='#') #{applicationName} 21 | #navbarsExampleDefault.collapse.navbar-collapse 22 | ul.navbar-nav.mr-auto 23 | li.nav-item.active 24 | a.nav-link(href='/') 25 | | Home 26 | 27 | li.nav-item 28 | a.nav-link(href='/shop') Shop 29 | 30 | li.nav-item 31 | a.nav-link(href='/admin/user') Manage Users 32 | 33 | li.nav-item 34 | a.nav-link(href='/admin/item') Manage Items 35 | 36 | li.nav-item 37 | a.nav-link(href='/admin/orders') Manage Orders 38 | 39 | if currentUser 40 | ul.navbar-nav.ml-auto 41 | li.nav-item 42 | a.nav-link(href='#') 43 | | Logged in as #{currentUser.email} 44 | 45 | if basketCount 46 | li.nav-item 47 | a.nav-link(href='/basket') 48 | if(basketCount > 1) 49 | | #{basketCount} items in basket 50 | else 51 | | #{basketCount} item in basket 52 | 53 | .container 54 | 55 | while messages.length > 0 56 | - const message =messages.pop(); 57 | .alert(class='alert-dismissible fade show alert-' + message.type) 58 | button.close(type='button', data-dismiss='alert', aria-label='Close') 59 | span(aria-hidden='true') × 60 | | #{message.text} 61 | 62 | block content 63 | 64 | 65 | script(src='//code.jquery.com/jquery-3.1.1.slim.min.js', integrity='sha384-A7FZj7v+d/sdmMqp/nOQwliLvUsJfDHW+k9Omg/a/EheAdgtzNs3hpfag6Ed950n', crossorigin='anonymous') 66 | script(src='//cdnjs.cloudflare.com/ajax/libs/tether/1.4.0/js/tether.min.js', integrity='sha384-DztdAPBWPRXSA/3eYEEUWrWCy7G5KFbe8fFjk5JAIxUYHKkDx6Qin1DkWx51bBrb', crossorigin='anonymous') 67 | script(src='//maxcdn.bootstrapcdn.com/bootstrap/4.0.0-alpha.6/js/bootstrap.min.js', integrity='sha384-vBWWzlZJ8ea9aCX4pEW3rVHjgjt7zpkNpZk+02D9phzyeVkE+jo0ieGizqPLForn', crossorigin='anonymous') 68 | -------------------------------------------------------------------------------- /shopper/server/views/shop.pug: -------------------------------------------------------------------------------- 1 | extends ./layout 2 | 3 | block content 4 | .row 5 | .col-md-12 6 | h1 7 | | A very basic Shop 8 | 9 | if items && items.length 10 | table.table.table-bordered.table-striped 11 | thead 12 | th 13 | | SKU 14 | 15 | th 16 | | Name 17 | 18 | th 19 | | Price 20 | 21 | th 22 | | Action 23 | tbody 24 | each item in items 25 | tr 26 | td=item.sku 27 | td=item.name 28 | td=item.price 29 | td 30 | a.m-1.btn.btn-primary(href='/shop/tobasket/' + item.id, role='button') 31 | | Add to Basket 32 | else 33 | p.lead 34 | | Nothing to see here yet --------------------------------------------------------------------------------