├── .gitignore ├── README.md ├── package-lock.json ├── package.json └── src ├── index.js ├── models ├── cartItem.js ├── order.js ├── orderItem.js ├── product.js └── user.js ├── resolvers ├── index.js ├── mutation.js └── query.js ├── schema ├── schema.graphql └── typeDefs.js ├── server.js └── utils ├── getUser.js ├── omiseUtils.js ├── passport.js └── socialProvidersAuth.js /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | .env -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # NodeJs-Apollo Server-Mongodb Tutorial Project 2 | 3 | Backend part of a full stack ecommerce web app. This backend was built in NodeJs, Graphql, and Mongodb as a database, specifically this app was built in NodeJs, Apollo-Server-Express, Mongoose, and Mongodb atlas. Its frontend is in separated repository (https://github.com/autsada/nextjs-apollo/tree/master). 4 | 5 | ### Tutorial link on youtube (in Thai) 6 | 7 | https://www.youtube.com/playlist?list=PLCT_w0Fqe_z7o3fVVqlU9NBeByrwjwfDd 8 | 9 | ### Prerequisites 10 | 11 | NodeJs v8+ (https://nodejs.org/en/) 12 | 13 | ### Installing 14 | 15 | 1. Clone the project 16 | ``` 17 | git clone https://github.com/autsada/nodejs-apolloserver-mongodb 18 | ``` 19 | 2. Install dependencies 20 | ``` 21 | npm install 22 | ``` 23 | 24 | ### Note: 25 | 26 | In order to have this app up an running for all related parts, you need to have Mongodb Atlas account (https://www.mongodb.com/cloud/atlas), email service provider - sendgrid account (https://sendgrid.com/), and omise account (https://www.omise.co/) or you can use other payment gateways such as Stripe or Paypal 27 | 28 | In .env file 29 | 30 | ``` 31 | DB_USER= 32 | DB_PASSWORD= 33 | DB_NAME= 34 | PORT=4444 35 | SECRET= 36 | EMAIL_API_KEY= 37 | OMISE_PUBLIC_KEY= 38 | OMISE_SECRET_KEY= 39 | ``` 40 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "graphql-basic", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@apollo/protobufjs": { 8 | "version": "1.0.3", 9 | "resolved": "https://registry.npmjs.org/@apollo/protobufjs/-/protobufjs-1.0.3.tgz", 10 | "integrity": "sha512-gqeT810Ect9WIqsrgfUvr+ljSB5m1PyBae9HGdrRyQ3HjHjTcjVvxpsMYXlUk4rUHnrfUqyoGvLSy2yLlRGEOw==", 11 | "requires": { 12 | "@protobufjs/aspromise": "^1.1.2", 13 | "@protobufjs/base64": "^1.1.2", 14 | "@protobufjs/codegen": "^2.0.4", 15 | "@protobufjs/eventemitter": "^1.1.0", 16 | "@protobufjs/fetch": "^1.1.0", 17 | "@protobufjs/float": "^1.0.2", 18 | "@protobufjs/inquire": "^1.1.0", 19 | "@protobufjs/path": "^1.1.2", 20 | "@protobufjs/pool": "^1.1.0", 21 | "@protobufjs/utf8": "^1.1.0", 22 | "@types/long": "^4.0.0", 23 | "@types/node": "^10.1.0", 24 | "long": "^4.0.0" 25 | }, 26 | "dependencies": { 27 | "@types/node": { 28 | "version": "10.17.9", 29 | "resolved": "https://registry.npmjs.org/@types/node/-/node-10.17.9.tgz", 30 | "integrity": "sha512-+6VygF9LbG7Gaqeog2G7u1+RUcmo0q1rI+2ZxdIg2fAUngk5Vz9fOCHXdloNUOHEPd1EuuOpL5O0CdgN9Fx5UQ==" 31 | } 32 | } 33 | }, 34 | "@apollographql/apollo-tools": { 35 | "version": "0.4.2", 36 | "resolved": "https://registry.npmjs.org/@apollographql/apollo-tools/-/apollo-tools-0.4.2.tgz", 37 | "integrity": "sha512-/kTaguTNSowXR/zWU4hjeL41yAdEbQO05f882c6cRIrVE7xIgJcBNEcYz2kzi94eaUbE2YY3SSxDJ6vPeV07OQ==", 38 | "requires": { 39 | "apollo-env": "^0.6.0" 40 | } 41 | }, 42 | "@apollographql/graphql-playground-html": { 43 | "version": "1.6.24", 44 | "resolved": "https://registry.npmjs.org/@apollographql/graphql-playground-html/-/graphql-playground-html-1.6.24.tgz", 45 | "integrity": "sha512-8GqG48m1XqyXh4mIZrtB5xOhUwSsh1WsrrsaZQOEYYql3YN9DEu9OOSg0ILzXHZo/h2Q74777YE4YzlArQzQEQ==" 46 | }, 47 | "@protobufjs/aspromise": { 48 | "version": "1.1.2", 49 | "resolved": "https://registry.npmjs.org/@protobufjs/aspromise/-/aspromise-1.1.2.tgz", 50 | "integrity": "sha1-m4sMxmPWaafY9vXQiToU00jzD78=" 51 | }, 52 | "@protobufjs/base64": { 53 | "version": "1.1.2", 54 | "resolved": "https://registry.npmjs.org/@protobufjs/base64/-/base64-1.1.2.tgz", 55 | "integrity": "sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg==" 56 | }, 57 | "@protobufjs/codegen": { 58 | "version": "2.0.4", 59 | "resolved": "https://registry.npmjs.org/@protobufjs/codegen/-/codegen-2.0.4.tgz", 60 | "integrity": "sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg==" 61 | }, 62 | "@protobufjs/eventemitter": { 63 | "version": "1.1.0", 64 | "resolved": "https://registry.npmjs.org/@protobufjs/eventemitter/-/eventemitter-1.1.0.tgz", 65 | "integrity": "sha1-NVy8mLr61ZePntCV85diHx0Ga3A=" 66 | }, 67 | "@protobufjs/fetch": { 68 | "version": "1.1.0", 69 | "resolved": "https://registry.npmjs.org/@protobufjs/fetch/-/fetch-1.1.0.tgz", 70 | "integrity": "sha1-upn7WYYUr2VwDBYZ/wbUVLDYTEU=", 71 | "requires": { 72 | "@protobufjs/aspromise": "^1.1.1", 73 | "@protobufjs/inquire": "^1.1.0" 74 | } 75 | }, 76 | "@protobufjs/float": { 77 | "version": "1.0.2", 78 | "resolved": "https://registry.npmjs.org/@protobufjs/float/-/float-1.0.2.tgz", 79 | "integrity": "sha1-Xp4avctz/Ap8uLKR33jIy9l7h9E=" 80 | }, 81 | "@protobufjs/inquire": { 82 | "version": "1.1.0", 83 | "resolved": "https://registry.npmjs.org/@protobufjs/inquire/-/inquire-1.1.0.tgz", 84 | "integrity": "sha1-/yAOPnzyQp4tyvwRQIKOjMY48Ik=" 85 | }, 86 | "@protobufjs/path": { 87 | "version": "1.1.2", 88 | "resolved": "https://registry.npmjs.org/@protobufjs/path/-/path-1.1.2.tgz", 89 | "integrity": "sha1-bMKyDFya1q0NzP0hynZz2Nf79o0=" 90 | }, 91 | "@protobufjs/pool": { 92 | "version": "1.1.0", 93 | "resolved": "https://registry.npmjs.org/@protobufjs/pool/-/pool-1.1.0.tgz", 94 | "integrity": "sha1-Cf0V8tbTq/qbZbw2ZQbWrXhG/1Q=" 95 | }, 96 | "@protobufjs/utf8": { 97 | "version": "1.1.0", 98 | "resolved": "https://registry.npmjs.org/@protobufjs/utf8/-/utf8-1.1.0.tgz", 99 | "integrity": "sha1-p3c2C1s5oaLlEG+OhY8v0tBgxXA=" 100 | }, 101 | "@sendgrid/client": { 102 | "version": "6.5.2", 103 | "resolved": "https://registry.npmjs.org/@sendgrid/client/-/client-6.5.2.tgz", 104 | "integrity": "sha512-P3NLYffFoJXaOzKcc2Ag0KGFxyuJliLKx8W3JQbv94ATu2gfmcePWGz8qPR+qbA7zyl2AIocU4y+Kbdl8sUs3g==", 105 | "requires": { 106 | "@sendgrid/helpers": "^6.5.1", 107 | "@types/request": "^2.48.4", 108 | "request": "^2.88.0" 109 | } 110 | }, 111 | "@sendgrid/helpers": { 112 | "version": "6.5.1", 113 | "resolved": "https://registry.npmjs.org/@sendgrid/helpers/-/helpers-6.5.1.tgz", 114 | "integrity": "sha512-Bd05zxnKRAKtYCXMjDSlXKmlX/RceWMIHYVwU+auMFUk+C8Mx755hGhBl8IHLz0kYL03dF8cSQA0iXlnoMGIpQ==", 115 | "requires": { 116 | "chalk": "^2.0.1", 117 | "deepmerge": "^2.1.1" 118 | } 119 | }, 120 | "@sendgrid/mail": { 121 | "version": "6.5.2", 122 | "resolved": "https://registry.npmjs.org/@sendgrid/mail/-/mail-6.5.2.tgz", 123 | "integrity": "sha512-JLg1Qo+ghzsk7MWNZabqjA6RxdfSkRTaBNoiI2A4Fryd6IphSdO+zLF+kplXADWnmtZZFNVtyvxUzy0kad8Ytg==", 124 | "requires": { 125 | "@sendgrid/client": "^6.5.2", 126 | "@sendgrid/helpers": "^6.5.1" 127 | } 128 | }, 129 | "@types/accepts": { 130 | "version": "1.3.5", 131 | "resolved": "https://registry.npmjs.org/@types/accepts/-/accepts-1.3.5.tgz", 132 | "integrity": "sha512-jOdnI/3qTpHABjM5cx1Hc0sKsPoYCp+DP/GJRGtDlPd7fiV9oXGGIcjW/ZOxLIvjGz8MA+uMZI9metHlgqbgwQ==", 133 | "requires": { 134 | "@types/node": "*" 135 | } 136 | }, 137 | "@types/bluebird": { 138 | "version": "3.5.30", 139 | "resolved": "https://registry.npmjs.org/@types/bluebird/-/bluebird-3.5.30.tgz", 140 | "integrity": "sha512-8LhzvcjIoqoi1TghEkRMkbbmM+jhHnBokPGkJWjclMK+Ks0MxEBow3/p2/iFTZ+OIbJHQDSfpgdZEb+af3gfVw==" 141 | }, 142 | "@types/body-parser": { 143 | "version": "1.17.1", 144 | "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.17.1.tgz", 145 | "integrity": "sha512-RoX2EZjMiFMjZh9lmYrwgoP9RTpAjSHiJxdp4oidAQVO02T7HER3xj9UKue5534ULWeqVEkujhWcyvUce+d68w==", 146 | "requires": { 147 | "@types/connect": "*", 148 | "@types/node": "*" 149 | } 150 | }, 151 | "@types/caseless": { 152 | "version": "0.12.2", 153 | "resolved": "https://registry.npmjs.org/@types/caseless/-/caseless-0.12.2.tgz", 154 | "integrity": "sha512-6ckxMjBBD8URvjB6J3NcnuAn5Pkl7t3TizAg+xdlzzQGSPSmBcXf8KoIH0ua/i+tio+ZRUHEXp0HEmvaR4kt0w==" 155 | }, 156 | "@types/connect": { 157 | "version": "3.4.32", 158 | "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.32.tgz", 159 | "integrity": "sha512-4r8qa0quOvh7lGD0pre62CAb1oni1OO6ecJLGCezTmhQ8Fz50Arx9RUszryR8KlgK6avuSXvviL6yWyViQABOg==", 160 | "requires": { 161 | "@types/node": "*" 162 | } 163 | }, 164 | "@types/cookies": { 165 | "version": "0.7.4", 166 | "resolved": "https://registry.npmjs.org/@types/cookies/-/cookies-0.7.4.tgz", 167 | "integrity": "sha512-oTGtMzZZAVuEjTwCjIh8T8FrC8n/uwy+PG0yTvQcdZ7etoel7C7/3MSd7qrukENTgQtotG7gvBlBojuVs7X5rw==", 168 | "requires": { 169 | "@types/connect": "*", 170 | "@types/express": "*", 171 | "@types/keygrip": "*", 172 | "@types/node": "*" 173 | } 174 | }, 175 | "@types/cors": { 176 | "version": "2.8.6", 177 | "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.6.tgz", 178 | "integrity": "sha512-invOmosX0DqbpA+cE2yoHGUlF/blyf7nB0OGYBBiH27crcVm5NmFaZkLP4Ta1hGaesckCi5lVLlydNJCxkTOSg==", 179 | "requires": { 180 | "@types/express": "*" 181 | } 182 | }, 183 | "@types/express": { 184 | "version": "4.17.1", 185 | "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.1.tgz", 186 | "integrity": "sha512-VfH/XCP0QbQk5B5puLqTLEeFgR8lfCJHZJKkInZ9mkYd+u8byX0kztXEQxEk4wZXJs8HI+7km2ALXjn4YKcX9w==", 187 | "requires": { 188 | "@types/body-parser": "*", 189 | "@types/express-serve-static-core": "*", 190 | "@types/serve-static": "*" 191 | } 192 | }, 193 | "@types/express-serve-static-core": { 194 | "version": "4.17.0", 195 | "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.0.tgz", 196 | "integrity": "sha512-Xnub7w57uvcBqFdIGoRg1KhNOeEj0vB6ykUM7uFWyxvbdE89GFyqgmUcanAriMr4YOxNFZBAWkfcWIb4WBPt3g==", 197 | "requires": { 198 | "@types/node": "*", 199 | "@types/range-parser": "*" 200 | } 201 | }, 202 | "@types/fs-capacitor": { 203 | "version": "2.0.0", 204 | "resolved": "https://registry.npmjs.org/@types/fs-capacitor/-/fs-capacitor-2.0.0.tgz", 205 | "integrity": "sha512-FKVPOCFbhCvZxpVAMhdBdTfVfXUpsh15wFHgqOKxh9N9vzWZVuWCSijZ5T4U34XYNnuj2oduh6xcs1i+LPI+BQ==", 206 | "requires": { 207 | "@types/node": "*" 208 | } 209 | }, 210 | "@types/graphql-upload": { 211 | "version": "8.0.3", 212 | "resolved": "https://registry.npmjs.org/@types/graphql-upload/-/graphql-upload-8.0.3.tgz", 213 | "integrity": "sha512-hmLg9pCU/GmxBscg8GCr1vmSoEmbItNNxdD5YH2TJkXm//8atjwuprB+xJBK714JG1dkxbbhp5RHX+Pz1KsCMA==", 214 | "requires": { 215 | "@types/express": "*", 216 | "@types/fs-capacitor": "*", 217 | "@types/koa": "*", 218 | "graphql": "^14.5.3" 219 | } 220 | }, 221 | "@types/http-assert": { 222 | "version": "1.5.1", 223 | "resolved": "https://registry.npmjs.org/@types/http-assert/-/http-assert-1.5.1.tgz", 224 | "integrity": "sha512-PGAK759pxyfXE78NbKxyfRcWYA/KwW17X290cNev/qAsn9eQIxkH4shoNBafH37wewhDG/0p1cHPbK6+SzZjWQ==" 225 | }, 226 | "@types/keygrip": { 227 | "version": "1.0.1", 228 | "resolved": "https://registry.npmjs.org/@types/keygrip/-/keygrip-1.0.1.tgz", 229 | "integrity": "sha1-/1QEYtL7TQqIRBzq8n0oewHD2Hg=" 230 | }, 231 | "@types/koa": { 232 | "version": "2.11.0", 233 | "resolved": "https://registry.npmjs.org/@types/koa/-/koa-2.11.0.tgz", 234 | "integrity": "sha512-Hgx/1/rVlJvqYBrdeCsS7PDiR2qbxlMt1RnmNWD4Uxi5FF9nwkYqIldo7urjc+dfNpk+2NRGcnAYd4L5xEhCcQ==", 235 | "requires": { 236 | "@types/accepts": "*", 237 | "@types/cookies": "*", 238 | "@types/http-assert": "*", 239 | "@types/keygrip": "*", 240 | "@types/koa-compose": "*", 241 | "@types/node": "*" 242 | } 243 | }, 244 | "@types/koa-compose": { 245 | "version": "3.2.5", 246 | "resolved": "https://registry.npmjs.org/@types/koa-compose/-/koa-compose-3.2.5.tgz", 247 | "integrity": "sha512-B8nG/OoE1ORZqCkBVsup/AKcvjdgoHnfi4pZMn5UwAPCbhk/96xyv284eBYW8JlQbQ7zDmnpFr68I/40mFoIBQ==", 248 | "requires": { 249 | "@types/koa": "*" 250 | } 251 | }, 252 | "@types/long": { 253 | "version": "4.0.0", 254 | "resolved": "https://registry.npmjs.org/@types/long/-/long-4.0.0.tgz", 255 | "integrity": "sha512-1w52Nyx4Gq47uuu0EVcsHBxZFJgurQ+rTKS3qMHxR1GY2T8c2AJYd6vZoZ9q1rupaDjU0yT+Jc2XTyXkjeMA+Q==" 256 | }, 257 | "@types/mime": { 258 | "version": "2.0.1", 259 | "resolved": "https://registry.npmjs.org/@types/mime/-/mime-2.0.1.tgz", 260 | "integrity": "sha512-FwI9gX75FgVBJ7ywgnq/P7tw+/o1GUbtP0KzbtusLigAOgIgNISRK0ZPl4qertvXSIE8YbsVJueQ90cDt9YYyw==" 261 | }, 262 | "@types/node": { 263 | "version": "12.12.17", 264 | "resolved": "https://registry.npmjs.org/@types/node/-/node-12.12.17.tgz", 265 | "integrity": "sha512-Is+l3mcHvs47sKy+afn2O1rV4ldZFU7W8101cNlOd+MRbjM4Onida8jSZnJdTe/0Pcf25g9BNIUsuugmE6puHA==" 266 | }, 267 | "@types/range-parser": { 268 | "version": "1.2.3", 269 | "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.3.tgz", 270 | "integrity": "sha512-ewFXqrQHlFsgc09MK5jP5iR7vumV/BYayNC6PgJO2LPe8vrnNFyjQjSppfEngITi0qvfKtzFvgKymGheFM9UOA==" 271 | }, 272 | "@types/request": { 273 | "version": "2.48.4", 274 | "resolved": "https://registry.npmjs.org/@types/request/-/request-2.48.4.tgz", 275 | "integrity": "sha512-W1t1MTKYR8PxICH+A4HgEIPuAC3sbljoEVfyZbeFJJDbr30guDspJri2XOaM2E+Un7ZjrihaDi7cf6fPa2tbgw==", 276 | "requires": { 277 | "@types/caseless": "*", 278 | "@types/node": "*", 279 | "@types/tough-cookie": "*", 280 | "form-data": "^2.5.0" 281 | } 282 | }, 283 | "@types/serve-static": { 284 | "version": "1.13.3", 285 | "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.13.3.tgz", 286 | "integrity": "sha512-oprSwp094zOglVrXdlo/4bAHtKTAxX6VT8FOZlBKrmyLbNvE1zxZyJ6yikMVtHIvwP45+ZQGJn+FdXGKTozq0g==", 287 | "requires": { 288 | "@types/express-serve-static-core": "*", 289 | "@types/mime": "*" 290 | } 291 | }, 292 | "@types/tough-cookie": { 293 | "version": "2.3.6", 294 | "resolved": "https://registry.npmjs.org/@types/tough-cookie/-/tough-cookie-2.3.6.tgz", 295 | "integrity": "sha512-wHNBMnkoEBiRAd3s8KTKwIuO9biFtTf0LehITzBhSco+HQI0xkXZbLOD55SW3Aqw3oUkHstkm5SPv58yaAdFPQ==" 296 | }, 297 | "@types/ws": { 298 | "version": "6.0.4", 299 | "resolved": "https://registry.npmjs.org/@types/ws/-/ws-6.0.4.tgz", 300 | "integrity": "sha512-PpPrX7SZW9re6+Ha8ojZG4Se8AZXgf0GK6zmfqEuCsY49LFDNXO3SByp44X3dFEqtB73lkCDAdUazhAjVPiNwg==", 301 | "requires": { 302 | "@types/node": "*" 303 | } 304 | }, 305 | "@wry/equality": { 306 | "version": "0.1.9", 307 | "resolved": "https://registry.npmjs.org/@wry/equality/-/equality-0.1.9.tgz", 308 | "integrity": "sha512-mB6ceGjpMGz1ZTza8HYnrPGos2mC6So4NhS1PtZ8s4Qt0K7fBiIGhpSxUbQmhwcSWE3no+bYxmI2OL6KuXYmoQ==", 309 | "requires": { 310 | "tslib": "^1.9.3" 311 | } 312 | }, 313 | "abbrev": { 314 | "version": "1.1.1", 315 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 316 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==" 317 | }, 318 | "accepts": { 319 | "version": "1.3.7", 320 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 321 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 322 | "requires": { 323 | "mime-types": "~2.1.24", 324 | "negotiator": "0.6.2" 325 | } 326 | }, 327 | "agent-base": { 328 | "version": "4.3.0", 329 | "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.3.0.tgz", 330 | "integrity": "sha512-salcGninV0nPrwpGNn4VTXBb1SOuXQBiqbrNXoeizJsHrsL6ERFM2Ne3JUSBWRE6aeNJI2ROP/WEEIDUiDe3cg==", 331 | "requires": { 332 | "es6-promisify": "^5.0.0" 333 | } 334 | }, 335 | "ajv": { 336 | "version": "6.11.0", 337 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.11.0.tgz", 338 | "integrity": "sha512-nCprB/0syFYy9fVYU1ox1l2KN8S9I+tziH8D4zdZuLT3N6RMlGSGt5FSTpAiHB/Whv8Qs1cWHma1aMKZyaHRKA==", 339 | "requires": { 340 | "fast-deep-equal": "^3.1.1", 341 | "fast-json-stable-stringify": "^2.0.0", 342 | "json-schema-traverse": "^0.4.1", 343 | "uri-js": "^4.2.2" 344 | } 345 | }, 346 | "ansi-align": { 347 | "version": "2.0.0", 348 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-2.0.0.tgz", 349 | "integrity": "sha1-w2rsy6VjuJzrVW82kPCx2eNUf38=", 350 | "requires": { 351 | "string-width": "^2.0.0" 352 | } 353 | }, 354 | "ansi-regex": { 355 | "version": "3.0.0", 356 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", 357 | "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=" 358 | }, 359 | "ansi-styles": { 360 | "version": "3.2.1", 361 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 362 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 363 | "requires": { 364 | "color-convert": "^1.9.0" 365 | } 366 | }, 367 | "anymatch": { 368 | "version": "3.1.1", 369 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", 370 | "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", 371 | "requires": { 372 | "normalize-path": "^3.0.0", 373 | "picomatch": "^2.0.4" 374 | } 375 | }, 376 | "apollo-cache-control": { 377 | "version": "0.8.8", 378 | "resolved": "https://registry.npmjs.org/apollo-cache-control/-/apollo-cache-control-0.8.8.tgz", 379 | "integrity": "sha512-hpIJg3Tmb6quA111lrVO+d3qcyYRlJ8JqbeQdcgwLT3fb2VQzk21SrBZYl2oMM4ZqSOWCZWg4/Cn9ARYqdWjKA==", 380 | "requires": { 381 | "apollo-server-env": "^2.4.3", 382 | "graphql-extensions": "^0.10.7" 383 | } 384 | }, 385 | "apollo-datasource": { 386 | "version": "0.6.3", 387 | "resolved": "https://registry.npmjs.org/apollo-datasource/-/apollo-datasource-0.6.3.tgz", 388 | "integrity": "sha512-gRYyFVpJgHE2hhS+VxMeOerxXQ/QYxWG7T6QddfugJWYAG9DRCl65e2b7txcGq2NP3r+O1iCm4GNwhRBDJbd8A==", 389 | "requires": { 390 | "apollo-server-caching": "^0.5.0", 391 | "apollo-server-env": "^2.4.3" 392 | } 393 | }, 394 | "apollo-engine-reporting": { 395 | "version": "1.4.11", 396 | "resolved": "https://registry.npmjs.org/apollo-engine-reporting/-/apollo-engine-reporting-1.4.11.tgz", 397 | "integrity": "sha512-7ZkbOGvPfWppN8+1KHzyHPrJTMOmrMUy38unao2c9TTToOAnEvx2MtUTo6mr3aw/g8UQYUf0x2Cq+K2YSlUTPw==", 398 | "requires": { 399 | "apollo-engine-reporting-protobuf": "^0.4.4", 400 | "apollo-graphql": "^0.3.4", 401 | "apollo-server-caching": "^0.5.0", 402 | "apollo-server-env": "^2.4.3", 403 | "apollo-server-types": "^0.2.8", 404 | "async-retry": "^1.2.1", 405 | "graphql-extensions": "^0.10.7" 406 | } 407 | }, 408 | "apollo-engine-reporting-protobuf": { 409 | "version": "0.4.4", 410 | "resolved": "https://registry.npmjs.org/apollo-engine-reporting-protobuf/-/apollo-engine-reporting-protobuf-0.4.4.tgz", 411 | "integrity": "sha512-SGrIkUR7Q/VjU8YG98xcvo340C4DaNUhg/TXOtGsMlfiJDzHwVau/Bv6zifAzBafp2lj0XND6Daj5kyT/eSI/w==", 412 | "requires": { 413 | "@apollo/protobufjs": "^1.0.3" 414 | } 415 | }, 416 | "apollo-env": { 417 | "version": "0.6.0", 418 | "resolved": "https://registry.npmjs.org/apollo-env/-/apollo-env-0.6.0.tgz", 419 | "integrity": "sha512-DttHOpLISRej8STjbXjQCXq3YeE2pATaC4UEd2YE7TjjYhQmp9yxohlkHfSR78BvPzczhyDs6WQQEzasHv0M0A==", 420 | "requires": { 421 | "core-js": "^3.0.1", 422 | "node-fetch": "^2.2.0", 423 | "sha.js": "^2.4.11" 424 | } 425 | }, 426 | "apollo-graphql": { 427 | "version": "0.3.6", 428 | "resolved": "https://registry.npmjs.org/apollo-graphql/-/apollo-graphql-0.3.6.tgz", 429 | "integrity": "sha512-PUBfW6t20U4CgPODTZB+3Z1Z+qhca8SNEHMPreiw+qEjXwEJF7SZItOIAs93HO0mA2K7eiZjCtZQZknaaQRZNA==", 430 | "requires": { 431 | "apollo-env": "^0.6.0", 432 | "lodash.sortby": "^4.7.0" 433 | } 434 | }, 435 | "apollo-link": { 436 | "version": "1.2.13", 437 | "resolved": "https://registry.npmjs.org/apollo-link/-/apollo-link-1.2.13.tgz", 438 | "integrity": "sha512-+iBMcYeevMm1JpYgwDEIDt/y0BB7VWyvlm/7x+TIPNLHCTCMgcEgDuW5kH86iQZWo0I7mNwQiTOz+/3ShPFmBw==", 439 | "requires": { 440 | "apollo-utilities": "^1.3.0", 441 | "ts-invariant": "^0.4.0", 442 | "tslib": "^1.9.3", 443 | "zen-observable-ts": "^0.8.20" 444 | } 445 | }, 446 | "apollo-server-caching": { 447 | "version": "0.5.0", 448 | "resolved": "https://registry.npmjs.org/apollo-server-caching/-/apollo-server-caching-0.5.0.tgz", 449 | "integrity": "sha512-l7ieNCGxUaUAVAAp600HjbUJxVaxjJygtPV0tPTe1Q3HkPy6LEWoY6mNHV7T268g1hxtPTxcdRu7WLsJrg7ufw==", 450 | "requires": { 451 | "lru-cache": "^5.0.0" 452 | } 453 | }, 454 | "apollo-server-core": { 455 | "version": "2.9.13", 456 | "resolved": "https://registry.npmjs.org/apollo-server-core/-/apollo-server-core-2.9.13.tgz", 457 | "integrity": "sha512-iXTGNCtouB0Xe37ySovuZO69NBYOByJlZfUc87gj0pdcz0WbdfUp7qUtNzy3onp63Zo60TFkHWhGNcBJYFluzw==", 458 | "requires": { 459 | "@apollographql/apollo-tools": "^0.4.0", 460 | "@apollographql/graphql-playground-html": "1.6.24", 461 | "@types/graphql-upload": "^8.0.0", 462 | "@types/ws": "^6.0.0", 463 | "apollo-cache-control": "^0.8.8", 464 | "apollo-datasource": "^0.6.3", 465 | "apollo-engine-reporting": "^1.4.11", 466 | "apollo-server-caching": "^0.5.0", 467 | "apollo-server-env": "^2.4.3", 468 | "apollo-server-errors": "^2.3.4", 469 | "apollo-server-plugin-base": "^0.6.8", 470 | "apollo-server-types": "^0.2.8", 471 | "apollo-tracing": "^0.8.8", 472 | "fast-json-stable-stringify": "^2.0.0", 473 | "graphql-extensions": "^0.10.7", 474 | "graphql-tag": "^2.9.2", 475 | "graphql-tools": "^4.0.0", 476 | "graphql-upload": "^8.0.2", 477 | "sha.js": "^2.4.11", 478 | "subscriptions-transport-ws": "^0.9.11", 479 | "ws": "^6.0.0" 480 | } 481 | }, 482 | "apollo-server-env": { 483 | "version": "2.4.3", 484 | "resolved": "https://registry.npmjs.org/apollo-server-env/-/apollo-server-env-2.4.3.tgz", 485 | "integrity": "sha512-23R5Xo9OMYX0iyTu2/qT0EUb+AULCBriA9w8HDfMoChB8M+lFClqUkYtaTTHDfp6eoARLW8kDBhPOBavsvKAjA==", 486 | "requires": { 487 | "node-fetch": "^2.1.2", 488 | "util.promisify": "^1.0.0" 489 | } 490 | }, 491 | "apollo-server-errors": { 492 | "version": "2.3.4", 493 | "resolved": "https://registry.npmjs.org/apollo-server-errors/-/apollo-server-errors-2.3.4.tgz", 494 | "integrity": "sha512-Y0PKQvkrb2Kd18d1NPlHdSqmlr8TgqJ7JQcNIfhNDgdb45CnqZlxL1abuIRhr8tiw8OhVOcFxz2KyglBi8TKdA==" 495 | }, 496 | "apollo-server-express": { 497 | "version": "2.9.13", 498 | "resolved": "https://registry.npmjs.org/apollo-server-express/-/apollo-server-express-2.9.13.tgz", 499 | "integrity": "sha512-M306e07dpZ8YpZx4VBYa0FWlt+wopj4Bwn0Iy1iJ6VjaRyGx2HCUJvLpHZ+D0TIXtQ2nX3DTYeOouVaDDwJeqQ==", 500 | "requires": { 501 | "@apollographql/graphql-playground-html": "1.6.24", 502 | "@types/accepts": "^1.3.5", 503 | "@types/body-parser": "1.17.1", 504 | "@types/cors": "^2.8.4", 505 | "@types/express": "4.17.1", 506 | "accepts": "^1.3.5", 507 | "apollo-server-core": "^2.9.13", 508 | "apollo-server-types": "^0.2.8", 509 | "body-parser": "^1.18.3", 510 | "cors": "^2.8.4", 511 | "express": "^4.17.1", 512 | "graphql-subscriptions": "^1.0.0", 513 | "graphql-tools": "^4.0.0", 514 | "parseurl": "^1.3.2", 515 | "subscriptions-transport-ws": "^0.9.16", 516 | "type-is": "^1.6.16" 517 | } 518 | }, 519 | "apollo-server-plugin-base": { 520 | "version": "0.6.8", 521 | "resolved": "https://registry.npmjs.org/apollo-server-plugin-base/-/apollo-server-plugin-base-0.6.8.tgz", 522 | "integrity": "sha512-0pKCjcg9gHBK8qlb280+N0jl99meixQtxXnMJFyIfD+45OpKQ+WolHIbO0oZgNEt7r/lNWwH8v3l5yYm1ghz1A==", 523 | "requires": { 524 | "apollo-server-types": "^0.2.8" 525 | } 526 | }, 527 | "apollo-server-types": { 528 | "version": "0.2.8", 529 | "resolved": "https://registry.npmjs.org/apollo-server-types/-/apollo-server-types-0.2.8.tgz", 530 | "integrity": "sha512-5OclxkAqjhuO75tTNHpSO/+doJZ+VlRtTefnrPJdK/uwVew9U/VUCWkYdryZWwEyVe1nvQ/4E7RYR4tGb8l8wA==", 531 | "requires": { 532 | "apollo-engine-reporting-protobuf": "^0.4.4", 533 | "apollo-server-caching": "^0.5.0", 534 | "apollo-server-env": "^2.4.3" 535 | } 536 | }, 537 | "apollo-tracing": { 538 | "version": "0.8.8", 539 | "resolved": "https://registry.npmjs.org/apollo-tracing/-/apollo-tracing-0.8.8.tgz", 540 | "integrity": "sha512-aIwT2PsH7VZZPaNrIoSjzLKMlG644d2Uf+GYcoMd3X6UEyg1sXdWqkKfCeoS6ChJKH2khO7MXAvOZC03UnCumQ==", 541 | "requires": { 542 | "apollo-server-env": "^2.4.3", 543 | "graphql-extensions": "^0.10.7" 544 | } 545 | }, 546 | "apollo-utilities": { 547 | "version": "1.3.2", 548 | "resolved": "https://registry.npmjs.org/apollo-utilities/-/apollo-utilities-1.3.2.tgz", 549 | "integrity": "sha512-JWNHj8XChz7S4OZghV6yc9FNnzEXj285QYp/nLNh943iObycI5GTDO3NGR9Dth12LRrSFMeDOConPfPln+WGfg==", 550 | "requires": { 551 | "@wry/equality": "^0.1.2", 552 | "fast-json-stable-stringify": "^2.0.0", 553 | "ts-invariant": "^0.4.0", 554 | "tslib": "^1.9.3" 555 | } 556 | }, 557 | "array-flatten": { 558 | "version": "1.1.1", 559 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 560 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 561 | }, 562 | "asn1": { 563 | "version": "0.2.4", 564 | "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", 565 | "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", 566 | "requires": { 567 | "safer-buffer": "~2.1.0" 568 | } 569 | }, 570 | "assert-plus": { 571 | "version": "1.0.0", 572 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 573 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 574 | }, 575 | "async-limiter": { 576 | "version": "1.0.1", 577 | "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz", 578 | "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==" 579 | }, 580 | "async-retry": { 581 | "version": "1.2.3", 582 | "resolved": "https://registry.npmjs.org/async-retry/-/async-retry-1.2.3.tgz", 583 | "integrity": "sha512-tfDb02Th6CE6pJUF2gjW5ZVjsgwlucVXOEQMvEX9JgSJMs9gAX+Nz3xRuJBKuUYjTSYORqvDBORdAQ3LU59g7Q==", 584 | "requires": { 585 | "retry": "0.12.0" 586 | } 587 | }, 588 | "asynckit": { 589 | "version": "0.4.0", 590 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 591 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" 592 | }, 593 | "aws-sign2": { 594 | "version": "0.7.0", 595 | "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", 596 | "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=" 597 | }, 598 | "aws4": { 599 | "version": "1.9.1", 600 | "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.9.1.tgz", 601 | "integrity": "sha512-wMHVg2EOHaMRxbzgFJ9gtjOOCrI80OHLG14rxi28XwOW8ux6IiEbRCGGGqCtdAIg4FQCbW20k9RsT4y3gJlFug==" 602 | }, 603 | "backo2": { 604 | "version": "1.0.2", 605 | "resolved": "https://registry.npmjs.org/backo2/-/backo2-1.0.2.tgz", 606 | "integrity": "sha1-MasayLEpNjRj41s+u2n038+6eUc=" 607 | }, 608 | "balanced-match": { 609 | "version": "1.0.0", 610 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 611 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" 612 | }, 613 | "base64url": { 614 | "version": "3.0.1", 615 | "resolved": "https://registry.npmjs.org/base64url/-/base64url-3.0.1.tgz", 616 | "integrity": "sha512-ir1UPr3dkwexU7FdV8qBBbNDRUhMmIekYMFZfi+C/sLNnRESKPl23nB9b2pltqfOQNnGzsDdId90AEtG5tCx4A==" 617 | }, 618 | "bcrypt-pbkdf": { 619 | "version": "1.0.2", 620 | "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", 621 | "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", 622 | "requires": { 623 | "tweetnacl": "^0.14.3" 624 | } 625 | }, 626 | "bcryptjs": { 627 | "version": "2.4.3", 628 | "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz", 629 | "integrity": "sha1-mrVie5PmBiH/fNrF2pczAn3x0Ms=" 630 | }, 631 | "binary-extensions": { 632 | "version": "2.0.0", 633 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.0.0.tgz", 634 | "integrity": "sha512-Phlt0plgpIIBOGTT/ehfFnbNlfsDEiqmzE2KRXoX1bLIlir4X/MR+zSyBEkL05ffWgnRSf/DXv+WrUAVr93/ow==" 635 | }, 636 | "bluebird": { 637 | "version": "3.5.1", 638 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz", 639 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA==" 640 | }, 641 | "body-parser": { 642 | "version": "1.19.0", 643 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", 644 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", 645 | "requires": { 646 | "bytes": "3.1.0", 647 | "content-type": "~1.0.4", 648 | "debug": "2.6.9", 649 | "depd": "~1.1.2", 650 | "http-errors": "1.7.2", 651 | "iconv-lite": "0.4.24", 652 | "on-finished": "~2.3.0", 653 | "qs": "6.7.0", 654 | "raw-body": "2.4.0", 655 | "type-is": "~1.6.17" 656 | }, 657 | "dependencies": { 658 | "http-errors": { 659 | "version": "1.7.2", 660 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 661 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 662 | "requires": { 663 | "depd": "~1.1.2", 664 | "inherits": "2.0.3", 665 | "setprototypeof": "1.1.1", 666 | "statuses": ">= 1.5.0 < 2", 667 | "toidentifier": "1.0.0" 668 | } 669 | }, 670 | "inherits": { 671 | "version": "2.0.3", 672 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 673 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 674 | } 675 | } 676 | }, 677 | "boxen": { 678 | "version": "1.3.0", 679 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-1.3.0.tgz", 680 | "integrity": "sha512-TNPjfTr432qx7yOjQyaXm3dSR0MH9vXp7eT1BFSl/C51g+EFnOR9hTg1IreahGBmDNCehscshe45f+C1TBZbLw==", 681 | "requires": { 682 | "ansi-align": "^2.0.0", 683 | "camelcase": "^4.0.0", 684 | "chalk": "^2.0.1", 685 | "cli-boxes": "^1.0.0", 686 | "string-width": "^2.0.0", 687 | "term-size": "^1.2.0", 688 | "widest-line": "^2.0.0" 689 | } 690 | }, 691 | "brace-expansion": { 692 | "version": "1.1.11", 693 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 694 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 695 | "requires": { 696 | "balanced-match": "^1.0.0", 697 | "concat-map": "0.0.1" 698 | } 699 | }, 700 | "braces": { 701 | "version": "3.0.2", 702 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 703 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 704 | "requires": { 705 | "fill-range": "^7.0.1" 706 | } 707 | }, 708 | "bson": { 709 | "version": "1.1.3", 710 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.3.tgz", 711 | "integrity": "sha512-TdiJxMVnodVS7r0BdL42y/pqC9cL2iKynVwA0Ho3qbsQYr428veL3l7BQyuqiw+Q5SqqoT0m4srSY/BlZ9AxXg==" 712 | }, 713 | "buffer-equal-constant-time": { 714 | "version": "1.0.1", 715 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", 716 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" 717 | }, 718 | "busboy": { 719 | "version": "0.3.1", 720 | "resolved": "https://registry.npmjs.org/busboy/-/busboy-0.3.1.tgz", 721 | "integrity": "sha512-y7tTxhGKXcyBxRKAni+awqx8uqaJKrSFSNFSeRG5CsWNdmy2BIK+6VGWEW7TZnIO/533mtMEA4rOevQV815YJw==", 722 | "requires": { 723 | "dicer": "0.3.0" 724 | } 725 | }, 726 | "bytes": { 727 | "version": "3.1.0", 728 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", 729 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" 730 | }, 731 | "camelcase": { 732 | "version": "4.1.0", 733 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-4.1.0.tgz", 734 | "integrity": "sha1-1UVjW+HjPFQmScaRc+Xeas+uNN0=" 735 | }, 736 | "capture-stack-trace": { 737 | "version": "1.0.1", 738 | "resolved": "https://registry.npmjs.org/capture-stack-trace/-/capture-stack-trace-1.0.1.tgz", 739 | "integrity": "sha512-mYQLZnx5Qt1JgB1WEiMCf2647plpGeQ2NMR/5L0HNZzGQo4fuSPnK+wjfPnKZV0aiJDgzmWqqkV/g7JD+DW0qw==" 740 | }, 741 | "caseless": { 742 | "version": "0.12.0", 743 | "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", 744 | "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=" 745 | }, 746 | "chalk": { 747 | "version": "2.4.2", 748 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 749 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 750 | "requires": { 751 | "ansi-styles": "^3.2.1", 752 | "escape-string-regexp": "^1.0.5", 753 | "supports-color": "^5.3.0" 754 | } 755 | }, 756 | "chokidar": { 757 | "version": "3.3.0", 758 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.3.0.tgz", 759 | "integrity": "sha512-dGmKLDdT3Gdl7fBUe8XK+gAtGmzy5Fn0XkkWQuYxGIgWVPPse2CxFA5mtrlD0TOHaHjEUqkWNyP1XdHoJES/4A==", 760 | "requires": { 761 | "anymatch": "~3.1.1", 762 | "braces": "~3.0.2", 763 | "fsevents": "~2.1.1", 764 | "glob-parent": "~5.1.0", 765 | "is-binary-path": "~2.1.0", 766 | "is-glob": "~4.0.1", 767 | "normalize-path": "~3.0.0", 768 | "readdirp": "~3.2.0" 769 | } 770 | }, 771 | "ci-info": { 772 | "version": "1.6.0", 773 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-1.6.0.tgz", 774 | "integrity": "sha512-vsGdkwSCDpWmP80ncATX7iea5DWQemg1UgCW5J8tqjU3lYw4FBYuj89J0CTVomA7BEfvSZd84GmHko+MxFQU2A==" 775 | }, 776 | "cli-boxes": { 777 | "version": "1.0.0", 778 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-1.0.0.tgz", 779 | "integrity": "sha1-T6kXw+WclKAEzWH47lCdplFocUM=" 780 | }, 781 | "color-convert": { 782 | "version": "1.9.3", 783 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 784 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 785 | "requires": { 786 | "color-name": "1.1.3" 787 | } 788 | }, 789 | "color-name": { 790 | "version": "1.1.3", 791 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 792 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" 793 | }, 794 | "combined-stream": { 795 | "version": "1.0.8", 796 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 797 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 798 | "requires": { 799 | "delayed-stream": "~1.0.0" 800 | } 801 | }, 802 | "concat-map": { 803 | "version": "0.0.1", 804 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 805 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" 806 | }, 807 | "configstore": { 808 | "version": "3.1.2", 809 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-3.1.2.tgz", 810 | "integrity": "sha512-vtv5HtGjcYUgFrXc6Kx747B83MRRVS5R1VTEQoXvuP+kMI+if6uywV0nDGoiydJRy4yk7h9od5Og0kxx4zUXmw==", 811 | "requires": { 812 | "dot-prop": "^4.1.0", 813 | "graceful-fs": "^4.1.2", 814 | "make-dir": "^1.0.0", 815 | "unique-string": "^1.0.0", 816 | "write-file-atomic": "^2.0.0", 817 | "xdg-basedir": "^3.0.0" 818 | } 819 | }, 820 | "content-disposition": { 821 | "version": "0.5.3", 822 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", 823 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", 824 | "requires": { 825 | "safe-buffer": "5.1.2" 826 | }, 827 | "dependencies": { 828 | "safe-buffer": { 829 | "version": "5.1.2", 830 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 831 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 832 | } 833 | } 834 | }, 835 | "content-type": { 836 | "version": "1.0.4", 837 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 838 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 839 | }, 840 | "cookie": { 841 | "version": "0.4.0", 842 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 843 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 844 | }, 845 | "cookie-signature": { 846 | "version": "1.0.6", 847 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 848 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 849 | }, 850 | "core-js": { 851 | "version": "3.5.0", 852 | "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.5.0.tgz", 853 | "integrity": "sha512-Ifh3kj78gzQ7NAoJXeTu+XwzDld0QRIwjBLRqAMhuLhP3d2Av5wmgE9ycfnvK6NAEjTkQ1sDPeoEZAWO3Hx1Uw==" 854 | }, 855 | "core-util-is": { 856 | "version": "1.0.2", 857 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 858 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 859 | }, 860 | "cors": { 861 | "version": "2.8.5", 862 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 863 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 864 | "requires": { 865 | "object-assign": "^4", 866 | "vary": "^1" 867 | } 868 | }, 869 | "create-error-class": { 870 | "version": "3.0.2", 871 | "resolved": "https://registry.npmjs.org/create-error-class/-/create-error-class-3.0.2.tgz", 872 | "integrity": "sha1-Br56vvlHo/FKMP1hBnHUAbyot7Y=", 873 | "requires": { 874 | "capture-stack-trace": "^1.0.0" 875 | } 876 | }, 877 | "cross-spawn": { 878 | "version": "5.1.0", 879 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz", 880 | "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=", 881 | "requires": { 882 | "lru-cache": "^4.0.1", 883 | "shebang-command": "^1.2.0", 884 | "which": "^1.2.9" 885 | }, 886 | "dependencies": { 887 | "lru-cache": { 888 | "version": "4.1.5", 889 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.5.tgz", 890 | "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", 891 | "requires": { 892 | "pseudomap": "^1.0.2", 893 | "yallist": "^2.1.2" 894 | } 895 | }, 896 | "yallist": { 897 | "version": "2.1.2", 898 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", 899 | "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=" 900 | } 901 | } 902 | }, 903 | "crypto-random-string": { 904 | "version": "1.0.0", 905 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-1.0.0.tgz", 906 | "integrity": "sha1-ojD2T1aDEOFJgAmUB5DsmVRbyn4=" 907 | }, 908 | "dashdash": { 909 | "version": "1.14.1", 910 | "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", 911 | "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", 912 | "requires": { 913 | "assert-plus": "^1.0.0" 914 | } 915 | }, 916 | "debug": { 917 | "version": "2.6.9", 918 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 919 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 920 | "requires": { 921 | "ms": "2.0.0" 922 | } 923 | }, 924 | "deep-extend": { 925 | "version": "0.6.0", 926 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", 927 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==" 928 | }, 929 | "deepmerge": { 930 | "version": "2.2.1", 931 | "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-2.2.1.tgz", 932 | "integrity": "sha512-R9hc1Xa/NOBi9WRVUWg19rl1UB7Tt4kuPd+thNJgFZoxXsTz7ncaPaeIm+40oSGuP33DfMb4sZt1QIGiJzC4EA==" 933 | }, 934 | "define-properties": { 935 | "version": "1.1.3", 936 | "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", 937 | "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", 938 | "requires": { 939 | "object-keys": "^1.0.12" 940 | } 941 | }, 942 | "delayed-stream": { 943 | "version": "1.0.0", 944 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 945 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=" 946 | }, 947 | "depd": { 948 | "version": "1.1.2", 949 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 950 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 951 | }, 952 | "deprecated-decorator": { 953 | "version": "0.1.6", 954 | "resolved": "https://registry.npmjs.org/deprecated-decorator/-/deprecated-decorator-0.1.6.tgz", 955 | "integrity": "sha1-AJZjF7ehL+kvPMgx91g68ym4bDc=" 956 | }, 957 | "destroy": { 958 | "version": "1.0.4", 959 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 960 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 961 | }, 962 | "dicer": { 963 | "version": "0.3.0", 964 | "resolved": "https://registry.npmjs.org/dicer/-/dicer-0.3.0.tgz", 965 | "integrity": "sha512-MdceRRWqltEG2dZqO769g27N/3PXfcKl04VhYnBlo2YhH7zPi88VebsjTKclaOyiuMaGU72hTfw3VkUitGcVCA==", 966 | "requires": { 967 | "streamsearch": "0.1.2" 968 | } 969 | }, 970 | "dot-prop": { 971 | "version": "4.2.0", 972 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-4.2.0.tgz", 973 | "integrity": "sha512-tUMXrxlExSW6U2EXiiKGSBVdYgtV8qlHL+C10TsW4PURY/ic+eaysnSkwB4kA/mBlCyy/IKDJ+Lc3wbWeaXtuQ==", 974 | "requires": { 975 | "is-obj": "^1.0.0" 976 | } 977 | }, 978 | "dotenv": { 979 | "version": "8.2.0", 980 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz", 981 | "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw==" 982 | }, 983 | "duplexer3": { 984 | "version": "0.1.4", 985 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", 986 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=" 987 | }, 988 | "ecc-jsbn": { 989 | "version": "0.1.2", 990 | "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", 991 | "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", 992 | "requires": { 993 | "jsbn": "~0.1.0", 994 | "safer-buffer": "^2.1.0" 995 | } 996 | }, 997 | "ecdsa-sig-formatter": { 998 | "version": "1.0.11", 999 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", 1000 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", 1001 | "requires": { 1002 | "safe-buffer": "^5.0.1" 1003 | } 1004 | }, 1005 | "ee-first": { 1006 | "version": "1.1.1", 1007 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 1008 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 1009 | }, 1010 | "encodeurl": { 1011 | "version": "1.0.2", 1012 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 1013 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 1014 | }, 1015 | "es-abstract": { 1016 | "version": "1.16.3", 1017 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.16.3.tgz", 1018 | "integrity": "sha512-WtY7Fx5LiOnSYgF5eg/1T+GONaGmpvpPdCpSnYij+U2gDTL0UPfWrhDw7b2IYb+9NQJsYpCA0wOQvZfsd6YwRw==", 1019 | "requires": { 1020 | "es-to-primitive": "^1.2.1", 1021 | "function-bind": "^1.1.1", 1022 | "has": "^1.0.3", 1023 | "has-symbols": "^1.0.1", 1024 | "is-callable": "^1.1.4", 1025 | "is-regex": "^1.0.4", 1026 | "object-inspect": "^1.7.0", 1027 | "object-keys": "^1.1.1", 1028 | "string.prototype.trimleft": "^2.1.0", 1029 | "string.prototype.trimright": "^2.1.0" 1030 | } 1031 | }, 1032 | "es-to-primitive": { 1033 | "version": "1.2.1", 1034 | "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", 1035 | "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", 1036 | "requires": { 1037 | "is-callable": "^1.1.4", 1038 | "is-date-object": "^1.0.1", 1039 | "is-symbol": "^1.0.2" 1040 | } 1041 | }, 1042 | "es6-promise": { 1043 | "version": "4.2.8", 1044 | "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", 1045 | "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==" 1046 | }, 1047 | "es6-promisify": { 1048 | "version": "5.0.0", 1049 | "resolved": "https://registry.npmjs.org/es6-promisify/-/es6-promisify-5.0.0.tgz", 1050 | "integrity": "sha1-UQnWLz5W6pZ8S2NQWu8IKRyKUgM=", 1051 | "requires": { 1052 | "es6-promise": "^4.0.3" 1053 | } 1054 | }, 1055 | "escape-html": { 1056 | "version": "1.0.3", 1057 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 1058 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 1059 | }, 1060 | "escape-string-regexp": { 1061 | "version": "1.0.5", 1062 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 1063 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=" 1064 | }, 1065 | "esm": { 1066 | "version": "3.2.25", 1067 | "resolved": "https://registry.npmjs.org/esm/-/esm-3.2.25.tgz", 1068 | "integrity": "sha512-U1suiZ2oDVWv4zPO56S0NcR5QriEahGtdN2OR6FiOG4WJvcjBVFB0qI4+eKoWFH483PKGuLuu6V8Z4T5g63UVA==" 1069 | }, 1070 | "etag": { 1071 | "version": "1.8.1", 1072 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 1073 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 1074 | }, 1075 | "eventemitter3": { 1076 | "version": "3.1.2", 1077 | "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-3.1.2.tgz", 1078 | "integrity": "sha512-tvtQIeLVHjDkJYnzf2dgVMxfuSGJeM/7UCG17TT4EumTfNtF+0nebF/4zWOIkCreAbtNqhGEboB6BWrwqNaw4Q==" 1079 | }, 1080 | "execa": { 1081 | "version": "0.7.0", 1082 | "resolved": "https://registry.npmjs.org/execa/-/execa-0.7.0.tgz", 1083 | "integrity": "sha1-lEvs00zEHuMqY6n68nrVpl/Fl3c=", 1084 | "requires": { 1085 | "cross-spawn": "^5.0.1", 1086 | "get-stream": "^3.0.0", 1087 | "is-stream": "^1.1.0", 1088 | "npm-run-path": "^2.0.0", 1089 | "p-finally": "^1.0.0", 1090 | "signal-exit": "^3.0.0", 1091 | "strip-eof": "^1.0.0" 1092 | } 1093 | }, 1094 | "express": { 1095 | "version": "4.17.1", 1096 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", 1097 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", 1098 | "requires": { 1099 | "accepts": "~1.3.7", 1100 | "array-flatten": "1.1.1", 1101 | "body-parser": "1.19.0", 1102 | "content-disposition": "0.5.3", 1103 | "content-type": "~1.0.4", 1104 | "cookie": "0.4.0", 1105 | "cookie-signature": "1.0.6", 1106 | "debug": "2.6.9", 1107 | "depd": "~1.1.2", 1108 | "encodeurl": "~1.0.2", 1109 | "escape-html": "~1.0.3", 1110 | "etag": "~1.8.1", 1111 | "finalhandler": "~1.1.2", 1112 | "fresh": "0.5.2", 1113 | "merge-descriptors": "1.0.1", 1114 | "methods": "~1.1.2", 1115 | "on-finished": "~2.3.0", 1116 | "parseurl": "~1.3.3", 1117 | "path-to-regexp": "0.1.7", 1118 | "proxy-addr": "~2.0.5", 1119 | "qs": "6.7.0", 1120 | "range-parser": "~1.2.1", 1121 | "safe-buffer": "5.1.2", 1122 | "send": "0.17.1", 1123 | "serve-static": "1.14.1", 1124 | "setprototypeof": "1.1.1", 1125 | "statuses": "~1.5.0", 1126 | "type-is": "~1.6.18", 1127 | "utils-merge": "1.0.1", 1128 | "vary": "~1.1.2" 1129 | }, 1130 | "dependencies": { 1131 | "safe-buffer": { 1132 | "version": "5.1.2", 1133 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1134 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 1135 | } 1136 | } 1137 | }, 1138 | "extend": { 1139 | "version": "3.0.2", 1140 | "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", 1141 | "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==" 1142 | }, 1143 | "extsprintf": { 1144 | "version": "1.3.0", 1145 | "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", 1146 | "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=" 1147 | }, 1148 | "fast-deep-equal": { 1149 | "version": "3.1.1", 1150 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.1.tgz", 1151 | "integrity": "sha512-8UEa58QDLauDNfpbrX55Q9jrGHThw2ZMdOky5Gl1CDtVeJDPVrG4Jxx1N8jw2gkWaff5UUuX1KJd+9zGe2B+ZA==" 1152 | }, 1153 | "fast-json-stable-stringify": { 1154 | "version": "2.0.0", 1155 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", 1156 | "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=" 1157 | }, 1158 | "fill-range": { 1159 | "version": "7.0.1", 1160 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 1161 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 1162 | "requires": { 1163 | "to-regex-range": "^5.0.1" 1164 | } 1165 | }, 1166 | "finalhandler": { 1167 | "version": "1.1.2", 1168 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", 1169 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", 1170 | "requires": { 1171 | "debug": "2.6.9", 1172 | "encodeurl": "~1.0.2", 1173 | "escape-html": "~1.0.3", 1174 | "on-finished": "~2.3.0", 1175 | "parseurl": "~1.3.3", 1176 | "statuses": "~1.5.0", 1177 | "unpipe": "~1.0.0" 1178 | } 1179 | }, 1180 | "forever-agent": { 1181 | "version": "0.6.1", 1182 | "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", 1183 | "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=" 1184 | }, 1185 | "form-data": { 1186 | "version": "2.5.1", 1187 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.5.1.tgz", 1188 | "integrity": "sha512-m21N3WOmEEURgk6B9GLOE4RuWOFf28Lhh9qGYeNlGq4VDXUlJy2th2slBNU8Gp8EzloYZOibZJ7t5ecIrFSjVA==", 1189 | "requires": { 1190 | "asynckit": "^0.4.0", 1191 | "combined-stream": "^1.0.6", 1192 | "mime-types": "^2.1.12" 1193 | } 1194 | }, 1195 | "forwarded": { 1196 | "version": "0.1.2", 1197 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 1198 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 1199 | }, 1200 | "fresh": { 1201 | "version": "0.5.2", 1202 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 1203 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 1204 | }, 1205 | "fs-capacitor": { 1206 | "version": "2.0.4", 1207 | "resolved": "https://registry.npmjs.org/fs-capacitor/-/fs-capacitor-2.0.4.tgz", 1208 | "integrity": "sha512-8S4f4WsCryNw2mJJchi46YgB6CR5Ze+4L1h8ewl9tEpL4SJ3ZO+c/bS4BWhB8bK+O3TMqhuZarTitd0S0eh2pA==" 1209 | }, 1210 | "fsevents": { 1211 | "version": "2.1.2", 1212 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.2.tgz", 1213 | "integrity": "sha512-R4wDiBwZ0KzpgOWetKDug1FZcYhqYnUYKtfZYt4mD5SBz76q0KR4Q9o7GIPamsVPGmW3EYPPJ0dOOjvx32ldZA==", 1214 | "optional": true 1215 | }, 1216 | "function-bind": { 1217 | "version": "1.1.1", 1218 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 1219 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" 1220 | }, 1221 | "get-stream": { 1222 | "version": "3.0.0", 1223 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz", 1224 | "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=" 1225 | }, 1226 | "getpass": { 1227 | "version": "0.1.7", 1228 | "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", 1229 | "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", 1230 | "requires": { 1231 | "assert-plus": "^1.0.0" 1232 | } 1233 | }, 1234 | "glob-parent": { 1235 | "version": "5.1.0", 1236 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.0.tgz", 1237 | "integrity": "sha512-qjtRgnIVmOfnKUE3NJAQEdk+lKrxfw8t5ke7SXtfMTHcjsBfOfWXCQfdb30zfDoZQ2IRSIiidmjtbHZPZ++Ihw==", 1238 | "requires": { 1239 | "is-glob": "^4.0.1" 1240 | } 1241 | }, 1242 | "global-dirs": { 1243 | "version": "0.1.1", 1244 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-0.1.1.tgz", 1245 | "integrity": "sha1-sxnA3UYH81PzvpzKTHL8FIxJ9EU=", 1246 | "requires": { 1247 | "ini": "^1.3.4" 1248 | } 1249 | }, 1250 | "got": { 1251 | "version": "6.7.1", 1252 | "resolved": "https://registry.npmjs.org/got/-/got-6.7.1.tgz", 1253 | "integrity": "sha1-JAzQV4WpoY5WHcG0S0HHY+8ejbA=", 1254 | "requires": { 1255 | "create-error-class": "^3.0.0", 1256 | "duplexer3": "^0.1.4", 1257 | "get-stream": "^3.0.0", 1258 | "is-redirect": "^1.0.0", 1259 | "is-retry-allowed": "^1.0.0", 1260 | "is-stream": "^1.0.0", 1261 | "lowercase-keys": "^1.0.0", 1262 | "safe-buffer": "^5.0.1", 1263 | "timed-out": "^4.0.0", 1264 | "unzip-response": "^2.0.1", 1265 | "url-parse-lax": "^1.0.0" 1266 | } 1267 | }, 1268 | "graceful-fs": { 1269 | "version": "4.2.3", 1270 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.3.tgz", 1271 | "integrity": "sha512-a30VEBm4PEdx1dRB7MFK7BejejvCvBronbLjht+sHuGYj8PHs7M/5Z+rt5lw551vZ7yfTCj4Vuyy3mSJytDWRQ==" 1272 | }, 1273 | "graphql": { 1274 | "version": "14.5.8", 1275 | "resolved": "https://registry.npmjs.org/graphql/-/graphql-14.5.8.tgz", 1276 | "integrity": "sha512-MMwmi0zlVLQKLdGiMfWkgQD7dY/TUKt4L+zgJ/aR0Howebod3aNgP5JkgvAULiR2HPVZaP2VEElqtdidHweLkg==", 1277 | "requires": { 1278 | "iterall": "^1.2.2" 1279 | } 1280 | }, 1281 | "graphql-extensions": { 1282 | "version": "0.10.7", 1283 | "resolved": "https://registry.npmjs.org/graphql-extensions/-/graphql-extensions-0.10.7.tgz", 1284 | "integrity": "sha512-YuP7VQxNePG4bWRQ5Vk+KRMbZ9r1IWCqCCogOMz/1ueeQ4gZe93eGRcb0vhpOdMFnCX6Vyvd4+sC+N6LR3YFOQ==", 1285 | "requires": { 1286 | "@apollographql/apollo-tools": "^0.4.0", 1287 | "apollo-server-env": "^2.4.3", 1288 | "apollo-server-types": "^0.2.8" 1289 | } 1290 | }, 1291 | "graphql-iso-date": { 1292 | "version": "3.6.1", 1293 | "resolved": "https://registry.npmjs.org/graphql-iso-date/-/graphql-iso-date-3.6.1.tgz", 1294 | "integrity": "sha512-AwFGIuYMJQXOEAgRlJlFL4H1ncFM8n8XmoVDTNypNOZyQ8LFDG2ppMFlsS862BSTCDcSUfHp8PD3/uJhv7t59Q==" 1295 | }, 1296 | "graphql-subscriptions": { 1297 | "version": "1.1.0", 1298 | "resolved": "https://registry.npmjs.org/graphql-subscriptions/-/graphql-subscriptions-1.1.0.tgz", 1299 | "integrity": "sha512-6WzlBFC0lWmXJbIVE8OgFgXIP4RJi3OQgTPa0DVMsDXdpRDjTsM1K9wfl5HSYX7R87QAGlvcv2Y4BIZa/ItonA==", 1300 | "requires": { 1301 | "iterall": "^1.2.1" 1302 | } 1303 | }, 1304 | "graphql-tag": { 1305 | "version": "2.10.1", 1306 | "resolved": "https://registry.npmjs.org/graphql-tag/-/graphql-tag-2.10.1.tgz", 1307 | "integrity": "sha512-jApXqWBzNXQ8jYa/HLkZJaVw9jgwNqZkywa2zfFn16Iv1Zb7ELNHkJaXHR7Quvd5SIGsy6Ny7SUKATgnu05uEg==" 1308 | }, 1309 | "graphql-tools": { 1310 | "version": "4.0.6", 1311 | "resolved": "https://registry.npmjs.org/graphql-tools/-/graphql-tools-4.0.6.tgz", 1312 | "integrity": "sha512-jHLQw8x3xmSNRBCsaZqelXXsFfUSUSktSCUP8KYHiX1Z9qEuwcMpAf+FkdBzk8aTAFqOlPdNZ3OI4DKKqGKUqg==", 1313 | "requires": { 1314 | "apollo-link": "^1.2.3", 1315 | "apollo-utilities": "^1.0.1", 1316 | "deprecated-decorator": "^0.1.6", 1317 | "iterall": "^1.1.3", 1318 | "uuid": "^3.1.0" 1319 | } 1320 | }, 1321 | "graphql-upload": { 1322 | "version": "8.1.0", 1323 | "resolved": "https://registry.npmjs.org/graphql-upload/-/graphql-upload-8.1.0.tgz", 1324 | "integrity": "sha512-U2OiDI5VxYmzRKw0Z2dmfk0zkqMRaecH9Smh1U277gVgVe9Qn+18xqf4skwr4YJszGIh7iQDZ57+5ygOK9sM/Q==", 1325 | "requires": { 1326 | "busboy": "^0.3.1", 1327 | "fs-capacitor": "^2.0.4", 1328 | "http-errors": "^1.7.3", 1329 | "object-path": "^0.11.4" 1330 | } 1331 | }, 1332 | "har-schema": { 1333 | "version": "2.0.0", 1334 | "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", 1335 | "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=" 1336 | }, 1337 | "har-validator": { 1338 | "version": "5.1.3", 1339 | "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.3.tgz", 1340 | "integrity": "sha512-sNvOCzEQNr/qrvJgc3UG/kD4QtlHycrzwS+6mfTrrSq97BvaYcPZZI1ZSqGSPR73Cxn4LKTD4PttRwfU7jWq5g==", 1341 | "requires": { 1342 | "ajv": "^6.5.5", 1343 | "har-schema": "^2.0.0" 1344 | } 1345 | }, 1346 | "has": { 1347 | "version": "1.0.3", 1348 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 1349 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 1350 | "requires": { 1351 | "function-bind": "^1.1.1" 1352 | } 1353 | }, 1354 | "has-flag": { 1355 | "version": "3.0.0", 1356 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 1357 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=" 1358 | }, 1359 | "has-symbols": { 1360 | "version": "1.0.1", 1361 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", 1362 | "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==" 1363 | }, 1364 | "http-errors": { 1365 | "version": "1.7.3", 1366 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.3.tgz", 1367 | "integrity": "sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw==", 1368 | "requires": { 1369 | "depd": "~1.1.2", 1370 | "inherits": "2.0.4", 1371 | "setprototypeof": "1.1.1", 1372 | "statuses": ">= 1.5.0 < 2", 1373 | "toidentifier": "1.0.0" 1374 | } 1375 | }, 1376 | "http-signature": { 1377 | "version": "1.2.0", 1378 | "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", 1379 | "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", 1380 | "requires": { 1381 | "assert-plus": "^1.0.0", 1382 | "jsprim": "^1.2.2", 1383 | "sshpk": "^1.7.0" 1384 | } 1385 | }, 1386 | "https-proxy-agent": { 1387 | "version": "2.2.4", 1388 | "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-2.2.4.tgz", 1389 | "integrity": "sha512-OmvfoQ53WLjtA9HeYP9RNrWMJzzAz1JGaSFr1nijg0PVR1JaD/xbJq1mdEIIlxGpXp9eSe/O2LgU9DJmTPd0Eg==", 1390 | "requires": { 1391 | "agent-base": "^4.3.0", 1392 | "debug": "^3.1.0" 1393 | }, 1394 | "dependencies": { 1395 | "debug": { 1396 | "version": "3.2.6", 1397 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", 1398 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", 1399 | "requires": { 1400 | "ms": "^2.1.1" 1401 | } 1402 | }, 1403 | "ms": { 1404 | "version": "2.1.2", 1405 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1406 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1407 | } 1408 | } 1409 | }, 1410 | "iconv-lite": { 1411 | "version": "0.4.24", 1412 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 1413 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 1414 | "requires": { 1415 | "safer-buffer": ">= 2.1.2 < 3" 1416 | } 1417 | }, 1418 | "ignore-by-default": { 1419 | "version": "1.0.1", 1420 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 1421 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=" 1422 | }, 1423 | "import-lazy": { 1424 | "version": "2.1.0", 1425 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", 1426 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=" 1427 | }, 1428 | "imurmurhash": { 1429 | "version": "0.1.4", 1430 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 1431 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=" 1432 | }, 1433 | "inherits": { 1434 | "version": "2.0.4", 1435 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 1436 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" 1437 | }, 1438 | "ini": { 1439 | "version": "1.3.5", 1440 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", 1441 | "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==" 1442 | }, 1443 | "ipaddr.js": { 1444 | "version": "1.9.0", 1445 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.0.tgz", 1446 | "integrity": "sha512-M4Sjn6N/+O6/IXSJseKqHoFc+5FdGJ22sXqnjTpdZweHK64MzEPAyQZyEU3R/KRv2GLoa7nNtg/C2Ev6m7z+eA==" 1447 | }, 1448 | "is-binary-path": { 1449 | "version": "2.1.0", 1450 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 1451 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 1452 | "requires": { 1453 | "binary-extensions": "^2.0.0" 1454 | } 1455 | }, 1456 | "is-callable": { 1457 | "version": "1.1.4", 1458 | "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.4.tgz", 1459 | "integrity": "sha512-r5p9sxJjYnArLjObpjA4xu5EKI3CuKHkJXMhT7kwbpUyIFD1n5PMAsoPvWnvtZiNz7LjkYDRZhd7FlI0eMijEA==" 1460 | }, 1461 | "is-ci": { 1462 | "version": "1.2.1", 1463 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-1.2.1.tgz", 1464 | "integrity": "sha512-s6tfsaQaQi3JNciBH6shVqEDvhGut0SUXr31ag8Pd8BBbVVlcGfWhpPmEOoM6RJ5TFhbypvf5yyRw/VXW1IiWg==", 1465 | "requires": { 1466 | "ci-info": "^1.5.0" 1467 | } 1468 | }, 1469 | "is-date-object": { 1470 | "version": "1.0.1", 1471 | "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.1.tgz", 1472 | "integrity": "sha1-mqIOtq7rv/d/vTPnTKAbM1gdOhY=" 1473 | }, 1474 | "is-extglob": { 1475 | "version": "2.1.1", 1476 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 1477 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=" 1478 | }, 1479 | "is-fullwidth-code-point": { 1480 | "version": "2.0.0", 1481 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 1482 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=" 1483 | }, 1484 | "is-glob": { 1485 | "version": "4.0.1", 1486 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 1487 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 1488 | "requires": { 1489 | "is-extglob": "^2.1.1" 1490 | } 1491 | }, 1492 | "is-installed-globally": { 1493 | "version": "0.1.0", 1494 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.1.0.tgz", 1495 | "integrity": "sha1-Df2Y9akRFxbdU13aZJL2e/PSWoA=", 1496 | "requires": { 1497 | "global-dirs": "^0.1.0", 1498 | "is-path-inside": "^1.0.0" 1499 | } 1500 | }, 1501 | "is-npm": { 1502 | "version": "1.0.0", 1503 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-1.0.0.tgz", 1504 | "integrity": "sha1-8vtjpl5JBbQGyGBydloaTceTufQ=" 1505 | }, 1506 | "is-number": { 1507 | "version": "7.0.0", 1508 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 1509 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==" 1510 | }, 1511 | "is-obj": { 1512 | "version": "1.0.1", 1513 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-1.0.1.tgz", 1514 | "integrity": "sha1-PkcprB9f3gJc19g6iW2rn09n2w8=" 1515 | }, 1516 | "is-path-inside": { 1517 | "version": "1.0.1", 1518 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.1.tgz", 1519 | "integrity": "sha1-jvW33lBDej/cprToZe96pVy0gDY=", 1520 | "requires": { 1521 | "path-is-inside": "^1.0.1" 1522 | } 1523 | }, 1524 | "is-redirect": { 1525 | "version": "1.0.0", 1526 | "resolved": "https://registry.npmjs.org/is-redirect/-/is-redirect-1.0.0.tgz", 1527 | "integrity": "sha1-HQPd7VO9jbDzDCbk+V02/HyH3CQ=" 1528 | }, 1529 | "is-regex": { 1530 | "version": "1.0.4", 1531 | "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.4.tgz", 1532 | "integrity": "sha1-VRdIm1RwkbCTDglWVM7SXul+lJE=", 1533 | "requires": { 1534 | "has": "^1.0.1" 1535 | } 1536 | }, 1537 | "is-retry-allowed": { 1538 | "version": "1.2.0", 1539 | "resolved": "https://registry.npmjs.org/is-retry-allowed/-/is-retry-allowed-1.2.0.tgz", 1540 | "integrity": "sha512-RUbUeKwvm3XG2VYamhJL1xFktgjvPzL0Hq8C+6yrWIswDy3BIXGqCxhxkc30N9jqK311gVU137K8Ei55/zVJRg==" 1541 | }, 1542 | "is-stream": { 1543 | "version": "1.1.0", 1544 | "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", 1545 | "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=" 1546 | }, 1547 | "is-symbol": { 1548 | "version": "1.0.3", 1549 | "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", 1550 | "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", 1551 | "requires": { 1552 | "has-symbols": "^1.0.1" 1553 | } 1554 | }, 1555 | "is-typedarray": { 1556 | "version": "1.0.0", 1557 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 1558 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=" 1559 | }, 1560 | "isexe": { 1561 | "version": "2.0.0", 1562 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 1563 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=" 1564 | }, 1565 | "isstream": { 1566 | "version": "0.1.2", 1567 | "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", 1568 | "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=" 1569 | }, 1570 | "iterall": { 1571 | "version": "1.2.2", 1572 | "resolved": "https://registry.npmjs.org/iterall/-/iterall-1.2.2.tgz", 1573 | "integrity": "sha512-yynBb1g+RFUPY64fTrFv7nsjRrENBQJaX2UL+2Szc9REFrSNm1rpSXHGzhmAy7a9uv3vlvgBlXnf9RqmPH1/DA==" 1574 | }, 1575 | "jsbn": { 1576 | "version": "0.1.1", 1577 | "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", 1578 | "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=" 1579 | }, 1580 | "json-schema": { 1581 | "version": "0.2.3", 1582 | "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", 1583 | "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=" 1584 | }, 1585 | "json-schema-traverse": { 1586 | "version": "0.4.1", 1587 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", 1588 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==" 1589 | }, 1590 | "json-stringify-safe": { 1591 | "version": "5.0.1", 1592 | "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", 1593 | "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=" 1594 | }, 1595 | "jsonwebtoken": { 1596 | "version": "8.5.1", 1597 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz", 1598 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==", 1599 | "requires": { 1600 | "jws": "^3.2.2", 1601 | "lodash.includes": "^4.3.0", 1602 | "lodash.isboolean": "^3.0.3", 1603 | "lodash.isinteger": "^4.0.4", 1604 | "lodash.isnumber": "^3.0.3", 1605 | "lodash.isplainobject": "^4.0.6", 1606 | "lodash.isstring": "^4.0.1", 1607 | "lodash.once": "^4.0.0", 1608 | "ms": "^2.1.1", 1609 | "semver": "^5.6.0" 1610 | }, 1611 | "dependencies": { 1612 | "ms": { 1613 | "version": "2.1.2", 1614 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1615 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1616 | } 1617 | } 1618 | }, 1619 | "jsprim": { 1620 | "version": "1.4.1", 1621 | "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", 1622 | "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", 1623 | "requires": { 1624 | "assert-plus": "1.0.0", 1625 | "extsprintf": "1.3.0", 1626 | "json-schema": "0.2.3", 1627 | "verror": "1.10.0" 1628 | } 1629 | }, 1630 | "jwa": { 1631 | "version": "1.4.1", 1632 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", 1633 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", 1634 | "requires": { 1635 | "buffer-equal-constant-time": "1.0.1", 1636 | "ecdsa-sig-formatter": "1.0.11", 1637 | "safe-buffer": "^5.0.1" 1638 | } 1639 | }, 1640 | "jws": { 1641 | "version": "3.2.2", 1642 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", 1643 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", 1644 | "requires": { 1645 | "jwa": "^1.4.1", 1646 | "safe-buffer": "^5.0.1" 1647 | } 1648 | }, 1649 | "kareem": { 1650 | "version": "2.3.1", 1651 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.1.tgz", 1652 | "integrity": "sha512-l3hLhffs9zqoDe8zjmb/mAN4B8VT3L56EUvKNqLFVs9YlFA+zx7ke1DO8STAdDyYNkeSo1nKmjuvQeI12So8Xw==" 1653 | }, 1654 | "latest-version": { 1655 | "version": "3.1.0", 1656 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-3.1.0.tgz", 1657 | "integrity": "sha1-ogU4P+oyKzO1rjsYq+4NwvNW7hU=", 1658 | "requires": { 1659 | "package-json": "^4.0.0" 1660 | } 1661 | }, 1662 | "lodash": { 1663 | "version": "4.17.15", 1664 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", 1665 | "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==" 1666 | }, 1667 | "lodash.includes": { 1668 | "version": "4.3.0", 1669 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", 1670 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=" 1671 | }, 1672 | "lodash.isboolean": { 1673 | "version": "3.0.3", 1674 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", 1675 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=" 1676 | }, 1677 | "lodash.isinteger": { 1678 | "version": "4.0.4", 1679 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", 1680 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=" 1681 | }, 1682 | "lodash.isnumber": { 1683 | "version": "3.0.3", 1684 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", 1685 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=" 1686 | }, 1687 | "lodash.isplainobject": { 1688 | "version": "4.0.6", 1689 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", 1690 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" 1691 | }, 1692 | "lodash.isstring": { 1693 | "version": "4.0.1", 1694 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", 1695 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=" 1696 | }, 1697 | "lodash.once": { 1698 | "version": "4.1.1", 1699 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", 1700 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=" 1701 | }, 1702 | "lodash.sortby": { 1703 | "version": "4.7.0", 1704 | "resolved": "https://registry.npmjs.org/lodash.sortby/-/lodash.sortby-4.7.0.tgz", 1705 | "integrity": "sha1-7dFMgk4sycHgsKG0K7UhBRakJDg=" 1706 | }, 1707 | "long": { 1708 | "version": "4.0.0", 1709 | "resolved": "https://registry.npmjs.org/long/-/long-4.0.0.tgz", 1710 | "integrity": "sha512-XsP+KhQif4bjX1kbuSiySJFNAehNxgLb6hPRGJ9QsUr8ajHkuXGdrHmFUTUUXhDwVX2R5bY4JNZEwbUiMhV+MA==" 1711 | }, 1712 | "lowercase-keys": { 1713 | "version": "1.0.1", 1714 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", 1715 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==" 1716 | }, 1717 | "lru-cache": { 1718 | "version": "5.1.1", 1719 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", 1720 | "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", 1721 | "requires": { 1722 | "yallist": "^3.0.2" 1723 | } 1724 | }, 1725 | "make-dir": { 1726 | "version": "1.3.0", 1727 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-1.3.0.tgz", 1728 | "integrity": "sha512-2w31R7SJtieJJnQtGc7RVL2StM2vGYVfqUOvUDxH6bC6aJTxPxTF0GnIgCyu7tjockiUWAYQRbxa7vKn34s5sQ==", 1729 | "requires": { 1730 | "pify": "^3.0.0" 1731 | } 1732 | }, 1733 | "media-typer": { 1734 | "version": "0.3.0", 1735 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 1736 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 1737 | }, 1738 | "memory-pager": { 1739 | "version": "1.5.0", 1740 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", 1741 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", 1742 | "optional": true 1743 | }, 1744 | "merge-descriptors": { 1745 | "version": "1.0.1", 1746 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 1747 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 1748 | }, 1749 | "methods": { 1750 | "version": "1.1.2", 1751 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 1752 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 1753 | }, 1754 | "mime": { 1755 | "version": "1.6.0", 1756 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 1757 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 1758 | }, 1759 | "mime-db": { 1760 | "version": "1.42.0", 1761 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.42.0.tgz", 1762 | "integrity": "sha512-UbfJCR4UAVRNgMpfImz05smAXK7+c+ZntjaA26ANtkXLlOe947Aag5zdIcKQULAiF9Cq4WxBi9jUs5zkA84bYQ==" 1763 | }, 1764 | "mime-types": { 1765 | "version": "2.1.25", 1766 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.25.tgz", 1767 | "integrity": "sha512-5KhStqB5xpTAeGqKBAMgwaYMnQik7teQN4IAzC7npDv6kzeU6prfkR67bc87J1kWMPGkoaZSq1npmexMgkmEVg==", 1768 | "requires": { 1769 | "mime-db": "1.42.0" 1770 | } 1771 | }, 1772 | "minimatch": { 1773 | "version": "3.0.4", 1774 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 1775 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 1776 | "requires": { 1777 | "brace-expansion": "^1.1.7" 1778 | } 1779 | }, 1780 | "minimist": { 1781 | "version": "1.2.0", 1782 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", 1783 | "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=" 1784 | }, 1785 | "mongodb": { 1786 | "version": "3.3.5", 1787 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.3.5.tgz", 1788 | "integrity": "sha512-6NAv5gTFdwRyVfCz+O+KDszvjpyxmZw+VlmqmqKR2GmpkeKrKFRv/ZslgTtZba2dc9JYixIf99T5Gih7TIWv7Q==", 1789 | "requires": { 1790 | "bson": "^1.1.1", 1791 | "require_optional": "^1.0.1", 1792 | "safe-buffer": "^5.1.2", 1793 | "saslprep": "^1.0.0" 1794 | } 1795 | }, 1796 | "mongoose": { 1797 | "version": "5.8.1", 1798 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.8.1.tgz", 1799 | "integrity": "sha512-8Cffl52cMK2iBlpLipoRKW/RdrhkxvVzXsy+xVsfbKHQBCWkFiS0T0jU4smYzomTMP4gW0sReJoRA7Gu/7VVgQ==", 1800 | "requires": { 1801 | "bson": "~1.1.1", 1802 | "kareem": "2.3.1", 1803 | "mongodb": "3.3.5", 1804 | "mongoose-legacy-pluralize": "1.0.2", 1805 | "mpath": "0.6.0", 1806 | "mquery": "3.2.2", 1807 | "ms": "2.1.2", 1808 | "regexp-clone": "1.0.0", 1809 | "safe-buffer": "5.1.2", 1810 | "sift": "7.0.1", 1811 | "sliced": "1.0.1" 1812 | }, 1813 | "dependencies": { 1814 | "ms": { 1815 | "version": "2.1.2", 1816 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1817 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1818 | }, 1819 | "safe-buffer": { 1820 | "version": "5.1.2", 1821 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1822 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 1823 | } 1824 | } 1825 | }, 1826 | "mongoose-legacy-pluralize": { 1827 | "version": "1.0.2", 1828 | "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz", 1829 | "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ==" 1830 | }, 1831 | "mpath": { 1832 | "version": "0.6.0", 1833 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.6.0.tgz", 1834 | "integrity": "sha512-i75qh79MJ5Xo/sbhxrDrPSEG0H/mr1kcZXJ8dH6URU5jD/knFxCVqVC/gVSW7GIXL/9hHWlT9haLbCXWOll3qw==" 1835 | }, 1836 | "mquery": { 1837 | "version": "3.2.2", 1838 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.2.tgz", 1839 | "integrity": "sha512-XB52992COp0KP230I3qloVUbkLUxJIu328HBP2t2EsxSFtf4W1HPSOBWOXf1bqxK4Xbb66lfMJ+Bpfd9/yZE1Q==", 1840 | "requires": { 1841 | "bluebird": "3.5.1", 1842 | "debug": "3.1.0", 1843 | "regexp-clone": "^1.0.0", 1844 | "safe-buffer": "5.1.2", 1845 | "sliced": "1.0.1" 1846 | }, 1847 | "dependencies": { 1848 | "debug": { 1849 | "version": "3.1.0", 1850 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", 1851 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", 1852 | "requires": { 1853 | "ms": "2.0.0" 1854 | } 1855 | }, 1856 | "safe-buffer": { 1857 | "version": "5.1.2", 1858 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1859 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 1860 | } 1861 | } 1862 | }, 1863 | "ms": { 1864 | "version": "2.0.0", 1865 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1866 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 1867 | }, 1868 | "negotiator": { 1869 | "version": "0.6.2", 1870 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 1871 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 1872 | }, 1873 | "node-fetch": { 1874 | "version": "2.6.0", 1875 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.0.tgz", 1876 | "integrity": "sha512-8dG4H5ujfvFiqDmVu9fQ5bOHUC15JMjMY/Zumv26oOvvVJjM67KF8koCWIabKQ1GJIa9r2mMZscBq/TbdOcmNA==" 1877 | }, 1878 | "nodemon": { 1879 | "version": "2.0.2", 1880 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.2.tgz", 1881 | "integrity": "sha512-GWhYPMfde2+M0FsHnggIHXTqPDHXia32HRhh6H0d75Mt9FKUoCBvumNHr7LdrpPBTKxsWmIEOjoN+P4IU6Hcaw==", 1882 | "requires": { 1883 | "chokidar": "^3.2.2", 1884 | "debug": "^3.2.6", 1885 | "ignore-by-default": "^1.0.1", 1886 | "minimatch": "^3.0.4", 1887 | "pstree.remy": "^1.1.7", 1888 | "semver": "^5.7.1", 1889 | "supports-color": "^5.5.0", 1890 | "touch": "^3.1.0", 1891 | "undefsafe": "^2.0.2", 1892 | "update-notifier": "^2.5.0" 1893 | }, 1894 | "dependencies": { 1895 | "debug": { 1896 | "version": "3.2.6", 1897 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", 1898 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", 1899 | "requires": { 1900 | "ms": "^2.1.1" 1901 | } 1902 | }, 1903 | "ms": { 1904 | "version": "2.1.2", 1905 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1906 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1907 | } 1908 | } 1909 | }, 1910 | "nopt": { 1911 | "version": "1.0.10", 1912 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", 1913 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", 1914 | "requires": { 1915 | "abbrev": "1" 1916 | } 1917 | }, 1918 | "normalize-path": { 1919 | "version": "3.0.0", 1920 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1921 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==" 1922 | }, 1923 | "npm-run-path": { 1924 | "version": "2.0.2", 1925 | "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz", 1926 | "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=", 1927 | "requires": { 1928 | "path-key": "^2.0.0" 1929 | } 1930 | }, 1931 | "oauth": { 1932 | "version": "0.9.15", 1933 | "resolved": "https://registry.npmjs.org/oauth/-/oauth-0.9.15.tgz", 1934 | "integrity": "sha1-vR/vr2hslrdUda7VGWQS/2DPucE=" 1935 | }, 1936 | "oauth-sign": { 1937 | "version": "0.9.0", 1938 | "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", 1939 | "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==" 1940 | }, 1941 | "object-assign": { 1942 | "version": "4.1.1", 1943 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 1944 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" 1945 | }, 1946 | "object-inspect": { 1947 | "version": "1.7.0", 1948 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.7.0.tgz", 1949 | "integrity": "sha512-a7pEHdh1xKIAgTySUGgLMx/xwDZskN1Ud6egYYN3EdRW4ZMPNEDUTF+hwy2LUC+Bl+SyLXANnwz/jyh/qutKUw==" 1950 | }, 1951 | "object-keys": { 1952 | "version": "1.1.1", 1953 | "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", 1954 | "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==" 1955 | }, 1956 | "object-path": { 1957 | "version": "0.11.4", 1958 | "resolved": "https://registry.npmjs.org/object-path/-/object-path-0.11.4.tgz", 1959 | "integrity": "sha1-NwrnUvvzfePqcKhhwju6iRVpGUk=" 1960 | }, 1961 | "object.getownpropertydescriptors": { 1962 | "version": "2.0.3", 1963 | "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.0.3.tgz", 1964 | "integrity": "sha1-h1jIRvW0B62rDyNuCYbxSwUcqhY=", 1965 | "requires": { 1966 | "define-properties": "^1.1.2", 1967 | "es-abstract": "^1.5.1" 1968 | } 1969 | }, 1970 | "omise": { 1971 | "version": "0.6.0", 1972 | "resolved": "https://registry.npmjs.org/omise/-/omise-0.6.0.tgz", 1973 | "integrity": "sha512-ZQYOJhU3A+wajRqiXFRfCA7o9uK5zfJpbKxz/WI+LcUMMoUCuYdtHIg5eBn+L5wMdM+hpfE/hLZ7Fc5qXEWHOA==", 1974 | "requires": { 1975 | "@types/bluebird": "^3.5.15", 1976 | "@types/node": "^8.0.34", 1977 | "bluebird": "^2.9.12", 1978 | "https-proxy-agent": "^2.2.4", 1979 | "lodash": "^4.17.15" 1980 | }, 1981 | "dependencies": { 1982 | "@types/node": { 1983 | "version": "8.10.59", 1984 | "resolved": "https://registry.npmjs.org/@types/node/-/node-8.10.59.tgz", 1985 | "integrity": "sha512-8RkBivJrDCyPpBXhVZcjh7cQxVBSmRk9QM7hOketZzp6Tg79c0N8kkpAIito9bnJ3HCVCHVYz+KHTEbfQNfeVQ==" 1986 | }, 1987 | "bluebird": { 1988 | "version": "2.11.0", 1989 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-2.11.0.tgz", 1990 | "integrity": "sha1-U0uQM8AiyVecVro7Plpcqvu2UOE=" 1991 | } 1992 | } 1993 | }, 1994 | "on-finished": { 1995 | "version": "2.3.0", 1996 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 1997 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 1998 | "requires": { 1999 | "ee-first": "1.1.1" 2000 | } 2001 | }, 2002 | "p-finally": { 2003 | "version": "1.0.0", 2004 | "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", 2005 | "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=" 2006 | }, 2007 | "package-json": { 2008 | "version": "4.0.1", 2009 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-4.0.1.tgz", 2010 | "integrity": "sha1-iGmgQBJTZhxMTKPabCEh7VVfXu0=", 2011 | "requires": { 2012 | "got": "^6.7.1", 2013 | "registry-auth-token": "^3.0.1", 2014 | "registry-url": "^3.0.3", 2015 | "semver": "^5.1.0" 2016 | } 2017 | }, 2018 | "parseurl": { 2019 | "version": "1.3.3", 2020 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 2021 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 2022 | }, 2023 | "passport": { 2024 | "version": "0.4.1", 2025 | "resolved": "https://registry.npmjs.org/passport/-/passport-0.4.1.tgz", 2026 | "integrity": "sha512-IxXgZZs8d7uFSt3eqNjM9NQ3g3uQCW5avD8mRNoXV99Yig50vjuaez6dQK2qC0kVWPRTujxY0dWgGfT09adjYg==", 2027 | "requires": { 2028 | "passport-strategy": "1.x.x", 2029 | "pause": "0.0.1" 2030 | } 2031 | }, 2032 | "passport-facebook": { 2033 | "version": "3.0.0", 2034 | "resolved": "https://registry.npmjs.org/passport-facebook/-/passport-facebook-3.0.0.tgz", 2035 | "integrity": "sha512-K/qNzuFsFISYAyC1Nma4qgY/12V3RSLFdFVsPKXiKZt434wOvthFW1p7zKa1iQihQMRhaWorVE1o3Vi1o+ZgeQ==", 2036 | "requires": { 2037 | "passport-oauth2": "1.x.x" 2038 | } 2039 | }, 2040 | "passport-google-oauth20": { 2041 | "version": "2.0.0", 2042 | "resolved": "https://registry.npmjs.org/passport-google-oauth20/-/passport-google-oauth20-2.0.0.tgz", 2043 | "integrity": "sha512-KSk6IJ15RoxuGq7D1UKK/8qKhNfzbLeLrG3gkLZ7p4A6DBCcv7xpyQwuXtWdpyR0+E0mwkpjY1VfPOhxQrKzdQ==", 2044 | "requires": { 2045 | "passport-oauth2": "1.x.x" 2046 | } 2047 | }, 2048 | "passport-oauth2": { 2049 | "version": "1.5.0", 2050 | "resolved": "https://registry.npmjs.org/passport-oauth2/-/passport-oauth2-1.5.0.tgz", 2051 | "integrity": "sha512-kqBt6vR/5VlCK8iCx1/KpY42kQ+NEHZwsSyt4Y6STiNjU+wWICG1i8ucc1FapXDGO15C5O5VZz7+7vRzrDPXXQ==", 2052 | "requires": { 2053 | "base64url": "3.x.x", 2054 | "oauth": "0.9.x", 2055 | "passport-strategy": "1.x.x", 2056 | "uid2": "0.0.x", 2057 | "utils-merge": "1.x.x" 2058 | } 2059 | }, 2060 | "passport-strategy": { 2061 | "version": "1.0.0", 2062 | "resolved": "https://registry.npmjs.org/passport-strategy/-/passport-strategy-1.0.0.tgz", 2063 | "integrity": "sha1-tVOaqPwiWj0a0XlHbd8ja0QPUuQ=" 2064 | }, 2065 | "path-is-inside": { 2066 | "version": "1.0.2", 2067 | "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", 2068 | "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=" 2069 | }, 2070 | "path-key": { 2071 | "version": "2.0.1", 2072 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", 2073 | "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=" 2074 | }, 2075 | "path-to-regexp": { 2076 | "version": "0.1.7", 2077 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 2078 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 2079 | }, 2080 | "pause": { 2081 | "version": "0.0.1", 2082 | "resolved": "https://registry.npmjs.org/pause/-/pause-0.0.1.tgz", 2083 | "integrity": "sha1-HUCLP9t2kjuVQ9lvtMnf1TXZy10=" 2084 | }, 2085 | "performance-now": { 2086 | "version": "2.1.0", 2087 | "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", 2088 | "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=" 2089 | }, 2090 | "picomatch": { 2091 | "version": "2.1.1", 2092 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.1.1.tgz", 2093 | "integrity": "sha512-OYMyqkKzK7blWO/+XZYP6w8hH0LDvkBvdvKukti+7kqYFCiEAk+gI3DWnryapc0Dau05ugGTy0foQ6mqn4AHYA==" 2094 | }, 2095 | "pify": { 2096 | "version": "3.0.0", 2097 | "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", 2098 | "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=" 2099 | }, 2100 | "prepend-http": { 2101 | "version": "1.0.4", 2102 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz", 2103 | "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=" 2104 | }, 2105 | "proxy-addr": { 2106 | "version": "2.0.5", 2107 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.5.tgz", 2108 | "integrity": "sha512-t/7RxHXPH6cJtP0pRG6smSr9QJidhB+3kXu0KgXnbGYMgzEnUxRQ4/LDdfOwZEMyIh3/xHb8PX3t+lfL9z+YVQ==", 2109 | "requires": { 2110 | "forwarded": "~0.1.2", 2111 | "ipaddr.js": "1.9.0" 2112 | } 2113 | }, 2114 | "pseudomap": { 2115 | "version": "1.0.2", 2116 | "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", 2117 | "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=" 2118 | }, 2119 | "psl": { 2120 | "version": "1.7.0", 2121 | "resolved": "https://registry.npmjs.org/psl/-/psl-1.7.0.tgz", 2122 | "integrity": "sha512-5NsSEDv8zY70ScRnOTn7bK7eanl2MvFrOrS/R6x+dBt5g1ghnj9Zv90kO8GwT8gxcu2ANyFprnFYB85IogIJOQ==" 2123 | }, 2124 | "pstree.remy": { 2125 | "version": "1.1.7", 2126 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.7.tgz", 2127 | "integrity": "sha512-xsMgrUwRpuGskEzBFkH8NmTimbZ5PcPup0LA8JJkHIm2IMUbQcpo3yeLNWVrufEYjh8YwtSVh0xz6UeWc5Oh5A==" 2128 | }, 2129 | "punycode": { 2130 | "version": "2.1.1", 2131 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 2132 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==" 2133 | }, 2134 | "qs": { 2135 | "version": "6.7.0", 2136 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", 2137 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" 2138 | }, 2139 | "range-parser": { 2140 | "version": "1.2.1", 2141 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 2142 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 2143 | }, 2144 | "raw-body": { 2145 | "version": "2.4.0", 2146 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", 2147 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", 2148 | "requires": { 2149 | "bytes": "3.1.0", 2150 | "http-errors": "1.7.2", 2151 | "iconv-lite": "0.4.24", 2152 | "unpipe": "1.0.0" 2153 | }, 2154 | "dependencies": { 2155 | "http-errors": { 2156 | "version": "1.7.2", 2157 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 2158 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 2159 | "requires": { 2160 | "depd": "~1.1.2", 2161 | "inherits": "2.0.3", 2162 | "setprototypeof": "1.1.1", 2163 | "statuses": ">= 1.5.0 < 2", 2164 | "toidentifier": "1.0.0" 2165 | } 2166 | }, 2167 | "inherits": { 2168 | "version": "2.0.3", 2169 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 2170 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 2171 | } 2172 | } 2173 | }, 2174 | "rc": { 2175 | "version": "1.2.8", 2176 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", 2177 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", 2178 | "requires": { 2179 | "deep-extend": "^0.6.0", 2180 | "ini": "~1.3.0", 2181 | "minimist": "^1.2.0", 2182 | "strip-json-comments": "~2.0.1" 2183 | } 2184 | }, 2185 | "readdirp": { 2186 | "version": "3.2.0", 2187 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.2.0.tgz", 2188 | "integrity": "sha512-crk4Qu3pmXwgxdSgGhgA/eXiJAPQiX4GMOZZMXnqKxHX7TaoL+3gQVo/WeuAiogr07DpnfjIMpXXa+PAIvwPGQ==", 2189 | "requires": { 2190 | "picomatch": "^2.0.4" 2191 | } 2192 | }, 2193 | "regexp-clone": { 2194 | "version": "1.0.0", 2195 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz", 2196 | "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw==" 2197 | }, 2198 | "registry-auth-token": { 2199 | "version": "3.4.0", 2200 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-3.4.0.tgz", 2201 | "integrity": "sha512-4LM6Fw8eBQdwMYcES4yTnn2TqIasbXuwDx3um+QRs7S55aMKCBKBxvPXl2RiUjHwuJLTyYfxSpmfSAjQpcuP+A==", 2202 | "requires": { 2203 | "rc": "^1.1.6", 2204 | "safe-buffer": "^5.0.1" 2205 | } 2206 | }, 2207 | "registry-url": { 2208 | "version": "3.1.0", 2209 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-3.1.0.tgz", 2210 | "integrity": "sha1-PU74cPc93h138M+aOBQyRE4XSUI=", 2211 | "requires": { 2212 | "rc": "^1.0.1" 2213 | } 2214 | }, 2215 | "request": { 2216 | "version": "2.88.2", 2217 | "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz", 2218 | "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", 2219 | "requires": { 2220 | "aws-sign2": "~0.7.0", 2221 | "aws4": "^1.8.0", 2222 | "caseless": "~0.12.0", 2223 | "combined-stream": "~1.0.6", 2224 | "extend": "~3.0.2", 2225 | "forever-agent": "~0.6.1", 2226 | "form-data": "~2.3.2", 2227 | "har-validator": "~5.1.3", 2228 | "http-signature": "~1.2.0", 2229 | "is-typedarray": "~1.0.0", 2230 | "isstream": "~0.1.2", 2231 | "json-stringify-safe": "~5.0.1", 2232 | "mime-types": "~2.1.19", 2233 | "oauth-sign": "~0.9.0", 2234 | "performance-now": "^2.1.0", 2235 | "qs": "~6.5.2", 2236 | "safe-buffer": "^5.1.2", 2237 | "tough-cookie": "~2.5.0", 2238 | "tunnel-agent": "^0.6.0", 2239 | "uuid": "^3.3.2" 2240 | }, 2241 | "dependencies": { 2242 | "form-data": { 2243 | "version": "2.3.3", 2244 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", 2245 | "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", 2246 | "requires": { 2247 | "asynckit": "^0.4.0", 2248 | "combined-stream": "^1.0.6", 2249 | "mime-types": "^2.1.12" 2250 | } 2251 | }, 2252 | "qs": { 2253 | "version": "6.5.2", 2254 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", 2255 | "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==" 2256 | } 2257 | } 2258 | }, 2259 | "require_optional": { 2260 | "version": "1.0.1", 2261 | "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz", 2262 | "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==", 2263 | "requires": { 2264 | "resolve-from": "^2.0.0", 2265 | "semver": "^5.1.0" 2266 | } 2267 | }, 2268 | "resolve-from": { 2269 | "version": "2.0.0", 2270 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz", 2271 | "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c=" 2272 | }, 2273 | "retry": { 2274 | "version": "0.12.0", 2275 | "resolved": "https://registry.npmjs.org/retry/-/retry-0.12.0.tgz", 2276 | "integrity": "sha1-G0KmJmoh8HQh0bC1S33BZ7AcATs=" 2277 | }, 2278 | "safe-buffer": { 2279 | "version": "5.2.0", 2280 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.0.tgz", 2281 | "integrity": "sha512-fZEwUGbVl7kouZs1jCdMLdt95hdIv0ZeHg6L7qPeciMZhZ+/gdesW4wgTARkrFWEpspjEATAzUGPG8N2jJiwbg==" 2282 | }, 2283 | "safer-buffer": { 2284 | "version": "2.1.2", 2285 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 2286 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 2287 | }, 2288 | "saslprep": { 2289 | "version": "1.0.3", 2290 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz", 2291 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==", 2292 | "optional": true, 2293 | "requires": { 2294 | "sparse-bitfield": "^3.0.3" 2295 | } 2296 | }, 2297 | "semver": { 2298 | "version": "5.7.1", 2299 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 2300 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" 2301 | }, 2302 | "semver-diff": { 2303 | "version": "2.1.0", 2304 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-2.1.0.tgz", 2305 | "integrity": "sha1-S7uEN8jTfksM8aaP1ybsbWRdbTY=", 2306 | "requires": { 2307 | "semver": "^5.0.3" 2308 | } 2309 | }, 2310 | "send": { 2311 | "version": "0.17.1", 2312 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", 2313 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", 2314 | "requires": { 2315 | "debug": "2.6.9", 2316 | "depd": "~1.1.2", 2317 | "destroy": "~1.0.4", 2318 | "encodeurl": "~1.0.2", 2319 | "escape-html": "~1.0.3", 2320 | "etag": "~1.8.1", 2321 | "fresh": "0.5.2", 2322 | "http-errors": "~1.7.2", 2323 | "mime": "1.6.0", 2324 | "ms": "2.1.1", 2325 | "on-finished": "~2.3.0", 2326 | "range-parser": "~1.2.1", 2327 | "statuses": "~1.5.0" 2328 | }, 2329 | "dependencies": { 2330 | "ms": { 2331 | "version": "2.1.1", 2332 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 2333 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 2334 | } 2335 | } 2336 | }, 2337 | "serve-static": { 2338 | "version": "1.14.1", 2339 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", 2340 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", 2341 | "requires": { 2342 | "encodeurl": "~1.0.2", 2343 | "escape-html": "~1.0.3", 2344 | "parseurl": "~1.3.3", 2345 | "send": "0.17.1" 2346 | } 2347 | }, 2348 | "setprototypeof": { 2349 | "version": "1.1.1", 2350 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", 2351 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" 2352 | }, 2353 | "sha.js": { 2354 | "version": "2.4.11", 2355 | "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz", 2356 | "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==", 2357 | "requires": { 2358 | "inherits": "^2.0.1", 2359 | "safe-buffer": "^5.0.1" 2360 | } 2361 | }, 2362 | "shebang-command": { 2363 | "version": "1.2.0", 2364 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", 2365 | "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", 2366 | "requires": { 2367 | "shebang-regex": "^1.0.0" 2368 | } 2369 | }, 2370 | "shebang-regex": { 2371 | "version": "1.0.0", 2372 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", 2373 | "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=" 2374 | }, 2375 | "sift": { 2376 | "version": "7.0.1", 2377 | "resolved": "https://registry.npmjs.org/sift/-/sift-7.0.1.tgz", 2378 | "integrity": "sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g==" 2379 | }, 2380 | "signal-exit": { 2381 | "version": "3.0.2", 2382 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", 2383 | "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=" 2384 | }, 2385 | "sliced": { 2386 | "version": "1.0.1", 2387 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz", 2388 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E=" 2389 | }, 2390 | "sparse-bitfield": { 2391 | "version": "3.0.3", 2392 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", 2393 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=", 2394 | "optional": true, 2395 | "requires": { 2396 | "memory-pager": "^1.0.2" 2397 | } 2398 | }, 2399 | "sshpk": { 2400 | "version": "1.16.1", 2401 | "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", 2402 | "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", 2403 | "requires": { 2404 | "asn1": "~0.2.3", 2405 | "assert-plus": "^1.0.0", 2406 | "bcrypt-pbkdf": "^1.0.0", 2407 | "dashdash": "^1.12.0", 2408 | "ecc-jsbn": "~0.1.1", 2409 | "getpass": "^0.1.1", 2410 | "jsbn": "~0.1.0", 2411 | "safer-buffer": "^2.0.2", 2412 | "tweetnacl": "~0.14.0" 2413 | } 2414 | }, 2415 | "statuses": { 2416 | "version": "1.5.0", 2417 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", 2418 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" 2419 | }, 2420 | "streamsearch": { 2421 | "version": "0.1.2", 2422 | "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-0.1.2.tgz", 2423 | "integrity": "sha1-gIudDlb8Jz2Am6VzOOkpkZoanxo=" 2424 | }, 2425 | "string-width": { 2426 | "version": "2.1.1", 2427 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", 2428 | "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", 2429 | "requires": { 2430 | "is-fullwidth-code-point": "^2.0.0", 2431 | "strip-ansi": "^4.0.0" 2432 | } 2433 | }, 2434 | "string.prototype.trimleft": { 2435 | "version": "2.1.0", 2436 | "resolved": "https://registry.npmjs.org/string.prototype.trimleft/-/string.prototype.trimleft-2.1.0.tgz", 2437 | "integrity": "sha512-FJ6b7EgdKxxbDxc79cOlok6Afd++TTs5szo+zJTUyow3ycrRfJVE2pq3vcN53XexvKZu/DJMDfeI/qMiZTrjTw==", 2438 | "requires": { 2439 | "define-properties": "^1.1.3", 2440 | "function-bind": "^1.1.1" 2441 | } 2442 | }, 2443 | "string.prototype.trimright": { 2444 | "version": "2.1.0", 2445 | "resolved": "https://registry.npmjs.org/string.prototype.trimright/-/string.prototype.trimright-2.1.0.tgz", 2446 | "integrity": "sha512-fXZTSV55dNBwv16uw+hh5jkghxSnc5oHq+5K/gXgizHwAvMetdAJlHqqoFC1FSDVPYWLkAKl2cxpUT41sV7nSg==", 2447 | "requires": { 2448 | "define-properties": "^1.1.3", 2449 | "function-bind": "^1.1.1" 2450 | } 2451 | }, 2452 | "strip-ansi": { 2453 | "version": "4.0.0", 2454 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", 2455 | "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", 2456 | "requires": { 2457 | "ansi-regex": "^3.0.0" 2458 | } 2459 | }, 2460 | "strip-eof": { 2461 | "version": "1.0.0", 2462 | "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz", 2463 | "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=" 2464 | }, 2465 | "strip-json-comments": { 2466 | "version": "2.0.1", 2467 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 2468 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=" 2469 | }, 2470 | "subscriptions-transport-ws": { 2471 | "version": "0.9.16", 2472 | "resolved": "https://registry.npmjs.org/subscriptions-transport-ws/-/subscriptions-transport-ws-0.9.16.tgz", 2473 | "integrity": "sha512-pQdoU7nC+EpStXnCfh/+ho0zE0Z+ma+i7xvj7bkXKb1dvYHSZxgRPaU6spRP+Bjzow67c/rRDoix5RT0uU9omw==", 2474 | "requires": { 2475 | "backo2": "^1.0.2", 2476 | "eventemitter3": "^3.1.0", 2477 | "iterall": "^1.2.1", 2478 | "symbol-observable": "^1.0.4", 2479 | "ws": "^5.2.0" 2480 | }, 2481 | "dependencies": { 2482 | "ws": { 2483 | "version": "5.2.2", 2484 | "resolved": "https://registry.npmjs.org/ws/-/ws-5.2.2.tgz", 2485 | "integrity": "sha512-jaHFD6PFv6UgoIVda6qZllptQsMlDEJkTQcybzzXDYM1XO9Y8em691FGMPmM46WGyLU4z9KMgQN+qrux/nhlHA==", 2486 | "requires": { 2487 | "async-limiter": "~1.0.0" 2488 | } 2489 | } 2490 | } 2491 | }, 2492 | "supports-color": { 2493 | "version": "5.5.0", 2494 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 2495 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 2496 | "requires": { 2497 | "has-flag": "^3.0.0" 2498 | } 2499 | }, 2500 | "symbol-observable": { 2501 | "version": "1.2.0", 2502 | "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.2.0.tgz", 2503 | "integrity": "sha512-e900nM8RRtGhlV36KGEU9k65K3mPb1WV70OdjfxlG2EAuM1noi/E/BaW/uMhL7bPEssK8QV57vN3esixjUvcXQ==" 2504 | }, 2505 | "term-size": { 2506 | "version": "1.2.0", 2507 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-1.2.0.tgz", 2508 | "integrity": "sha1-RYuDiH8oj8Vtb/+/rSYuJmOO+mk=", 2509 | "requires": { 2510 | "execa": "^0.7.0" 2511 | } 2512 | }, 2513 | "timed-out": { 2514 | "version": "4.0.1", 2515 | "resolved": "https://registry.npmjs.org/timed-out/-/timed-out-4.0.1.tgz", 2516 | "integrity": "sha1-8y6srFoXW+ol1/q1Zas+2HQe9W8=" 2517 | }, 2518 | "to-regex-range": { 2519 | "version": "5.0.1", 2520 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 2521 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 2522 | "requires": { 2523 | "is-number": "^7.0.0" 2524 | } 2525 | }, 2526 | "toidentifier": { 2527 | "version": "1.0.0", 2528 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", 2529 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" 2530 | }, 2531 | "touch": { 2532 | "version": "3.1.0", 2533 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", 2534 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", 2535 | "requires": { 2536 | "nopt": "~1.0.10" 2537 | } 2538 | }, 2539 | "tough-cookie": { 2540 | "version": "2.5.0", 2541 | "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz", 2542 | "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==", 2543 | "requires": { 2544 | "psl": "^1.1.28", 2545 | "punycode": "^2.1.1" 2546 | } 2547 | }, 2548 | "ts-invariant": { 2549 | "version": "0.4.4", 2550 | "resolved": "https://registry.npmjs.org/ts-invariant/-/ts-invariant-0.4.4.tgz", 2551 | "integrity": "sha512-uEtWkFM/sdZvRNNDL3Ehu4WVpwaulhwQszV8mrtcdeE8nN00BV9mAmQ88RkrBhFgl9gMgvjJLAQcZbnPXI9mlA==", 2552 | "requires": { 2553 | "tslib": "^1.9.3" 2554 | } 2555 | }, 2556 | "tslib": { 2557 | "version": "1.10.0", 2558 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.10.0.tgz", 2559 | "integrity": "sha512-qOebF53frne81cf0S9B41ByenJ3/IuH8yJKngAX35CmiZySA0khhkovshKK+jGCaMnVomla7gVlIcc3EvKPbTQ==" 2560 | }, 2561 | "tunnel-agent": { 2562 | "version": "0.6.0", 2563 | "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", 2564 | "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", 2565 | "requires": { 2566 | "safe-buffer": "^5.0.1" 2567 | } 2568 | }, 2569 | "tweetnacl": { 2570 | "version": "0.14.5", 2571 | "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", 2572 | "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=" 2573 | }, 2574 | "type-is": { 2575 | "version": "1.6.18", 2576 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 2577 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 2578 | "requires": { 2579 | "media-typer": "0.3.0", 2580 | "mime-types": "~2.1.24" 2581 | } 2582 | }, 2583 | "uid2": { 2584 | "version": "0.0.3", 2585 | "resolved": "https://registry.npmjs.org/uid2/-/uid2-0.0.3.tgz", 2586 | "integrity": "sha1-SDEm4Rd03y9xuLY53NeZw3YWK4I=" 2587 | }, 2588 | "undefsafe": { 2589 | "version": "2.0.2", 2590 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.2.tgz", 2591 | "integrity": "sha1-Il9rngM3Zj4Njnz9aG/Cg2zKznY=", 2592 | "requires": { 2593 | "debug": "^2.2.0" 2594 | } 2595 | }, 2596 | "unique-string": { 2597 | "version": "1.0.0", 2598 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-1.0.0.tgz", 2599 | "integrity": "sha1-nhBXzKhRq7kzmPizOuGHuZyuwRo=", 2600 | "requires": { 2601 | "crypto-random-string": "^1.0.0" 2602 | } 2603 | }, 2604 | "unpipe": { 2605 | "version": "1.0.0", 2606 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 2607 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 2608 | }, 2609 | "unzip-response": { 2610 | "version": "2.0.1", 2611 | "resolved": "https://registry.npmjs.org/unzip-response/-/unzip-response-2.0.1.tgz", 2612 | "integrity": "sha1-0vD3N9FrBhXnKmk17QQhRXLVb5c=" 2613 | }, 2614 | "update-notifier": { 2615 | "version": "2.5.0", 2616 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-2.5.0.tgz", 2617 | "integrity": "sha512-gwMdhgJHGuj/+wHJJs9e6PcCszpxR1b236igrOkUofGhqJuG+amlIKwApH1IW1WWl7ovZxsX49lMBWLxSdm5Dw==", 2618 | "requires": { 2619 | "boxen": "^1.2.1", 2620 | "chalk": "^2.0.1", 2621 | "configstore": "^3.0.0", 2622 | "import-lazy": "^2.1.0", 2623 | "is-ci": "^1.0.10", 2624 | "is-installed-globally": "^0.1.0", 2625 | "is-npm": "^1.0.0", 2626 | "latest-version": "^3.0.0", 2627 | "semver-diff": "^2.0.0", 2628 | "xdg-basedir": "^3.0.0" 2629 | } 2630 | }, 2631 | "uri-js": { 2632 | "version": "4.2.2", 2633 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz", 2634 | "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==", 2635 | "requires": { 2636 | "punycode": "^2.1.0" 2637 | } 2638 | }, 2639 | "url-parse-lax": { 2640 | "version": "1.0.0", 2641 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-1.0.0.tgz", 2642 | "integrity": "sha1-evjzA2Rem9eaJy56FKxovAYJ2nM=", 2643 | "requires": { 2644 | "prepend-http": "^1.0.1" 2645 | } 2646 | }, 2647 | "util.promisify": { 2648 | "version": "1.0.0", 2649 | "resolved": "https://registry.npmjs.org/util.promisify/-/util.promisify-1.0.0.tgz", 2650 | "integrity": "sha512-i+6qA2MPhvoKLuxnJNpXAGhg7HphQOSUq2LKMZD0m15EiskXUkMvKdF4Uui0WYeCUGea+o2cw/ZuwehtfsrNkA==", 2651 | "requires": { 2652 | "define-properties": "^1.1.2", 2653 | "object.getownpropertydescriptors": "^2.0.3" 2654 | } 2655 | }, 2656 | "utils-merge": { 2657 | "version": "1.0.1", 2658 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 2659 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 2660 | }, 2661 | "uuid": { 2662 | "version": "3.3.3", 2663 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.3.3.tgz", 2664 | "integrity": "sha512-pW0No1RGHgzlpHJO1nsVrHKpOEIxkGg1xB+v0ZmdNH5OAeAwzAVrCnI2/6Mtx+Uys6iaylxa+D3g4j63IKKjSQ==" 2665 | }, 2666 | "vary": { 2667 | "version": "1.1.2", 2668 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 2669 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 2670 | }, 2671 | "verror": { 2672 | "version": "1.10.0", 2673 | "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", 2674 | "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", 2675 | "requires": { 2676 | "assert-plus": "^1.0.0", 2677 | "core-util-is": "1.0.2", 2678 | "extsprintf": "^1.2.0" 2679 | } 2680 | }, 2681 | "which": { 2682 | "version": "1.3.1", 2683 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", 2684 | "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", 2685 | "requires": { 2686 | "isexe": "^2.0.0" 2687 | } 2688 | }, 2689 | "widest-line": { 2690 | "version": "2.0.1", 2691 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-2.0.1.tgz", 2692 | "integrity": "sha512-Ba5m9/Fa4Xt9eb2ELXt77JxVDV8w7qQrH0zS/TWSJdLyAwQjWoOzpzj5lwVftDz6n/EOu3tNACS84v509qwnJA==", 2693 | "requires": { 2694 | "string-width": "^2.1.1" 2695 | } 2696 | }, 2697 | "write-file-atomic": { 2698 | "version": "2.4.3", 2699 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-2.4.3.tgz", 2700 | "integrity": "sha512-GaETH5wwsX+GcnzhPgKcKjJ6M2Cq3/iZp1WyY/X1CSqrW+jVNM9Y7D8EC2sM4ZG/V8wZlSniJnCKWPmBYAucRQ==", 2701 | "requires": { 2702 | "graceful-fs": "^4.1.11", 2703 | "imurmurhash": "^0.1.4", 2704 | "signal-exit": "^3.0.2" 2705 | } 2706 | }, 2707 | "ws": { 2708 | "version": "6.2.1", 2709 | "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz", 2710 | "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==", 2711 | "requires": { 2712 | "async-limiter": "~1.0.0" 2713 | } 2714 | }, 2715 | "xdg-basedir": { 2716 | "version": "3.0.0", 2717 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-3.0.0.tgz", 2718 | "integrity": "sha1-SWsswQnsqNus/i3HK2A8F8WHCtQ=" 2719 | }, 2720 | "yallist": { 2721 | "version": "3.1.1", 2722 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", 2723 | "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==" 2724 | }, 2725 | "zen-observable": { 2726 | "version": "0.8.15", 2727 | "resolved": "https://registry.npmjs.org/zen-observable/-/zen-observable-0.8.15.tgz", 2728 | "integrity": "sha512-PQ2PC7R9rslx84ndNBZB/Dkv8V8fZEpk83RLgXtYd0fwUgEjseMn1Dgajh2x6S8QbZAFa9p2qVCEuYZNgve0dQ==" 2729 | }, 2730 | "zen-observable-ts": { 2731 | "version": "0.8.20", 2732 | "resolved": "https://registry.npmjs.org/zen-observable-ts/-/zen-observable-ts-0.8.20.tgz", 2733 | "integrity": "sha512-2rkjiPALhOtRaDX6pWyNqK1fnP5KkJJybYebopNSn6wDG1lxBoFs2+nwwXKoA6glHIrtwrfBBy6da0stkKtTAA==", 2734 | "requires": { 2735 | "tslib": "^1.9.3", 2736 | "zen-observable": "^0.8.0" 2737 | } 2738 | } 2739 | } 2740 | } 2741 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "graphql-basic", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "nodemon -r esm src/index.js" 8 | }, 9 | "keywords": [], 10 | "author": "", 11 | "license": "ISC", 12 | "dependencies": { 13 | "@sendgrid/mail": "^6.5.2", 14 | "apollo-server-express": "^2.9.13", 15 | "bcryptjs": "^2.4.3", 16 | "dotenv": "^8.2.0", 17 | "esm": "^3.2.25", 18 | "express": "^4.17.1", 19 | "graphql": "^14.5.8", 20 | "graphql-iso-date": "^3.6.1", 21 | "jsonwebtoken": "^8.5.1", 22 | "mongoose": "^5.8.1", 23 | "nodemon": "^2.0.2", 24 | "omise": "^0.6.0", 25 | "passport": "^0.4.1", 26 | "passport-facebook": "^3.0.0", 27 | "passport-google-oauth20": "^2.0.0" 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | import dotenv from 'dotenv' 2 | dotenv.config() 3 | import express from 'express' 4 | import mongoose from 'mongoose' 5 | import passport from 'passport' 6 | 7 | import server from './server' 8 | import { facebookPassportConfig, googlePassportConfig } from './utils/passport' 9 | import { facebookAuth, googleAuth } from './utils/socialProvidersAuth' 10 | 11 | const { DB_USER, DB_PASSWORD, DB_NAME, PORT } = process.env 12 | 13 | facebookPassportConfig() 14 | googlePassportConfig() 15 | 16 | const createServer = async () => { 17 | try { 18 | await mongoose.connect( 19 | `mongodb+srv://${DB_USER}:${DB_PASSWORD}@graphql-basic-o1icg.mongodb.net/${DB_NAME}?retryWrites=true&w=majority`, 20 | { useUnifiedTopology: true } 21 | ) 22 | 23 | const app = express() 24 | 25 | app.get('/auth/facebook', passport.authenticate('facebook')) 26 | 27 | app.get( 28 | '/auth/facebook/callback', 29 | passport.authenticate('facebook', { 30 | session: false, 31 | failureRedirect: 'http://localhost:3000/signin', 32 | }), 33 | facebookAuth 34 | ) 35 | 36 | app.get( 37 | '/auth/google', 38 | passport.authenticate('google', { scope: ['profile', 'email'] }) 39 | ) 40 | 41 | app.get( 42 | '/auth/google/callback', 43 | passport.authenticate('google', { 44 | session: false, 45 | failureRedirect: 'http://localhost:3000/signin', 46 | }), 47 | googleAuth 48 | ) 49 | 50 | server.applyMiddleware({ app }) 51 | 52 | app.listen({ port: PORT }, () => 53 | console.log( 54 | `🚀 Server ready at http://localhost:${PORT}${server.graphqlPath}` 55 | ) 56 | ) 57 | } catch (error) { 58 | console.log(error) 59 | } 60 | } 61 | 62 | createServer() 63 | -------------------------------------------------------------------------------- /src/models/cartItem.js: -------------------------------------------------------------------------------- 1 | import mongoose from "mongoose" 2 | 3 | const cartItemSchema = new mongoose.Schema({ 4 | product: { 5 | type: mongoose.Schema.Types.ObjectId, 6 | ref: "Product" 7 | }, 8 | quantity: { 9 | type: Number, 10 | required: true 11 | }, 12 | user: { 13 | type: mongoose.Schema.Types.ObjectId, 14 | ref: "User" 15 | }, 16 | createdAt: { 17 | type: Date, 18 | required: true, 19 | default: () => Date.now() 20 | } 21 | }) 22 | 23 | const CartItem = mongoose.model("CartItem", cartItemSchema) 24 | 25 | export default CartItem 26 | -------------------------------------------------------------------------------- /src/models/order.js: -------------------------------------------------------------------------------- 1 | import mongoose from 'mongoose' 2 | 3 | const orderSchema = new mongoose.Schema({ 4 | user: { 5 | type: mongoose.Schema.Types.ObjectId, 6 | ref: 'User' 7 | }, 8 | items: [ 9 | { 10 | type: mongoose.Schema.Types.ObjectId, 11 | ref: 'OrderItem' 12 | } 13 | ], 14 | authorize_uri: { 15 | type: String 16 | }, 17 | createdAt: { 18 | type: Date, 19 | required: true, 20 | default: () => Date.now() 21 | } 22 | }) 23 | 24 | const Order = mongoose.model('Order', orderSchema) 25 | 26 | export default Order 27 | -------------------------------------------------------------------------------- /src/models/orderItem.js: -------------------------------------------------------------------------------- 1 | import mongoose from 'mongoose' 2 | 3 | const orderItemSchema = new mongoose.Schema({ 4 | product: { 5 | type: mongoose.Schema.Types.ObjectId, 6 | ref: 'Product' 7 | }, 8 | quantity: { 9 | type: Number, 10 | required: true 11 | }, 12 | user: { 13 | type: mongoose.Schema.Types.ObjectId, 14 | ref: 'User' 15 | }, 16 | createdAt: { 17 | type: Date, 18 | required: true, 19 | default: () => Date.now() 20 | } 21 | }) 22 | 23 | const OrderItem = mongoose.model('OrderItem', orderItemSchema) 24 | 25 | export default OrderItem 26 | -------------------------------------------------------------------------------- /src/models/product.js: -------------------------------------------------------------------------------- 1 | import mongoose from "mongoose" 2 | 3 | const productSchema = new mongoose.Schema({ 4 | description: { 5 | type: String, 6 | required: true, 7 | trim: true 8 | }, 9 | price: { 10 | type: Number, 11 | required: true 12 | }, 13 | imageUrl: { 14 | type: String, 15 | required: true, 16 | trim: true 17 | }, 18 | user: { 19 | type: mongoose.Schema.Types.ObjectId, 20 | ref: "User", 21 | required: true 22 | }, 23 | createdAt: { 24 | type: Date, 25 | required: true, 26 | default: () => Date.now() 27 | } 28 | }) 29 | 30 | const Product = mongoose.model("Product", productSchema) 31 | 32 | export default Product 33 | -------------------------------------------------------------------------------- /src/models/user.js: -------------------------------------------------------------------------------- 1 | import mongoose from 'mongoose' 2 | 3 | const userSchema = new mongoose.Schema({ 4 | name: { 5 | type: String, 6 | required: true, 7 | }, 8 | email: { 9 | type: String, 10 | required: true, 11 | lowercase: true, 12 | trim: true, 13 | }, 14 | password: { 15 | type: String, 16 | required: true, 17 | trim: true, 18 | }, 19 | providerId: { 20 | type: String, 21 | }, 22 | resetPasswordToken: { 23 | type: String, 24 | }, 25 | resetTokenExpiry: { 26 | type: Number, 27 | }, 28 | products: [ 29 | { 30 | type: mongoose.Schema.Types.ObjectId, 31 | ref: 'Product', 32 | }, 33 | ], 34 | carts: [ 35 | { 36 | type: mongoose.Schema.Types.ObjectId, 37 | ref: 'CartItem', 38 | }, 39 | ], 40 | orders: [ 41 | { 42 | type: mongoose.Schema.Types.ObjectId, 43 | ref: 'Order', 44 | }, 45 | ], 46 | cards: [ 47 | { 48 | id: String, 49 | cardInfo: { 50 | id: String, 51 | expiration_month: Number, 52 | expiration_year: Number, 53 | brand: String, 54 | last_digits: String, 55 | }, 56 | }, 57 | ], 58 | createdAt: { 59 | type: Date, 60 | required: true, 61 | default: () => Date.now(), 62 | }, 63 | }) 64 | 65 | const User = mongoose.model('User', userSchema) 66 | 67 | export default User 68 | -------------------------------------------------------------------------------- /src/resolvers/index.js: -------------------------------------------------------------------------------- 1 | import { GraphQLDateTime } from "graphql-iso-date" 2 | 3 | import Query from "./query" 4 | import Mutation from "./mutation" 5 | 6 | const resolvers = { 7 | Query, 8 | Mutation, 9 | Date: GraphQLDateTime 10 | } 11 | 12 | export default resolvers 13 | -------------------------------------------------------------------------------- /src/resolvers/mutation.js: -------------------------------------------------------------------------------- 1 | import { randomBytes } from 'crypto' 2 | import bcrypt from 'bcryptjs' 3 | import jwt from 'jsonwebtoken' 4 | import sgMail from '@sendgrid/mail' 5 | 6 | import User from '../models/user' 7 | import Product from '../models/product' 8 | import CartItem from '../models/cartItem' 9 | import { 10 | retrieveCustomer, 11 | createCustomer, 12 | createCharge, 13 | createChargeInternetBanking 14 | } from '../utils/omiseUtils' 15 | import OrderItem from '../models/orderItem' 16 | import Order from '../models/order' 17 | 18 | const Mutation = { 19 | signup: async (parent, args, context, info) => { 20 | // Trim and lower case email 21 | const email = args.email.trim().toLowerCase() 22 | 23 | // Check if email already exist in database 24 | const currentUsers = await User.find({}) 25 | const isEmailExist = 26 | currentUsers.findIndex(user => user.email === email) > -1 27 | 28 | if (isEmailExist) { 29 | throw new Error('Email already exist.') 30 | } 31 | 32 | // Validate password 33 | if (args.password.trim().length < 6) { 34 | throw new Error('Password must be at least 6 characters.') 35 | } 36 | 37 | const password = await bcrypt.hash(args.password, 10) 38 | 39 | return User.create({ ...args, email, password }) 40 | }, 41 | login: async (parent, args, context, info) => { 42 | const { email, password } = args 43 | 44 | // Find user in database 45 | const user = await User.findOne({ email }) 46 | .populate({ 47 | path: 'products', 48 | populate: { path: 'user' } 49 | }) 50 | .populate({ path: 'carts', populate: { path: 'product' } }) 51 | .populate({ 52 | path: 'orders', 53 | options: { sort: { createdAt: 'desc' } }, 54 | populate: { path: 'items', populate: { path: 'product' } } 55 | }) 56 | 57 | if (!user) throw new Error('Email not found, please sign up.') 58 | 59 | // Check if password is correct 60 | const validPassword = await bcrypt.compare(password, user.password) 61 | 62 | if (!validPassword) throw new Error('Invalid email or password.') 63 | 64 | const token = jwt.sign({ userId: user.id }, process.env.SECRET, { 65 | expiresIn: '7days' 66 | }) 67 | 68 | return { user, jwt: token } 69 | }, 70 | requestResetPassword: async (parent, { email }, context, info) => { 71 | // 1. Find user in database by email 72 | const user = await User.findOne({ email }) 73 | 74 | // 2. If no user found, throw error 75 | if (!user) throw new Error('Email not found, please sign up instead.') 76 | 77 | // 3. Create resetPasswordToken and resetTokenExpiry 78 | const resetPasswordToken = randomBytes(32).toString('hex') 79 | const resetTokenExpiry = Date.now() + 30 * 60 * 1000 80 | 81 | // 4. Update user (save reset token and token expiry) 82 | await User.findByIdAndUpdate(user.id, { 83 | resetPasswordToken, 84 | resetTokenExpiry 85 | }) 86 | 87 | // 5. Send link for set password to user email 88 | sgMail.setApiKey(process.env.EMAIL_API_KEY) 89 | 90 | const message = { 91 | from: 'graphql_basic@test.com', 92 | to: user.email, 93 | subject: 'Reset password link', 94 | html: ` 95 |
96 |

Please click the link below to reset your password.

\n\n 97 | Click to reset your password 98 |
99 | ` 100 | } 101 | 102 | sgMail.send(message) 103 | 104 | // 6. Return message to frontend 105 | return { message: 'Please check your email to proceed reset password.' } 106 | }, 107 | resetPassword: async (parent, { password, token }, context, info) => { 108 | // Find user in database by reset token 109 | const user = await User.findOne({ resetPasswordToken: token }) 110 | 111 | // If no user found throw error 112 | if (!user) throw new Error('Invalid token, cannot reset password.') 113 | 114 | // Check if token is expired 115 | const isTokenExpired = user.resetTokenExpiry < Date.now() 116 | 117 | // If token is expired throw error 118 | if (isTokenExpired) throw new Error('Invalid token, cannot reset password.') 119 | 120 | // Hash new password 121 | const hashedPassword = await bcrypt.hash(password, 10) 122 | 123 | // Update user in database (save new hashed password, delete reset token and token expiry time) 124 | await User.findByIdAndUpdate(user.id, { 125 | password: hashedPassword, 126 | resetPasswordToken: null, 127 | resetTokenExpiry: null 128 | }) 129 | 130 | // return message 131 | return { 132 | message: 'You have successfully reset your password, please sign in.' 133 | } 134 | }, 135 | createProduct: async (parent, args, { userId }, info) => { 136 | // const userId = "5e132cabae30211b84ad5d4f" 137 | 138 | // Check if user logged in 139 | if (!userId) throw new Error('Please log in.') 140 | 141 | if (!args.description || !args.price || !args.imageUrl) { 142 | throw new Error('Please provide all required fields.') 143 | } 144 | 145 | const product = await Product.create({ ...args, user: userId }) 146 | const user = await User.findById(userId) 147 | 148 | if (!user.products) { 149 | user.products = [product] 150 | } else { 151 | user.products.push(product) 152 | } 153 | 154 | await user.save() 155 | 156 | return Product.findById(product.id).populate({ 157 | path: 'user', 158 | populate: { path: 'products' } 159 | }) 160 | }, 161 | updateProduct: async (parent, args, { userId }, info) => { 162 | const { id, description, price, imageUrl } = args 163 | 164 | // TODO: Check if user logged in 165 | if (!userId) throw new Error('Please log in.') 166 | 167 | // Find product in database 168 | const product = await Product.findById(id) 169 | 170 | // TODO: Check if user is the owner of the product 171 | // const userId = "5e132cabae30211b84ad5d4f" 172 | 173 | if (userId !== product.user.toString()) { 174 | throw new Error('You are not authorized.') 175 | } 176 | 177 | // Form updated information 178 | const updateInfo = { 179 | description: !!description ? description : product.description, 180 | price: !!price ? price : product.price, 181 | imageUrl: !!imageUrl ? imageUrl : product.imageUrl 182 | } 183 | 184 | // Update product in database 185 | await Product.findByIdAndUpdate(id, updateInfo) 186 | 187 | // Find the updated Product 188 | const updatedProduct = await Product.findById(id).populate({ path: 'user' }) 189 | 190 | return updatedProduct 191 | }, 192 | addToCart: async (parent, args, { userId }, info) => { 193 | // id --> productId 194 | const { id } = args 195 | 196 | if (!userId) throw new Error('Please log in.') 197 | 198 | try { 199 | // Find user who perform add to cart --> from logged in 200 | // const userId = "5e15cb313cc0bd1270a2180d" 201 | 202 | // Check if the new addToCart item is already in user.carts 203 | const user = await User.findById(userId).populate({ 204 | path: 'carts', 205 | populate: { path: 'product' } 206 | }) 207 | 208 | const findCartItemIndex = user.carts.findIndex( 209 | cartItem => cartItem.product.id === id 210 | ) 211 | 212 | if (findCartItemIndex > -1) { 213 | // A. The new addToCart item is already in cart 214 | // A.1 Find the cartItem and update in database 215 | user.carts[findCartItemIndex].quantity += 1 216 | 217 | await CartItem.findByIdAndUpdate(user.carts[findCartItemIndex].id, { 218 | quantity: user.carts[findCartItemIndex].quantity 219 | }) 220 | 221 | // A.2 Find updated cartItem 222 | const updatedCartItem = await CartItem.findById( 223 | user.carts[findCartItemIndex].id 224 | ) 225 | .populate({ path: 'product' }) 226 | .populate({ path: 'user' }) 227 | 228 | return updatedCartItem 229 | } else { 230 | // B. The new addToCart item is not in cart yet 231 | // B.1 Create new cartItem 232 | const cartItem = await CartItem.create({ 233 | product: id, 234 | quantity: 1, 235 | user: userId 236 | }) 237 | 238 | // B.2 find new cartItem 239 | const newCartItem = await CartItem.findById(cartItem.id) 240 | .populate({ path: 'product' }) 241 | .populate({ path: 'user' }) 242 | 243 | // B.2 Update user.carts 244 | await User.findByIdAndUpdate(userId, { 245 | carts: [...user.carts, newCartItem] 246 | }) 247 | 248 | return newCartItem 249 | } 250 | } catch (error) { 251 | console.log(error) 252 | } 253 | }, 254 | deleteCart: async (parent, args, { userId }, info) => { 255 | const { id } = args 256 | 257 | // TODO: Check if user logged in 258 | if (!userId) throw new Error('Please log in.') 259 | 260 | // Find cart from given id 261 | const cart = await CartItem.findById(id) 262 | 263 | // TODO: user id from request --> Find user 264 | // const userId = "5e15cb313cc0bd1270a2180d" 265 | 266 | const user = await User.findById(userId) 267 | 268 | // Check ownership of the cart 269 | if (cart.user.toString() !== userId) { 270 | throw new Error('Not authorized.') 271 | } 272 | 273 | // Delete cart 274 | const deletedCart = await CartItem.findByIdAndRemove(id) 275 | 276 | // Update user's carts 277 | const updatedUserCarts = user.carts.filter( 278 | cartId => cartId.toString() !== deletedCart.id.toString() 279 | ) 280 | 281 | await User.findByIdAndUpdate(userId, { carts: updatedUserCarts }) 282 | 283 | return deletedCart 284 | }, 285 | createOrder: async ( 286 | parent, 287 | { amount, cardId, token, return_uri }, 288 | { userId }, 289 | info 290 | ) => { 291 | // Check if user logged in 292 | if (!userId) throw new Error('Please log in.') 293 | 294 | // Query user from the database 295 | const user = await User.findById(userId).populate({ 296 | path: 'carts', 297 | populate: { path: 'product' } 298 | }) 299 | 300 | // Create charge with omise 301 | let customer 302 | 303 | // Credit Card: User use existing card 304 | if (amount && cardId && !token && !return_uri) { 305 | const cust = await retrieveCustomer(cardId) 306 | 307 | if (!cust) throw new Error('Cannot process payment') 308 | 309 | customer = cust 310 | } 311 | 312 | // Credit Card: User use new card 313 | if (amount && token && !cardId && !return_uri) { 314 | const newCustomer = await createCustomer(user.email, user.name, token) 315 | 316 | if (!newCustomer) throw new Error('Cannot process payment') 317 | 318 | customer = newCustomer 319 | 320 | // update user'cards field 321 | const { 322 | id, 323 | expiration_month, 324 | expiration_year, 325 | brand, 326 | last_digits 327 | } = newCustomer.cards.data[0] 328 | 329 | const newCard = { 330 | id: newCustomer.id, 331 | cardInfo: { 332 | id, 333 | expiration_month, 334 | expiration_year, 335 | brand, 336 | last_digits 337 | } 338 | } 339 | 340 | await User.findByIdAndUpdate(userId, { cards: [newCard, ...user.cards] }) 341 | } 342 | 343 | let charge 344 | 345 | if (token && return_uri) { 346 | // Internet Banking 347 | charge = await createChargeInternetBanking(amount, token, return_uri) 348 | } else { 349 | // Credit Card 350 | charge = await createCharge(amount, customer.id) 351 | } 352 | 353 | if (!charge) 354 | throw new Error('Something went wrong with payment, please try again.') 355 | 356 | // Convert cartItem to OrderItem 357 | const convertCartToOrder = async () => { 358 | return Promise.all( 359 | user.carts.map(cart => 360 | OrderItem.create({ 361 | product: cart.product, 362 | quantity: cart.quantity, 363 | user: cart.user 364 | }) 365 | ) 366 | ) 367 | } 368 | 369 | // Create order 370 | const orderItemArray = await convertCartToOrder() 371 | 372 | const order = await Order.create({ 373 | user: userId, 374 | items: orderItemArray.map(orderItem => orderItem.id), 375 | authorize_uri: charge.authorize_uri 376 | }) 377 | 378 | // Delete cartItem from the database 379 | const deleteCartItems = async () => { 380 | return Promise.all( 381 | user.carts.map(cart => CartItem.findByIdAndRemove(cart.id)) 382 | ) 383 | } 384 | 385 | await deleteCartItems() 386 | 387 | // Update user info in the database 388 | await User.findByIdAndUpdate(userId, { 389 | carts: [], 390 | orders: !user.orders ? [order.id] : [...user.orders, order.id] 391 | }) 392 | 393 | // return order 394 | return Order.findById(order.id) 395 | .populate({ path: 'user' }) 396 | .populate({ path: 'items', populate: { path: 'product' } }) 397 | } 398 | } 399 | 400 | export default Mutation 401 | -------------------------------------------------------------------------------- /src/resolvers/query.js: -------------------------------------------------------------------------------- 1 | import User from '../models/user' 2 | import Product from '../models/product' 3 | 4 | const Query = { 5 | user: (parent, args, { userId }, info) => { 6 | // Check if user logged in 7 | if (!userId) throw new Error('Please log in') 8 | 9 | return User.findById(userId) 10 | .populate({ 11 | path: 'products', 12 | options: { sort: { createdAt: 'desc' } }, 13 | populate: { path: 'user' } 14 | }) 15 | .populate({ path: 'carts', populate: { path: 'product' } }) 16 | .populate({ 17 | path: 'orders', 18 | options: { sort: { createdAt: 'desc' } }, 19 | populate: { path: 'items', populate: { path: 'product' } } 20 | }) 21 | }, 22 | users: (parent, args, context, info) => 23 | User.find({}) 24 | .populate({ 25 | path: 'products', 26 | populate: { path: 'user' } 27 | }) 28 | .populate({ path: 'carts', populate: { path: 'product' } }), 29 | product: (parent, args, context, info) => 30 | Product.findById(args.id).populate({ 31 | path: 'user', 32 | populate: { path: 'products' } 33 | }), 34 | products: (parent, args, context, info) => 35 | Product.find() 36 | .populate({ 37 | path: 'user', 38 | populate: { path: 'products' } 39 | }) 40 | .sort({ createdAt: 'desc' }) 41 | } 42 | 43 | export default Query 44 | -------------------------------------------------------------------------------- /src/schema/schema.graphql: -------------------------------------------------------------------------------- 1 | type Query { 2 | user: User 3 | users: [User]! 4 | product(id: ID!): Product 5 | products: [Product]! 6 | } 7 | 8 | type Mutation { 9 | signup(name: String!, email: String!, password: String!): User 10 | login(email: String!, password: String!): AuthData 11 | requestResetPassword(email: String!): Message! 12 | resetPassword(password: String!, token: String!): Message 13 | createProduct( 14 | description: String! 15 | price: Float! 16 | imageUrl: String! 17 | ): Product! 18 | updateProduct( 19 | id: ID! 20 | description: String 21 | price: Float 22 | imageUrl: String 23 | ): Product! 24 | addToCart(id: ID!): CartItem! 25 | deleteCart(id: ID!): CartItem! 26 | createOrder( 27 | amount: Float! 28 | cardId: String 29 | token: String 30 | return_uri: String 31 | ): Order 32 | } 33 | 34 | scalar Date 35 | 36 | type User { 37 | id: ID! 38 | name: String! 39 | email: String! 40 | products: [Product] 41 | carts: [CartItem]! 42 | orders: [Order]! 43 | cards: [Card]! 44 | createdAt: Date! 45 | } 46 | 47 | type Product { 48 | id: ID! 49 | description: String! 50 | price: Float! 51 | imageUrl: String! 52 | user: User! 53 | createdAt: Date! 54 | } 55 | 56 | type CartItem { 57 | id: ID! 58 | product: Product! 59 | quantity: Int! 60 | user: User! 61 | createdAt: Date! 62 | } 63 | 64 | type OrderItem { 65 | id: ID! 66 | product: Product! 67 | quantity: Int! 68 | user: User! 69 | createdAt: Date! 70 | } 71 | 72 | type Order { 73 | id: ID! 74 | user: User! 75 | items: [OrderItem!]! 76 | authorize_uri: String 77 | createdAt: Date! 78 | } 79 | 80 | type Card { 81 | id: ID! 82 | cardInfo: CardInfo 83 | } 84 | 85 | type CardInfo { 86 | id: ID! 87 | expiration_month: Int! 88 | expiration_year: Int! 89 | brand: String! 90 | last_digits: String! 91 | } 92 | 93 | type AuthData { 94 | user: User 95 | jwt: String 96 | } 97 | 98 | type Message { 99 | message: String! 100 | } 101 | -------------------------------------------------------------------------------- /src/schema/typeDefs.js: -------------------------------------------------------------------------------- 1 | import { gql } from "apollo-server-express" 2 | 3 | const typeDefs = gql` 4 | type Query { 5 | me: User! 6 | user(id: ID!): User 7 | users: [User]! 8 | } 9 | 10 | type User { 11 | id: ID! 12 | name: String! 13 | } 14 | ` 15 | 16 | export default typeDefs 17 | -------------------------------------------------------------------------------- /src/server.js: -------------------------------------------------------------------------------- 1 | import fs from "fs" 2 | import path from "path" 3 | import { ApolloServer } from "apollo-server-express" 4 | 5 | // import typeDefs from "./schema/typeDefs" 6 | import resolvers from "./resolvers" 7 | import getUser from "./utils/getUser" 8 | 9 | const typeDefs = fs 10 | .readFileSync(path.join(__dirname, "./schema", "schema.graphql"), "utf8") 11 | .toString() 12 | 13 | const server = new ApolloServer({ 14 | typeDefs, 15 | resolvers, 16 | context: ({ req }) => { 17 | // Check token from headers 18 | const token = req.headers.authorization || "" 19 | 20 | // Extract userId from token 21 | const userId = getUser(token) 22 | 23 | return { userId } 24 | } 25 | }) 26 | 27 | export default server 28 | -------------------------------------------------------------------------------- /src/utils/getUser.js: -------------------------------------------------------------------------------- 1 | import jwt from "jsonwebtoken" 2 | 3 | const getUser = token => { 4 | if (!token) return null 5 | 6 | // 'Bearer zbowltobla..' -> [Bearer, zobaohooh] 7 | const parsedToken = token.split(" ")[1] 8 | 9 | try { 10 | const decodedToken = jwt.verify(parsedToken, process.env.SECRET) 11 | 12 | return decodedToken.userId 13 | } catch (error) { 14 | return null 15 | } 16 | } 17 | 18 | export default getUser 19 | -------------------------------------------------------------------------------- /src/utils/omiseUtils.js: -------------------------------------------------------------------------------- 1 | import dotenv from 'dotenv' 2 | dotenv.config() 3 | import OmiseFn from 'omise' 4 | 5 | const omise = OmiseFn({ 6 | publicKey: process.env.OMISE_PUBLIC_KEY, 7 | secretKey: process.env.OMISE_SECRET_KEY 8 | }) 9 | 10 | export const retrieveCustomer = id => { 11 | if (!id) return null 12 | 13 | return new Promise((resolve, reject) => { 14 | omise.customers.retrieve(id, function(err, res) { 15 | if (res) { 16 | resolve(res) 17 | } else { 18 | resolve(null) 19 | } 20 | }) 21 | }) 22 | } 23 | 24 | export const createCustomer = (email, description, card) => { 25 | return new Promise((resolve, reject) => { 26 | omise.customers.create({ email, description, card }, function(err, res) { 27 | if (res) { 28 | resolve(res) 29 | } else { 30 | resolve(null) 31 | } 32 | }) 33 | }) 34 | } 35 | 36 | export const createCharge = (amount, customer) => { 37 | return new Promise((resolve, reject) => { 38 | omise.charges.create({ amount, currency: 'thb', customer }, function( 39 | err, 40 | res 41 | ) { 42 | if (res) { 43 | resolve(res) 44 | } else { 45 | resolve(null) 46 | } 47 | }) 48 | }) 49 | } 50 | 51 | export const createChargeInternetBanking = (amount, source, return_uri) => { 52 | return new Promise((resolve, reject) => { 53 | omise.charges.create( 54 | { amount, currency: 'thb', source, return_uri }, 55 | function(err, res) { 56 | if (res) { 57 | resolve(res) 58 | } else { 59 | resolve(null) 60 | } 61 | } 62 | ) 63 | }) 64 | } 65 | -------------------------------------------------------------------------------- /src/utils/passport.js: -------------------------------------------------------------------------------- 1 | import passport from 'passport' 2 | import { Strategy as FacebookStrategy } from 'passport-facebook' 3 | import { Strategy as GoogleStrategy } from 'passport-google-oauth20' 4 | 5 | export const facebookPassportConfig = () => { 6 | return passport.use( 7 | new FacebookStrategy( 8 | { 9 | clientID: process.env.FACEBOOK_CLIENT_ID, 10 | clientSecret: process.env.FACEBOOK_CLIENT_SECRET, 11 | callbackURL: 'http://localhost:4444/auth/facebook/callback', 12 | profileFields: ['id', 'displayName', 'name', 'email'], 13 | passReqToCallback: true, 14 | }, 15 | function (req, accessToken, refreshToken, profile, done) { 16 | try { 17 | if (profile) { 18 | req.user = profile 19 | done(null, profile) 20 | } 21 | } catch (error) { 22 | done(error) 23 | } 24 | } 25 | ) 26 | ) 27 | } 28 | 29 | export const googlePassportConfig = () => { 30 | return passport.use( 31 | new GoogleStrategy( 32 | { 33 | clientID: process.env.GOOGLE_CLIENT_ID, 34 | clientSecret: process.env.GOOGLE_CLIENT_SECRET, 35 | callbackURL: 'http://localhost:4444/auth/google/callback', 36 | passReqToCallback: true, 37 | }, 38 | function (req, accessToken, refreshToken, profile, done) { 39 | try { 40 | if (profile) { 41 | req.user = profile 42 | done(null, profile) 43 | } 44 | } catch (error) { 45 | done(error) 46 | } 47 | } 48 | ) 49 | ) 50 | } 51 | -------------------------------------------------------------------------------- /src/utils/socialProvidersAuth.js: -------------------------------------------------------------------------------- 1 | import User from '../models/user' 2 | import jwt from 'jsonwebtoken' 3 | 4 | export const facebookAuth = async (req, res) => { 5 | const { 6 | id, 7 | name: { givenName }, 8 | emails: [{ value }], 9 | } = req.user 10 | 11 | try { 12 | // Find user in the database 13 | const user = await User.findOne({ providerId: id }) 14 | let token 15 | 16 | if (!user) { 17 | // User not found --> new user --> create new user in the database 18 | const newUser = await User.create({ 19 | name: givenName, 20 | email: value, 21 | password: `facebook_${id}`, 22 | providerId: id, 23 | }) 24 | 25 | // Send cookie to frontend 26 | 27 | token = jwt.sign({ userId: newUser.id }, process.env.SECRET, { 28 | expiresIn: '7days', 29 | }) 30 | } else { 31 | token = jwt.sign({ userId: user.id }, process.env.SECRET, { 32 | expiresIn: '7days', 33 | }) 34 | } 35 | res.cookie('jwt', token) 36 | res.redirect('http://localhost:3000/products') 37 | } catch (error) { 38 | res.redirect('http://localhost:3000/signin') 39 | } 40 | } 41 | 42 | export const googleAuth = async (req, res) => { 43 | const { 44 | id, 45 | displayName, 46 | emails: [{ value }], 47 | } = req.user 48 | 49 | try { 50 | // Find user in the database 51 | const user = await User.findOne({ providerId: id }) 52 | let token 53 | 54 | if (!user) { 55 | // User not found --> new user --> create new user in the database 56 | const newUser = await User.create({ 57 | name: displayName, 58 | email: value, 59 | password: `google_${id}`, 60 | providerId: id, 61 | }) 62 | 63 | // Send cookie to frontend 64 | 65 | token = jwt.sign({ userId: newUser.id }, process.env.SECRET, { 66 | expiresIn: '7days', 67 | }) 68 | } else { 69 | token = jwt.sign({ userId: user.id }, process.env.SECRET, { 70 | expiresIn: '7days', 71 | }) 72 | } 73 | res.cookie('jwt', token) 74 | res.redirect('http://localhost:3000/products') 75 | } catch (error) { 76 | res.redirect('http://localhost:3000/signin') 77 | } 78 | } 79 | --------------------------------------------------------------------------------