├── .env ├── .env.development ├── .env.production ├── .gitignore ├── CONTRIBUTING.md ├── README.md ├── certbot.ini ├── package-lock.json ├── package.json ├── pm2.json ├── src ├── Server.ts ├── config │ ├── constants.ts │ ├── environments.ts │ └── paths.ts ├── controllers │ ├── AlgorithmsController.ts │ ├── AuthController.ts │ ├── Controller.ts │ ├── TracersController.ts │ ├── VisualizationsController.ts │ └── index.ts ├── index.ts ├── middlewares │ ├── errorHandlerMiddleware.ts │ ├── frontendMiddleware.ts │ ├── index.ts │ └── redirectMiddleware.ts ├── models │ ├── Algorithm.ts │ ├── Category.ts │ ├── File.ts │ ├── Hierarchy.ts │ └── index.ts ├── tracers │ ├── DockerTracer.ts │ ├── LambdaTracer.ts │ ├── Tracer.ts │ ├── cpp │ │ ├── CppTracer.ts │ │ └── Dockerfile │ ├── index.ts │ ├── java │ │ └── JavaTracer.ts │ └── js │ │ ├── JsTracer.ts │ │ └── worker.js └── utils │ ├── apis.ts │ ├── hierarchy.ts │ └── misc.ts ├── tsconfig.json └── tslint.json /.env: -------------------------------------------------------------------------------- 1 | HTTP_PORT = 8080 2 | HTTPS_PORT = 8443 3 | -------------------------------------------------------------------------------- /.env.development: -------------------------------------------------------------------------------- 1 | CREDENTIALS_ENABLED = 0 2 | WEBHOOK_ENABLED = 0 3 | -------------------------------------------------------------------------------- /.env.production: -------------------------------------------------------------------------------- 1 | CREDENTIALS_ENABLED = 1 2 | CREDENTIALS_PATH = /home/ubuntu/.certbot/config/live/algorithm-visualizer.org 3 | CREDENTIALS_CA = fullchain.pem 4 | CREDENTIALS_KEY = privkey.pem 5 | CREDENTIALS_CERT = cert.pem 6 | WEBHOOK_ENABLED = 1 7 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # WebStorm settings 2 | /.idea 3 | 4 | # npm 5 | /node_modules 6 | /npm-debug.log 7 | 8 | # macOS 9 | .DS_Store 10 | 11 | # local .env* files 12 | .env.local 13 | .env.*.local 14 | 15 | # downloaded files 16 | /public 17 | -------------------------------------------------------------------------------- /CONTRIBUTING.md: -------------------------------------------------------------------------------- 1 | # Contributing 2 | 3 | > #### Table of Contents 4 | > - [Running Locally](#running-locally) 5 | > - [Directory Structure](#directory-structure) 6 | 7 | Are you a first-timer in contributing to open source? [These guidelines](https://opensource.guide/how-to-contribute/#how-to-submit-a-contribution) from GitHub might help! 8 | 9 | ## Running Locally 10 | 11 | 1. Fork this repository. 12 | 13 | 2. Clone your forked repo to your machine. 14 | 15 | ```bash 16 | git clone https://github.com//server.git 17 | ``` 18 | 19 | 3. Install [Docker](https://docs.docker.com/install/), if not done already. 20 | 21 | 4. Create `.env.local` in the project root: 22 | ```bash 23 | # By putting dummy values, GitHub sign in will not work locally 24 | GITHUB_CLIENT_ID = dummy 25 | GITHUB_CLIENT_SECRET = dummy 26 | 27 | # By putting dummy values, extracting visualizing commands will not work locally (except for JavaScript). 28 | AWS_ACCESS_KEY_ID = dummy 29 | AWS_SECRET_ACCESS_KEY = dummy 30 | ``` 31 | 32 | 5. Install dependencies, and run the server. 33 | 34 | ```bash 35 | cd server 36 | 37 | npm install 38 | 39 | npm run watch 40 | ``` 41 | 42 | 6. Open [`http://localhost:8080/`](http://localhost:8080/) in a web browser. 43 | 44 | ## Directory Structure 45 | 46 | - [**src/**](src) contains source code. 47 | - [**config/**](src/config) contains configuration files. 48 | - [**controllers/**](src/controllers) routes and processes incoming requests. 49 | - [**middlewares/**](src/middlewares) contains Express middlewares. 50 | - [**models/**](src/models) manages algorithm visualizations and their hierarchy. 51 | - [**tracers/**](src/tracers) build visualization libraries and compiles/runs code. 52 | - [**utils/**](src/utils) contains utility files. 53 | 54 | **NOTE** that for JavaScript, it builds a web worker rather than a docker image. Once a browser fetches the web worker, it will submit users' code to the web worker locally, instead of submitting to the remote server, to extract visualizing commands. 55 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Server 2 | 3 | > This repository is part of the project [Algorithm Visualizer](https://github.com/algorithm-visualizer). 4 | 5 | `server` serves [`algorithm-visualizer`](https://github.com/algorithm-visualizer/algorithm-visualizer) and provides APIs that the web app needs on the fly. (e.g., GitHub sign in, compiling/running code, etc.) 6 | 7 | ## Contributing 8 | 9 | Check out the [contributing guidelines](https://github.com/algorithm-visualizer/server/blob/master/CONTRIBUTING.md). 10 | -------------------------------------------------------------------------------- /certbot.ini: -------------------------------------------------------------------------------- 1 | config-dir = /home/ubuntu/.certbot/config 2 | work-dir = /home/ubuntu/.certbot/work 3 | logs-dir = /home/ubuntu/.certbot/logs 4 | email = parkjs814@gmail.com 5 | authenticator = webroot 6 | webroot-path = /home/ubuntu/server/public/frontend-built 7 | domains = algorithm-visualizer.org 8 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "@algorithm-visualizer/server", 3 | "version": "2.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@babel/code-frame": { 8 | "version": "7.0.0", 9 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.0.0.tgz", 10 | "integrity": "sha512-OfC2uemaknXr87bdLUkWog7nYuliM9Ij5HUcajsVcMCpQrcLmtxRbVFTIqmcSkSeYRBFBRxs2FiUqFJDLdiebA==", 11 | "dev": true, 12 | "requires": { 13 | "@babel/highlight": "^7.0.0" 14 | } 15 | }, 16 | "@babel/highlight": { 17 | "version": "7.0.0", 18 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.0.0.tgz", 19 | "integrity": "sha512-UFMC4ZeFC48Tpvj7C8UgLvtkaUuovQX+5xNWrsIoMG8o2z+XFKjKaN9iVmS84dPwVN00W4wPmqvYoZF3EGAsfw==", 20 | "dev": true, 21 | "requires": { 22 | "chalk": "^2.0.0", 23 | "esutils": "^2.0.2", 24 | "js-tokens": "^4.0.0" 25 | } 26 | }, 27 | "@types/body-parser": { 28 | "version": "1.17.0", 29 | "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.17.0.tgz", 30 | "integrity": "sha512-a2+YeUjPkztKJu5aIF2yArYFQQp8d51wZ7DavSHjFuY1mqVgidGyzEQ41JIVNy82fXj8yPgy2vJmfIywgESW6w==", 31 | "dev": true, 32 | "requires": { 33 | "@types/connect": "*", 34 | "@types/node": "*" 35 | } 36 | }, 37 | "@types/compression": { 38 | "version": "0.0.36", 39 | "resolved": "https://registry.npmjs.org/@types/compression/-/compression-0.0.36.tgz", 40 | "integrity": "sha512-B66iZCIcD2eB2F8e8YDIVtCUKgfiseOR5YOIbmMN2tM57Wu55j1xSdxdSw78aVzsPmbZ6G+hINc+1xe1tt4NBg==", 41 | "dev": true, 42 | "requires": { 43 | "@types/express": "*" 44 | } 45 | }, 46 | "@types/connect": { 47 | "version": "3.4.32", 48 | "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.32.tgz", 49 | "integrity": "sha512-4r8qa0quOvh7lGD0pre62CAb1oni1OO6ecJLGCezTmhQ8Fz50Arx9RUszryR8KlgK6avuSXvviL6yWyViQABOg==", 50 | "dev": true, 51 | "requires": { 52 | "@types/node": "*" 53 | } 54 | }, 55 | "@types/execa": { 56 | "version": "0.9.0", 57 | "resolved": "https://registry.npmjs.org/@types/execa/-/execa-0.9.0.tgz", 58 | "integrity": "sha512-mgfd93RhzjYBUHHV532turHC2j4l/qxsF/PbfDmprHDEUHmNZGlDn1CEsulGK3AfsPdhkWzZQT/S/k0UGhLGsA==", 59 | "dev": true, 60 | "requires": { 61 | "@types/node": "*" 62 | } 63 | }, 64 | "@types/express": { 65 | "version": "4.17.0", 66 | "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.0.tgz", 67 | "integrity": "sha512-CjaMu57cjgjuZbh9DpkloeGxV45CnMGlVd+XpG7Gm9QgVrd7KFq+X4HY0vM+2v0bczS48Wg7bvnMY5TN+Xmcfw==", 68 | "dev": true, 69 | "requires": { 70 | "@types/body-parser": "*", 71 | "@types/express-serve-static-core": "*", 72 | "@types/serve-static": "*" 73 | } 74 | }, 75 | "@types/express-serve-static-core": { 76 | "version": "4.16.7", 77 | "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.16.7.tgz", 78 | "integrity": "sha512-847KvL8Q1y3TtFLRTXcVakErLJQgdpFSaq+k043xefz9raEf0C7HalpSY7OW5PyjCnY8P7bPW5t/Co9qqp+USg==", 79 | "dev": true, 80 | "requires": { 81 | "@types/node": "*", 82 | "@types/range-parser": "*" 83 | } 84 | }, 85 | "@types/fs-extra": { 86 | "version": "7.0.0", 87 | "resolved": "https://registry.npmjs.org/@types/fs-extra/-/fs-extra-7.0.0.tgz", 88 | "integrity": "sha512-ndoMMbGyuToTy4qB6Lex/inR98nPiNHacsgMPvy+zqMLgSxbt8VtWpDArpGp69h1fEDQHn1KB+9DWD++wgbwYA==", 89 | "dev": true, 90 | "requires": { 91 | "@types/node": "*" 92 | } 93 | }, 94 | "@types/mime": { 95 | "version": "2.0.1", 96 | "resolved": "https://registry.npmjs.org/@types/mime/-/mime-2.0.1.tgz", 97 | "integrity": "sha512-FwI9gX75FgVBJ7ywgnq/P7tw+/o1GUbtP0KzbtusLigAOgIgNISRK0ZPl4qertvXSIE8YbsVJueQ90cDt9YYyw==", 98 | "dev": true 99 | }, 100 | "@types/morgan": { 101 | "version": "1.7.35", 102 | "resolved": "https://registry.npmjs.org/@types/morgan/-/morgan-1.7.35.tgz", 103 | "integrity": "sha512-E9qFi0seOkdlQnCTPv54brNfGWeFdRaEhI5tSue4pdx/V+xfxvMETsxXhOEcj1cYL+0n/jcTEmj/jD2gjzCwMg==", 104 | "dev": true, 105 | "requires": { 106 | "@types/express": "*" 107 | } 108 | }, 109 | "@types/node": { 110 | "version": "12.0.7", 111 | "resolved": "https://registry.npmjs.org/@types/node/-/node-12.0.7.tgz", 112 | "integrity": "sha512-1YKeT4JitGgE4SOzyB9eMwO0nGVNkNEsm9qlIt1Lqm/tG2QEiSMTD4kS3aO6L+w5SClLVxALmIBESK6Mk5wX0A==", 113 | "dev": true 114 | }, 115 | "@types/node-cron": { 116 | "version": "3.0.1", 117 | "resolved": "https://registry.npmjs.org/@types/node-cron/-/node-cron-3.0.1.tgz", 118 | "integrity": "sha512-BkMHHonDT8NJUE/pQ3kr5v2GLDKm5or9btLBoBx4F2MB2cuqYC748LYMDC55VlrLI5qZZv+Qgc3m4P3dBPcmeg==", 119 | "dev": true 120 | }, 121 | "@types/range-parser": { 122 | "version": "1.2.3", 123 | "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.3.tgz", 124 | "integrity": "sha512-ewFXqrQHlFsgc09MK5jP5iR7vumV/BYayNC6PgJO2LPe8vrnNFyjQjSppfEngITi0qvfKtzFvgKymGheFM9UOA==", 125 | "dev": true 126 | }, 127 | "@types/remove-markdown": { 128 | "version": "0.1.1", 129 | "resolved": "https://registry.npmjs.org/@types/remove-markdown/-/remove-markdown-0.1.1.tgz", 130 | "integrity": "sha512-SCYOFMHUqyiJU5M0V2gBB6UDdBhPwma34j0vYX0JgWaqp/74ila2Ops1jt5tB/C1UQXVXqK+is61884bITn3LQ==", 131 | "dev": true 132 | }, 133 | "@types/serve-static": { 134 | "version": "1.13.2", 135 | "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.13.2.tgz", 136 | "integrity": "sha512-/BZ4QRLpH/bNYgZgwhKEh+5AsboDBcUdlBYgzoLX0fpj3Y2gp6EApyOlM3bK53wQS/OE1SrdSYBAbux2D1528Q==", 137 | "dev": true, 138 | "requires": { 139 | "@types/express-serve-static-core": "*", 140 | "@types/mime": "*" 141 | } 142 | }, 143 | "@types/strip-bom": { 144 | "version": "3.0.0", 145 | "resolved": "https://registry.npmjs.org/@types/strip-bom/-/strip-bom-3.0.0.tgz", 146 | "integrity": "sha1-FKjsOVbC6B7bdSB5CuzyHCkK69I=", 147 | "dev": true 148 | }, 149 | "@types/strip-json-comments": { 150 | "version": "0.0.30", 151 | "resolved": "https://registry.npmjs.org/@types/strip-json-comments/-/strip-json-comments-0.0.30.tgz", 152 | "integrity": "sha512-7NQmHra/JILCd1QqpSzl8+mJRc8ZHz3uDm8YV1Ks9IhK0epEiTw8aIErbvH9PI+6XbqhyIQy3462nEsn7UVzjQ==", 153 | "dev": true 154 | }, 155 | "@types/uuid": { 156 | "version": "3.4.4", 157 | "resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-3.4.4.tgz", 158 | "integrity": "sha512-tPIgT0GUmdJQNSHxp0X2jnpQfBSTfGxUMc/2CXBU2mnyTFVYVa2ojpoQ74w0U2yn2vw3jnC640+77lkFFpdVDw==", 159 | "dev": true, 160 | "requires": { 161 | "@types/node": "*" 162 | } 163 | }, 164 | "accepts": { 165 | "version": "1.3.7", 166 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 167 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 168 | "requires": { 169 | "mime-types": "~2.1.24", 170 | "negotiator": "0.6.2" 171 | } 172 | }, 173 | "ansi-styles": { 174 | "version": "3.2.1", 175 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 176 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 177 | "dev": true, 178 | "requires": { 179 | "color-convert": "^1.9.0" 180 | } 181 | }, 182 | "arg": { 183 | "version": "4.1.0", 184 | "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.0.tgz", 185 | "integrity": "sha512-ZWc51jO3qegGkVh8Hwpv636EkbesNV5ZNQPCtRa+0qytRYPEs9IYT9qITY9buezqUH5uqyzlWLcufrzU2rffdg==" 186 | }, 187 | "argparse": { 188 | "version": "1.0.10", 189 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 190 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 191 | "dev": true, 192 | "requires": { 193 | "sprintf-js": "~1.0.2" 194 | } 195 | }, 196 | "array-find-index": { 197 | "version": "1.0.2", 198 | "resolved": "https://registry.npmjs.org/array-find-index/-/array-find-index-1.0.2.tgz", 199 | "integrity": "sha1-3wEKoSh+Fku9pvlyOwqWoexBh6E=", 200 | "dev": true 201 | }, 202 | "array-flatten": { 203 | "version": "1.1.1", 204 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 205 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 206 | }, 207 | "aws-sdk": { 208 | "version": "2.814.0", 209 | "resolved": "https://registry.npmjs.org/aws-sdk/-/aws-sdk-2.814.0.tgz", 210 | "integrity": "sha512-empd1m/J/MAkL6d9OeRpmg9thobULu0wk4v8W3JToaxGi2TD7PIdvE6yliZKyOVAdJINhBWEBhxR4OUIHhcGbQ==", 211 | "requires": { 212 | "buffer": "4.9.2", 213 | "events": "1.1.1", 214 | "ieee754": "1.1.13", 215 | "jmespath": "0.15.0", 216 | "querystring": "0.2.0", 217 | "sax": "1.2.1", 218 | "url": "0.10.3", 219 | "uuid": "3.3.2", 220 | "xml2js": "0.4.19" 221 | } 222 | }, 223 | "axios": { 224 | "version": "0.21.2", 225 | "resolved": "https://registry.npmjs.org/axios/-/axios-0.21.2.tgz", 226 | "integrity": "sha512-87otirqUw3e8CzHTMO+/9kh/FSgXt/eVDvipijwDtEuwbkySWZ9SBm6VEubmJ/kLKEoLQV/POhxXFb66bfekfg==", 227 | "requires": { 228 | "follow-redirects": "^1.14.0" 229 | } 230 | }, 231 | "balanced-match": { 232 | "version": "1.0.0", 233 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 234 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 235 | "dev": true 236 | }, 237 | "base64-js": { 238 | "version": "1.5.1", 239 | "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", 240 | "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" 241 | }, 242 | "basic-auth": { 243 | "version": "2.0.1", 244 | "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz", 245 | "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==", 246 | "requires": { 247 | "safe-buffer": "5.1.2" 248 | } 249 | }, 250 | "body-parser": { 251 | "version": "1.19.0", 252 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", 253 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", 254 | "requires": { 255 | "bytes": "3.1.0", 256 | "content-type": "~1.0.4", 257 | "debug": "2.6.9", 258 | "depd": "~1.1.2", 259 | "http-errors": "1.7.2", 260 | "iconv-lite": "0.4.24", 261 | "on-finished": "~2.3.0", 262 | "qs": "6.7.0", 263 | "raw-body": "2.4.0", 264 | "type-is": "~1.6.17" 265 | }, 266 | "dependencies": { 267 | "debug": { 268 | "version": "2.6.9", 269 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 270 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 271 | "requires": { 272 | "ms": "2.0.0" 273 | } 274 | } 275 | } 276 | }, 277 | "brace-expansion": { 278 | "version": "1.1.11", 279 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 280 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 281 | "dev": true, 282 | "requires": { 283 | "balanced-match": "^1.0.0", 284 | "concat-map": "0.0.1" 285 | } 286 | }, 287 | "buffer": { 288 | "version": "4.9.2", 289 | "resolved": "https://registry.npmjs.org/buffer/-/buffer-4.9.2.tgz", 290 | "integrity": "sha512-xq+q3SRMOxGivLhBNaUdC64hDTQwejJ+H0T/NB1XMtTVEwNTrfFF3gAxiyW0Bu/xWEGhjVKgUcMhCrUy2+uCWg==", 291 | "requires": { 292 | "base64-js": "^1.0.2", 293 | "ieee754": "^1.1.4", 294 | "isarray": "^1.0.0" 295 | } 296 | }, 297 | "buffer-equal-constant-time": { 298 | "version": "1.0.1", 299 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", 300 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" 301 | }, 302 | "buffer-from": { 303 | "version": "1.1.1", 304 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", 305 | "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==" 306 | }, 307 | "builtin-modules": { 308 | "version": "1.1.1", 309 | "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", 310 | "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", 311 | "dev": true 312 | }, 313 | "bytes": { 314 | "version": "3.1.0", 315 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", 316 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" 317 | }, 318 | "camelcase": { 319 | "version": "2.1.1", 320 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-2.1.1.tgz", 321 | "integrity": "sha1-fB0W1nmhu+WcoCys7PsBHiAfWh8=", 322 | "dev": true 323 | }, 324 | "camelcase-keys": { 325 | "version": "2.1.0", 326 | "resolved": "https://registry.npmjs.org/camelcase-keys/-/camelcase-keys-2.1.0.tgz", 327 | "integrity": "sha1-MIvur/3ygRkFHvodkyITyRuPkuc=", 328 | "dev": true, 329 | "requires": { 330 | "camelcase": "^2.0.0", 331 | "map-obj": "^1.0.0" 332 | } 333 | }, 334 | "chalk": { 335 | "version": "2.4.2", 336 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 337 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 338 | "dev": true, 339 | "requires": { 340 | "ansi-styles": "^3.2.1", 341 | "escape-string-regexp": "^1.0.5", 342 | "supports-color": "^5.3.0" 343 | } 344 | }, 345 | "color-convert": { 346 | "version": "1.9.3", 347 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 348 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 349 | "dev": true, 350 | "requires": { 351 | "color-name": "1.1.3" 352 | } 353 | }, 354 | "color-name": { 355 | "version": "1.1.3", 356 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 357 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 358 | "dev": true 359 | }, 360 | "commander": { 361 | "version": "2.20.0", 362 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.0.tgz", 363 | "integrity": "sha512-7j2y+40w61zy6YC2iRNpUe/NwhNyoXrYpHMrSunaMG64nRnaf96zO/KMQR4OyN/UnE5KLyEBnKHd4aG3rskjpQ==", 364 | "dev": true 365 | }, 366 | "compressible": { 367 | "version": "2.0.17", 368 | "resolved": "https://registry.npmjs.org/compressible/-/compressible-2.0.17.tgz", 369 | "integrity": "sha512-BGHeLCK1GV7j1bSmQQAi26X+GgWcTjLr/0tzSvMCl3LH1w1IJ4PFSPoV5316b30cneTziC+B1a+3OjoSUcQYmw==", 370 | "requires": { 371 | "mime-db": ">= 1.40.0 < 2" 372 | } 373 | }, 374 | "compression": { 375 | "version": "1.7.4", 376 | "resolved": "https://registry.npmjs.org/compression/-/compression-1.7.4.tgz", 377 | "integrity": "sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ==", 378 | "requires": { 379 | "accepts": "~1.3.5", 380 | "bytes": "3.0.0", 381 | "compressible": "~2.0.16", 382 | "debug": "2.6.9", 383 | "on-headers": "~1.0.2", 384 | "safe-buffer": "5.1.2", 385 | "vary": "~1.1.2" 386 | }, 387 | "dependencies": { 388 | "bytes": { 389 | "version": "3.0.0", 390 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", 391 | "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" 392 | }, 393 | "debug": { 394 | "version": "2.6.9", 395 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 396 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 397 | "requires": { 398 | "ms": "2.0.0" 399 | } 400 | } 401 | } 402 | }, 403 | "concat-map": { 404 | "version": "0.0.1", 405 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 406 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 407 | "dev": true 408 | }, 409 | "content-disposition": { 410 | "version": "0.5.3", 411 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", 412 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", 413 | "requires": { 414 | "safe-buffer": "5.1.2" 415 | } 416 | }, 417 | "content-type": { 418 | "version": "1.0.4", 419 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 420 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 421 | }, 422 | "cookie": { 423 | "version": "0.4.0", 424 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 425 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 426 | }, 427 | "cookie-signature": { 428 | "version": "1.0.6", 429 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 430 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 431 | }, 432 | "currently-unhandled": { 433 | "version": "0.4.1", 434 | "resolved": "https://registry.npmjs.org/currently-unhandled/-/currently-unhandled-0.4.1.tgz", 435 | "integrity": "sha1-mI3zP+qxke95mmE2nddsF635V+o=", 436 | "dev": true, 437 | "requires": { 438 | "array-find-index": "^1.0.1" 439 | } 440 | }, 441 | "dateformat": { 442 | "version": "1.0.12", 443 | "resolved": "https://registry.npmjs.org/dateformat/-/dateformat-1.0.12.tgz", 444 | "integrity": "sha1-nxJLZ1lMk3/3BpMuSmQsyo27/uk=", 445 | "dev": true, 446 | "requires": { 447 | "get-stdin": "^4.0.1", 448 | "meow": "^3.3.0" 449 | } 450 | }, 451 | "debounce": { 452 | "version": "1.2.0", 453 | "resolved": "https://registry.npmjs.org/debounce/-/debounce-1.2.0.tgz", 454 | "integrity": "sha512-mYtLl1xfZLi1m4RtQYlZgJUNQjl4ZxVnHzIR8nLLgi4q1YT8o/WM+MK/f8yfcc9s5Ir5zRaPZyZU6xs1Syoocg==", 455 | "dev": true 456 | }, 457 | "decamelize": { 458 | "version": "1.2.0", 459 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", 460 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", 461 | "dev": true 462 | }, 463 | "depd": { 464 | "version": "1.1.2", 465 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 466 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 467 | }, 468 | "destroy": { 469 | "version": "1.0.4", 470 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 471 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 472 | }, 473 | "diff": { 474 | "version": "4.0.1", 475 | "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.1.tgz", 476 | "integrity": "sha512-s2+XdvhPCOF01LRQBC8hf4vhbVmI2CGS5aZnxLJlT5FtdhPCDFq80q++zK2KlrVorVDdL5BOGZ/VfLrVtYNF+Q==" 477 | }, 478 | "dotenv": { 479 | "version": "8.0.0", 480 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.0.0.tgz", 481 | "integrity": "sha512-30xVGqjLjiUOArT4+M5q9sYdvuR4riM6yK9wMcas9Vbp6zZa+ocC9dp6QoftuhTPhFAiLK/0C5Ni2nou/Bk8lg==" 482 | }, 483 | "dotenv-flow": { 484 | "version": "2.0.0", 485 | "resolved": "https://registry.npmjs.org/dotenv-flow/-/dotenv-flow-2.0.0.tgz", 486 | "integrity": "sha512-Gfey3wkr644VCq8e0EsX7d2ywbX9WTCOu5cakwE8seUXuoSUqr/aA1Ft1VqfZcnPaPPD0ADHhQ5IrjuZieMlVw==", 487 | "requires": { 488 | "dotenv": "^8.0.0" 489 | } 490 | }, 491 | "dynamic-dedupe": { 492 | "version": "0.3.0", 493 | "resolved": "https://registry.npmjs.org/dynamic-dedupe/-/dynamic-dedupe-0.3.0.tgz", 494 | "integrity": "sha1-BuRMIj9eTpTXjvnbI6ZRXOL5YqE=", 495 | "dev": true, 496 | "requires": { 497 | "xtend": "^4.0.0" 498 | } 499 | }, 500 | "ee-first": { 501 | "version": "1.1.1", 502 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 503 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 504 | }, 505 | "encodeurl": { 506 | "version": "1.0.2", 507 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 508 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 509 | }, 510 | "error-ex": { 511 | "version": "1.3.2", 512 | "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", 513 | "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", 514 | "dev": true, 515 | "requires": { 516 | "is-arrayish": "^0.2.1" 517 | } 518 | }, 519 | "escape-html": { 520 | "version": "1.0.3", 521 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 522 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 523 | }, 524 | "escape-string-regexp": { 525 | "version": "1.0.5", 526 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 527 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 528 | "dev": true 529 | }, 530 | "esprima": { 531 | "version": "4.0.1", 532 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 533 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", 534 | "dev": true 535 | }, 536 | "esutils": { 537 | "version": "2.0.2", 538 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz", 539 | "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=", 540 | "dev": true 541 | }, 542 | "etag": { 543 | "version": "1.8.1", 544 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 545 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 546 | }, 547 | "events": { 548 | "version": "1.1.1", 549 | "resolved": "https://registry.npmjs.org/events/-/events-1.1.1.tgz", 550 | "integrity": "sha1-nr23Y1rQmccNzEwqH1AEKI6L2SQ=" 551 | }, 552 | "express": { 553 | "version": "4.17.1", 554 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", 555 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", 556 | "requires": { 557 | "accepts": "~1.3.7", 558 | "array-flatten": "1.1.1", 559 | "body-parser": "1.19.0", 560 | "content-disposition": "0.5.3", 561 | "content-type": "~1.0.4", 562 | "cookie": "0.4.0", 563 | "cookie-signature": "1.0.6", 564 | "debug": "2.6.9", 565 | "depd": "~1.1.2", 566 | "encodeurl": "~1.0.2", 567 | "escape-html": "~1.0.3", 568 | "etag": "~1.8.1", 569 | "finalhandler": "~1.1.2", 570 | "fresh": "0.5.2", 571 | "merge-descriptors": "1.0.1", 572 | "methods": "~1.1.2", 573 | "on-finished": "~2.3.0", 574 | "parseurl": "~1.3.3", 575 | "path-to-regexp": "0.1.7", 576 | "proxy-addr": "~2.0.5", 577 | "qs": "6.7.0", 578 | "range-parser": "~1.2.1", 579 | "safe-buffer": "5.1.2", 580 | "send": "0.17.1", 581 | "serve-static": "1.14.1", 582 | "setprototypeof": "1.1.1", 583 | "statuses": "~1.5.0", 584 | "type-is": "~1.6.18", 585 | "utils-merge": "1.0.1", 586 | "vary": "~1.1.2" 587 | }, 588 | "dependencies": { 589 | "debug": { 590 | "version": "2.6.9", 591 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 592 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 593 | "requires": { 594 | "ms": "2.0.0" 595 | } 596 | } 597 | } 598 | }, 599 | "express-github-webhook": { 600 | "version": "1.0.6", 601 | "resolved": "https://registry.npmjs.org/express-github-webhook/-/express-github-webhook-1.0.6.tgz", 602 | "integrity": "sha1-qYDKCriLjL2ke5Sh6RjD8KaPTK0=", 603 | "requires": { 604 | "buffer-equal-constant-time": "^1.0.1" 605 | } 606 | }, 607 | "filewatcher": { 608 | "version": "3.0.1", 609 | "resolved": "https://registry.npmjs.org/filewatcher/-/filewatcher-3.0.1.tgz", 610 | "integrity": "sha1-9KGVc1Xdr0Q8zXiolfPVXiPIoDQ=", 611 | "dev": true, 612 | "requires": { 613 | "debounce": "^1.0.0" 614 | } 615 | }, 616 | "finalhandler": { 617 | "version": "1.1.2", 618 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", 619 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", 620 | "requires": { 621 | "debug": "2.6.9", 622 | "encodeurl": "~1.0.2", 623 | "escape-html": "~1.0.3", 624 | "on-finished": "~2.3.0", 625 | "parseurl": "~1.3.3", 626 | "statuses": "~1.5.0", 627 | "unpipe": "~1.0.0" 628 | }, 629 | "dependencies": { 630 | "debug": { 631 | "version": "2.6.9", 632 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 633 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 634 | "requires": { 635 | "ms": "2.0.0" 636 | } 637 | } 638 | } 639 | }, 640 | "find-up": { 641 | "version": "1.1.2", 642 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-1.1.2.tgz", 643 | "integrity": "sha1-ay6YIrGizgpgq2TWEOzK1TyyTQ8=", 644 | "dev": true, 645 | "requires": { 646 | "path-exists": "^2.0.0", 647 | "pinkie-promise": "^2.0.0" 648 | } 649 | }, 650 | "follow-redirects": { 651 | "version": "1.14.5", 652 | "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.14.5.tgz", 653 | "integrity": "sha512-wtphSXy7d4/OR+MvIFbCVBDzZ5520qV8XfPklSN5QtxuMUJZ+b0Wnst1e1lCDocfzuCkHqj8k0FpZqO+UIaKNA==" 654 | }, 655 | "forwarded": { 656 | "version": "0.1.2", 657 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 658 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 659 | }, 660 | "fresh": { 661 | "version": "0.5.2", 662 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 663 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 664 | }, 665 | "fs-extra": { 666 | "version": "6.0.1", 667 | "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-6.0.1.tgz", 668 | "integrity": "sha512-GnyIkKhhzXZUWFCaJzvyDLEEgDkPfb4/TPvJCJVuS8MWZgoSsErf++QpiAlDnKFcqhRlm+tIOcencCjyJE6ZCA==", 669 | "requires": { 670 | "graceful-fs": "^4.1.2", 671 | "jsonfile": "^4.0.0", 672 | "universalify": "^0.1.0" 673 | } 674 | }, 675 | "fs.realpath": { 676 | "version": "1.0.0", 677 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 678 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", 679 | "dev": true 680 | }, 681 | "get-stdin": { 682 | "version": "4.0.1", 683 | "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-4.0.1.tgz", 684 | "integrity": "sha1-uWjGsKBDhDJJAui/Gl3zJXmkUP4=", 685 | "dev": true 686 | }, 687 | "glob": { 688 | "version": "7.1.4", 689 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.4.tgz", 690 | "integrity": "sha512-hkLPepehmnKk41pUGm3sYxoFs/umurYfYJCerbXEyFIWcAzvpipAgVkBqqT9RBKMGjnq6kMuyYwha6csxbiM1A==", 691 | "dev": true, 692 | "requires": { 693 | "fs.realpath": "^1.0.0", 694 | "inflight": "^1.0.4", 695 | "inherits": "2", 696 | "minimatch": "^3.0.4", 697 | "once": "^1.3.0", 698 | "path-is-absolute": "^1.0.0" 699 | } 700 | }, 701 | "graceful-fs": { 702 | "version": "4.1.15", 703 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.15.tgz", 704 | "integrity": "sha512-6uHUhOPEBgQ24HM+r6b/QwWfZq+yiFcipKFrOFiBEnWdy5sdzYoi+pJeQaPI5qOLRFqWmAXUPQNsielzdLoecA==" 705 | }, 706 | "growly": { 707 | "version": "1.3.0", 708 | "resolved": "https://registry.npmjs.org/growly/-/growly-1.3.0.tgz", 709 | "integrity": "sha1-8QdIy+dq+WS3yWyTxrzCivEgwIE=", 710 | "dev": true 711 | }, 712 | "has-flag": { 713 | "version": "3.0.0", 714 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 715 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 716 | "dev": true 717 | }, 718 | "hosted-git-info": { 719 | "version": "2.8.9", 720 | "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz", 721 | "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==", 722 | "dev": true 723 | }, 724 | "http-errors": { 725 | "version": "1.7.2", 726 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 727 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 728 | "requires": { 729 | "depd": "~1.1.2", 730 | "inherits": "2.0.3", 731 | "setprototypeof": "1.1.1", 732 | "statuses": ">= 1.5.0 < 2", 733 | "toidentifier": "1.0.0" 734 | } 735 | }, 736 | "iconv-lite": { 737 | "version": "0.4.24", 738 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 739 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 740 | "requires": { 741 | "safer-buffer": ">= 2.1.2 < 3" 742 | } 743 | }, 744 | "ieee754": { 745 | "version": "1.1.13", 746 | "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.1.13.tgz", 747 | "integrity": "sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg==" 748 | }, 749 | "indent-string": { 750 | "version": "2.1.0", 751 | "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-2.1.0.tgz", 752 | "integrity": "sha1-ji1INIdCEhtKghi3oTfppSBJ3IA=", 753 | "dev": true, 754 | "requires": { 755 | "repeating": "^2.0.0" 756 | } 757 | }, 758 | "inflight": { 759 | "version": "1.0.6", 760 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 761 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 762 | "dev": true, 763 | "requires": { 764 | "once": "^1.3.0", 765 | "wrappy": "1" 766 | } 767 | }, 768 | "inherits": { 769 | "version": "2.0.3", 770 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 771 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 772 | }, 773 | "ipaddr.js": { 774 | "version": "1.9.0", 775 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.0.tgz", 776 | "integrity": "sha512-M4Sjn6N/+O6/IXSJseKqHoFc+5FdGJ22sXqnjTpdZweHK64MzEPAyQZyEU3R/KRv2GLoa7nNtg/C2Ev6m7z+eA==" 777 | }, 778 | "is-arrayish": { 779 | "version": "0.2.1", 780 | "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", 781 | "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", 782 | "dev": true 783 | }, 784 | "is-finite": { 785 | "version": "1.0.2", 786 | "resolved": "https://registry.npmjs.org/is-finite/-/is-finite-1.0.2.tgz", 787 | "integrity": "sha1-zGZ3aVYCvlUO8R6LSqYwU0K20Ko=", 788 | "dev": true, 789 | "requires": { 790 | "number-is-nan": "^1.0.0" 791 | } 792 | }, 793 | "is-utf8": { 794 | "version": "0.2.1", 795 | "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz", 796 | "integrity": "sha1-Sw2hRCEE0bM2NA6AeX6GXPOffXI=", 797 | "dev": true 798 | }, 799 | "is-wsl": { 800 | "version": "1.1.0", 801 | "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", 802 | "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", 803 | "dev": true 804 | }, 805 | "isarray": { 806 | "version": "1.0.0", 807 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 808 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 809 | }, 810 | "isexe": { 811 | "version": "2.0.0", 812 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 813 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 814 | "dev": true 815 | }, 816 | "jmespath": { 817 | "version": "0.15.0", 818 | "resolved": "https://registry.npmjs.org/jmespath/-/jmespath-0.15.0.tgz", 819 | "integrity": "sha1-o/Iiqarp+Wb10nx5ZRDigJF2Qhc=" 820 | }, 821 | "js-tokens": { 822 | "version": "4.0.0", 823 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 824 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 825 | "dev": true 826 | }, 827 | "js-yaml": { 828 | "version": "3.13.1", 829 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", 830 | "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", 831 | "dev": true, 832 | "requires": { 833 | "argparse": "^1.0.7", 834 | "esprima": "^4.0.0" 835 | } 836 | }, 837 | "jsonfile": { 838 | "version": "4.0.0", 839 | "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", 840 | "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", 841 | "requires": { 842 | "graceful-fs": "^4.1.6" 843 | } 844 | }, 845 | "load-json-file": { 846 | "version": "1.1.0", 847 | "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-1.1.0.tgz", 848 | "integrity": "sha1-lWkFcI1YtLq0wiYbBPWfMcmTdMA=", 849 | "dev": true, 850 | "requires": { 851 | "graceful-fs": "^4.1.2", 852 | "parse-json": "^2.2.0", 853 | "pify": "^2.0.0", 854 | "pinkie-promise": "^2.0.0", 855 | "strip-bom": "^2.0.0" 856 | } 857 | }, 858 | "loud-rejection": { 859 | "version": "1.6.0", 860 | "resolved": "https://registry.npmjs.org/loud-rejection/-/loud-rejection-1.6.0.tgz", 861 | "integrity": "sha1-W0b4AUft7leIcPCG0Eghz5mOVR8=", 862 | "dev": true, 863 | "requires": { 864 | "currently-unhandled": "^0.4.1", 865 | "signal-exit": "^3.0.0" 866 | } 867 | }, 868 | "make-error": { 869 | "version": "1.3.5", 870 | "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.5.tgz", 871 | "integrity": "sha512-c3sIjNUow0+8swNwVpqoH4YCShKNFkMaw6oH1mNS2haDZQqkeZFlHS3dhoeEbKKmJB4vXpJucU6oH75aDYeE9g==" 872 | }, 873 | "map-obj": { 874 | "version": "1.0.1", 875 | "resolved": "https://registry.npmjs.org/map-obj/-/map-obj-1.0.1.tgz", 876 | "integrity": "sha1-2TPOuSBdgr3PSIb2dCvcK03qFG0=", 877 | "dev": true 878 | }, 879 | "media-typer": { 880 | "version": "0.3.0", 881 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 882 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 883 | }, 884 | "meow": { 885 | "version": "3.7.0", 886 | "resolved": "https://registry.npmjs.org/meow/-/meow-3.7.0.tgz", 887 | "integrity": "sha1-cstmi0JSKCkKu/qFaJJYcwioAfs=", 888 | "dev": true, 889 | "requires": { 890 | "camelcase-keys": "^2.0.0", 891 | "decamelize": "^1.1.2", 892 | "loud-rejection": "^1.0.0", 893 | "map-obj": "^1.0.1", 894 | "minimist": "^1.1.3", 895 | "normalize-package-data": "^2.3.4", 896 | "object-assign": "^4.0.1", 897 | "read-pkg-up": "^1.0.1", 898 | "redent": "^1.0.0", 899 | "trim-newlines": "^1.0.0" 900 | } 901 | }, 902 | "merge-descriptors": { 903 | "version": "1.0.1", 904 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 905 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 906 | }, 907 | "methods": { 908 | "version": "1.1.2", 909 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 910 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 911 | }, 912 | "mime": { 913 | "version": "1.6.0", 914 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 915 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 916 | }, 917 | "mime-db": { 918 | "version": "1.40.0", 919 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.40.0.tgz", 920 | "integrity": "sha512-jYdeOMPy9vnxEqFRRo6ZvTZ8d9oPb+k18PKoYNYUe2stVEBPPwsln/qWzdbmaIvnhZ9v2P+CuecK+fpUfsV2mA==" 921 | }, 922 | "mime-types": { 923 | "version": "2.1.24", 924 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.24.tgz", 925 | "integrity": "sha512-WaFHS3MCl5fapm3oLxU4eYDw77IQM2ACcxQ9RIxfaC3ooc6PFuBMGZZsYpvoXS5D5QTWPieo1jjLdAm3TBP3cQ==", 926 | "requires": { 927 | "mime-db": "1.40.0" 928 | } 929 | }, 930 | "minimatch": { 931 | "version": "3.0.4", 932 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 933 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 934 | "dev": true, 935 | "requires": { 936 | "brace-expansion": "^1.1.7" 937 | } 938 | }, 939 | "minimist": { 940 | "version": "1.2.0", 941 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", 942 | "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=", 943 | "dev": true 944 | }, 945 | "mkdirp": { 946 | "version": "0.5.1", 947 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", 948 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", 949 | "dev": true, 950 | "requires": { 951 | "minimist": "0.0.8" 952 | }, 953 | "dependencies": { 954 | "minimist": { 955 | "version": "0.0.8", 956 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", 957 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", 958 | "dev": true 959 | } 960 | } 961 | }, 962 | "moment": { 963 | "version": "2.29.1", 964 | "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz", 965 | "integrity": "sha512-kHmoybcPV8Sqy59DwNDY3Jefr64lK/by/da0ViFcuA4DH0vQg5Q6Ze5VimxkfQNSC+Mls/Kx53s7TjP1RhFEDQ==" 966 | }, 967 | "moment-timezone": { 968 | "version": "0.5.34", 969 | "resolved": "https://registry.npmjs.org/moment-timezone/-/moment-timezone-0.5.34.tgz", 970 | "integrity": "sha512-3zAEHh2hKUs3EXLESx/wsgw6IQdusOT8Bxm3D9UrHPQR7zlMmzwybC8zHEM1tQ4LJwP7fcxrWr8tuBg05fFCbg==", 971 | "requires": { 972 | "moment": ">= 2.9.0" 973 | } 974 | }, 975 | "morgan": { 976 | "version": "1.9.1", 977 | "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.9.1.tgz", 978 | "integrity": "sha512-HQStPIV4y3afTiCYVxirakhlCfGkI161c76kKFca7Fk1JusM//Qeo1ej2XaMniiNeaZklMVrh3vTtIzpzwbpmA==", 979 | "requires": { 980 | "basic-auth": "~2.0.0", 981 | "debug": "2.6.9", 982 | "depd": "~1.1.2", 983 | "on-finished": "~2.3.0", 984 | "on-headers": "~1.0.1" 985 | }, 986 | "dependencies": { 987 | "debug": { 988 | "version": "2.6.9", 989 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 990 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 991 | "requires": { 992 | "ms": "2.0.0" 993 | } 994 | } 995 | } 996 | }, 997 | "ms": { 998 | "version": "2.0.0", 999 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1000 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 1001 | }, 1002 | "negotiator": { 1003 | "version": "0.6.2", 1004 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 1005 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 1006 | }, 1007 | "node-cron": { 1008 | "version": "3.0.0", 1009 | "resolved": "https://registry.npmjs.org/node-cron/-/node-cron-3.0.0.tgz", 1010 | "integrity": "sha512-DDwIvvuCwrNiaU7HEivFDULcaQualDv7KoNlB/UU1wPW0n1tDEmBJKhEIE6DlF2FuoOHcNbLJ8ITL2Iv/3AWmA==", 1011 | "requires": { 1012 | "moment-timezone": "^0.5.31" 1013 | } 1014 | }, 1015 | "node-notifier": { 1016 | "version": "5.4.0", 1017 | "resolved": "https://registry.npmjs.org/node-notifier/-/node-notifier-5.4.0.tgz", 1018 | "integrity": "sha512-SUDEb+o71XR5lXSTyivXd9J7fCloE3SyP4lSgt3lU2oSANiox+SxlNRGPjDKrwU1YN3ix2KN/VGGCg0t01rttQ==", 1019 | "dev": true, 1020 | "requires": { 1021 | "growly": "^1.3.0", 1022 | "is-wsl": "^1.1.0", 1023 | "semver": "^5.5.0", 1024 | "shellwords": "^0.1.1", 1025 | "which": "^1.3.0" 1026 | } 1027 | }, 1028 | "normalize-package-data": { 1029 | "version": "2.5.0", 1030 | "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", 1031 | "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", 1032 | "dev": true, 1033 | "requires": { 1034 | "hosted-git-info": "^2.1.4", 1035 | "resolve": "^1.10.0", 1036 | "semver": "2 || 3 || 4 || 5", 1037 | "validate-npm-package-license": "^3.0.1" 1038 | } 1039 | }, 1040 | "number-is-nan": { 1041 | "version": "1.0.1", 1042 | "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", 1043 | "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=", 1044 | "dev": true 1045 | }, 1046 | "object-assign": { 1047 | "version": "4.1.1", 1048 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 1049 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", 1050 | "dev": true 1051 | }, 1052 | "on-finished": { 1053 | "version": "2.3.0", 1054 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 1055 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 1056 | "requires": { 1057 | "ee-first": "1.1.1" 1058 | } 1059 | }, 1060 | "on-headers": { 1061 | "version": "1.0.2", 1062 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", 1063 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==" 1064 | }, 1065 | "once": { 1066 | "version": "1.4.0", 1067 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1068 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1069 | "dev": true, 1070 | "requires": { 1071 | "wrappy": "1" 1072 | } 1073 | }, 1074 | "parse-json": { 1075 | "version": "2.2.0", 1076 | "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", 1077 | "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=", 1078 | "dev": true, 1079 | "requires": { 1080 | "error-ex": "^1.2.0" 1081 | } 1082 | }, 1083 | "parseurl": { 1084 | "version": "1.3.3", 1085 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 1086 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 1087 | }, 1088 | "path-exists": { 1089 | "version": "2.1.0", 1090 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-2.1.0.tgz", 1091 | "integrity": "sha1-D+tsZPD8UY2adU3V77YscCJ2H0s=", 1092 | "dev": true, 1093 | "requires": { 1094 | "pinkie-promise": "^2.0.0" 1095 | } 1096 | }, 1097 | "path-is-absolute": { 1098 | "version": "1.0.1", 1099 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1100 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 1101 | "dev": true 1102 | }, 1103 | "path-parse": { 1104 | "version": "1.0.7", 1105 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", 1106 | "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", 1107 | "dev": true 1108 | }, 1109 | "path-to-regexp": { 1110 | "version": "0.1.7", 1111 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 1112 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 1113 | }, 1114 | "path-type": { 1115 | "version": "1.1.0", 1116 | "resolved": "https://registry.npmjs.org/path-type/-/path-type-1.1.0.tgz", 1117 | "integrity": "sha1-WcRPfuSR2nBNpBXaWkBwuk+P5EE=", 1118 | "dev": true, 1119 | "requires": { 1120 | "graceful-fs": "^4.1.2", 1121 | "pify": "^2.0.0", 1122 | "pinkie-promise": "^2.0.0" 1123 | } 1124 | }, 1125 | "pify": { 1126 | "version": "2.3.0", 1127 | "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", 1128 | "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", 1129 | "dev": true 1130 | }, 1131 | "pinkie": { 1132 | "version": "2.0.4", 1133 | "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", 1134 | "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=", 1135 | "dev": true 1136 | }, 1137 | "pinkie-promise": { 1138 | "version": "2.0.1", 1139 | "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", 1140 | "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=", 1141 | "dev": true, 1142 | "requires": { 1143 | "pinkie": "^2.0.0" 1144 | } 1145 | }, 1146 | "proxy-addr": { 1147 | "version": "2.0.5", 1148 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.5.tgz", 1149 | "integrity": "sha512-t/7RxHXPH6cJtP0pRG6smSr9QJidhB+3kXu0KgXnbGYMgzEnUxRQ4/LDdfOwZEMyIh3/xHb8PX3t+lfL9z+YVQ==", 1150 | "requires": { 1151 | "forwarded": "~0.1.2", 1152 | "ipaddr.js": "1.9.0" 1153 | } 1154 | }, 1155 | "punycode": { 1156 | "version": "1.3.2", 1157 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", 1158 | "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=" 1159 | }, 1160 | "qs": { 1161 | "version": "6.7.0", 1162 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", 1163 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" 1164 | }, 1165 | "querystring": { 1166 | "version": "0.2.0", 1167 | "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", 1168 | "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA=" 1169 | }, 1170 | "range-parser": { 1171 | "version": "1.2.1", 1172 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 1173 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 1174 | }, 1175 | "raw-body": { 1176 | "version": "2.4.0", 1177 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", 1178 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", 1179 | "requires": { 1180 | "bytes": "3.1.0", 1181 | "http-errors": "1.7.2", 1182 | "iconv-lite": "0.4.24", 1183 | "unpipe": "1.0.0" 1184 | } 1185 | }, 1186 | "read-pkg": { 1187 | "version": "1.1.0", 1188 | "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-1.1.0.tgz", 1189 | "integrity": "sha1-9f+qXs0pyzHAR0vKfXVra7KePyg=", 1190 | "dev": true, 1191 | "requires": { 1192 | "load-json-file": "^1.0.0", 1193 | "normalize-package-data": "^2.3.2", 1194 | "path-type": "^1.0.0" 1195 | } 1196 | }, 1197 | "read-pkg-up": { 1198 | "version": "1.0.1", 1199 | "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-1.0.1.tgz", 1200 | "integrity": "sha1-nWPBMnbAZZGNV/ACpX9AobZD+wI=", 1201 | "dev": true, 1202 | "requires": { 1203 | "find-up": "^1.0.0", 1204 | "read-pkg": "^1.0.0" 1205 | } 1206 | }, 1207 | "redent": { 1208 | "version": "1.0.0", 1209 | "resolved": "https://registry.npmjs.org/redent/-/redent-1.0.0.tgz", 1210 | "integrity": "sha1-z5Fqsf1fHxbfsggi3W7H9zDCr94=", 1211 | "dev": true, 1212 | "requires": { 1213 | "indent-string": "^2.1.0", 1214 | "strip-indent": "^1.0.1" 1215 | } 1216 | }, 1217 | "remove-markdown": { 1218 | "version": "0.3.0", 1219 | "resolved": "https://registry.npmjs.org/remove-markdown/-/remove-markdown-0.3.0.tgz", 1220 | "integrity": "sha1-XktmdJOpNXlyjz1S7MHbnKUF3Jg=" 1221 | }, 1222 | "repeating": { 1223 | "version": "2.0.1", 1224 | "resolved": "https://registry.npmjs.org/repeating/-/repeating-2.0.1.tgz", 1225 | "integrity": "sha1-UhTFOpJtNVJwdSf7q0FdvAjQbdo=", 1226 | "dev": true, 1227 | "requires": { 1228 | "is-finite": "^1.0.0" 1229 | } 1230 | }, 1231 | "resolve": { 1232 | "version": "1.11.1", 1233 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.11.1.tgz", 1234 | "integrity": "sha512-vIpgF6wfuJOZI7KKKSP+HmiKggadPQAdsp5HiC1mvqnfp0gF1vdwgBWZIdrVft9pgqoMFQN+R7BSWZiBxx+BBw==", 1235 | "dev": true, 1236 | "requires": { 1237 | "path-parse": "^1.0.6" 1238 | } 1239 | }, 1240 | "rimraf": { 1241 | "version": "2.6.3", 1242 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", 1243 | "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", 1244 | "dev": true, 1245 | "requires": { 1246 | "glob": "^7.1.3" 1247 | } 1248 | }, 1249 | "safe-buffer": { 1250 | "version": "5.1.2", 1251 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1252 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 1253 | }, 1254 | "safer-buffer": { 1255 | "version": "2.1.2", 1256 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1257 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 1258 | }, 1259 | "sax": { 1260 | "version": "1.2.1", 1261 | "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.1.tgz", 1262 | "integrity": "sha1-e45lYZCyKOgaZq6nSEgNgozS03o=" 1263 | }, 1264 | "semver": { 1265 | "version": "5.7.0", 1266 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.0.tgz", 1267 | "integrity": "sha512-Ya52jSX2u7QKghxeoFGpLwCtGlt7j0oY9DYb5apt9nPlJ42ID+ulTXESnt/qAQcoSERyZ5sl3LDIOw0nAn/5DA==", 1268 | "dev": true 1269 | }, 1270 | "send": { 1271 | "version": "0.17.1", 1272 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", 1273 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", 1274 | "requires": { 1275 | "debug": "2.6.9", 1276 | "depd": "~1.1.2", 1277 | "destroy": "~1.0.4", 1278 | "encodeurl": "~1.0.2", 1279 | "escape-html": "~1.0.3", 1280 | "etag": "~1.8.1", 1281 | "fresh": "0.5.2", 1282 | "http-errors": "~1.7.2", 1283 | "mime": "1.6.0", 1284 | "ms": "2.1.1", 1285 | "on-finished": "~2.3.0", 1286 | "range-parser": "~1.2.1", 1287 | "statuses": "~1.5.0" 1288 | }, 1289 | "dependencies": { 1290 | "debug": { 1291 | "version": "2.6.9", 1292 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 1293 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 1294 | "requires": { 1295 | "ms": "2.0.0" 1296 | }, 1297 | "dependencies": { 1298 | "ms": { 1299 | "version": "2.0.0", 1300 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1301 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 1302 | } 1303 | } 1304 | }, 1305 | "ms": { 1306 | "version": "2.1.1", 1307 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 1308 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 1309 | } 1310 | } 1311 | }, 1312 | "serve-static": { 1313 | "version": "1.14.1", 1314 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", 1315 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", 1316 | "requires": { 1317 | "encodeurl": "~1.0.2", 1318 | "escape-html": "~1.0.3", 1319 | "parseurl": "~1.3.3", 1320 | "send": "0.17.1" 1321 | } 1322 | }, 1323 | "setprototypeof": { 1324 | "version": "1.1.1", 1325 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", 1326 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" 1327 | }, 1328 | "shellwords": { 1329 | "version": "0.1.1", 1330 | "resolved": "https://registry.npmjs.org/shellwords/-/shellwords-0.1.1.tgz", 1331 | "integrity": "sha512-vFwSUfQvqybiICwZY5+DAWIPLKsWO31Q91JSKl3UYv+K5c2QRPzn0qzec6QPu1Qc9eHYItiP3NdJqNVqetYAww==", 1332 | "dev": true 1333 | }, 1334 | "signal-exit": { 1335 | "version": "3.0.2", 1336 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", 1337 | "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", 1338 | "dev": true 1339 | }, 1340 | "source-map": { 1341 | "version": "0.6.1", 1342 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 1343 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==" 1344 | }, 1345 | "source-map-support": { 1346 | "version": "0.5.12", 1347 | "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.12.tgz", 1348 | "integrity": "sha512-4h2Pbvyy15EE02G+JOZpUCmqWJuqrs+sEkzewTm++BPi7Hvn/HwcqLAcNxYAyI0x13CpPPn+kMjl+hplXMHITQ==", 1349 | "requires": { 1350 | "buffer-from": "^1.0.0", 1351 | "source-map": "^0.6.0" 1352 | } 1353 | }, 1354 | "spdx-correct": { 1355 | "version": "3.1.0", 1356 | "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.0.tgz", 1357 | "integrity": "sha512-lr2EZCctC2BNR7j7WzJ2FpDznxky1sjfxvvYEyzxNyb6lZXHODmEoJeFu4JupYlkfha1KZpJyoqiJ7pgA1qq8Q==", 1358 | "dev": true, 1359 | "requires": { 1360 | "spdx-expression-parse": "^3.0.0", 1361 | "spdx-license-ids": "^3.0.0" 1362 | } 1363 | }, 1364 | "spdx-exceptions": { 1365 | "version": "2.2.0", 1366 | "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.2.0.tgz", 1367 | "integrity": "sha512-2XQACfElKi9SlVb1CYadKDXvoajPgBVPn/gOQLrTvHdElaVhr7ZEbqJaRnJLVNeaI4cMEAgVCeBMKF6MWRDCRA==", 1368 | "dev": true 1369 | }, 1370 | "spdx-expression-parse": { 1371 | "version": "3.0.0", 1372 | "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.0.tgz", 1373 | "integrity": "sha512-Yg6D3XpRD4kkOmTpdgbUiEJFKghJH03fiC1OPll5h/0sO6neh2jqRDVHOQ4o/LMea0tgCkbMgea5ip/e+MkWyg==", 1374 | "dev": true, 1375 | "requires": { 1376 | "spdx-exceptions": "^2.1.0", 1377 | "spdx-license-ids": "^3.0.0" 1378 | } 1379 | }, 1380 | "spdx-license-ids": { 1381 | "version": "3.0.4", 1382 | "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.4.tgz", 1383 | "integrity": "sha512-7j8LYJLeY/Yb6ACbQ7F76qy5jHkp0U6jgBfJsk97bwWlVUnUWsAgpyaCvo17h0/RQGnQ036tVDomiwoI4pDkQA==", 1384 | "dev": true 1385 | }, 1386 | "sprintf-js": { 1387 | "version": "1.0.3", 1388 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 1389 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", 1390 | "dev": true 1391 | }, 1392 | "statuses": { 1393 | "version": "1.5.0", 1394 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", 1395 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" 1396 | }, 1397 | "strip-bom": { 1398 | "version": "2.0.0", 1399 | "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-2.0.0.tgz", 1400 | "integrity": "sha1-YhmoVhZSBJHzV4i9vxRHqZx+aw4=", 1401 | "dev": true, 1402 | "requires": { 1403 | "is-utf8": "^0.2.0" 1404 | } 1405 | }, 1406 | "strip-indent": { 1407 | "version": "1.0.1", 1408 | "resolved": "https://registry.npmjs.org/strip-indent/-/strip-indent-1.0.1.tgz", 1409 | "integrity": "sha1-DHlipq3vp7vUrDZkYKY4VSrhoKI=", 1410 | "dev": true, 1411 | "requires": { 1412 | "get-stdin": "^4.0.1" 1413 | } 1414 | }, 1415 | "strip-json-comments": { 1416 | "version": "2.0.1", 1417 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 1418 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 1419 | "dev": true 1420 | }, 1421 | "supports-color": { 1422 | "version": "5.5.0", 1423 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1424 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1425 | "dev": true, 1426 | "requires": { 1427 | "has-flag": "^3.0.0" 1428 | } 1429 | }, 1430 | "toidentifier": { 1431 | "version": "1.0.0", 1432 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", 1433 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" 1434 | }, 1435 | "tree-kill": { 1436 | "version": "1.2.2", 1437 | "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", 1438 | "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", 1439 | "dev": true 1440 | }, 1441 | "trim-newlines": { 1442 | "version": "1.0.0", 1443 | "resolved": "https://registry.npmjs.org/trim-newlines/-/trim-newlines-1.0.0.tgz", 1444 | "integrity": "sha1-WIeWa7WCpFA6QetST301ARgVphM=", 1445 | "dev": true 1446 | }, 1447 | "ts-httpexceptions": { 1448 | "version": "4.1.0", 1449 | "resolved": "https://registry.npmjs.org/ts-httpexceptions/-/ts-httpexceptions-4.1.0.tgz", 1450 | "integrity": "sha512-eNppUDEIt7zE/YnJ9tIvDnqzvs41qssS23+QsgOrUm+MZXoVsohAP+XBDAah5QWMk0nOyaLkS7EUpN8HeAdhSA==", 1451 | "requires": { 1452 | "tslib": "^1.9.3" 1453 | } 1454 | }, 1455 | "ts-node": { 1456 | "version": "8.2.0", 1457 | "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-8.2.0.tgz", 1458 | "integrity": "sha512-m8XQwUurkbYqXrKqr3WHCW310utRNvV5OnRVeISeea7LoCWVcdfeB/Ntl8JYWFh+WRoUAdBgESrzKochQt7sMw==", 1459 | "requires": { 1460 | "arg": "^4.1.0", 1461 | "diff": "^4.0.1", 1462 | "make-error": "^1.1.1", 1463 | "source-map-support": "^0.5.6", 1464 | "yn": "^3.0.0" 1465 | } 1466 | }, 1467 | "ts-node-dev": { 1468 | "version": "1.0.0-pre.39", 1469 | "resolved": "https://registry.npmjs.org/ts-node-dev/-/ts-node-dev-1.0.0-pre.39.tgz", 1470 | "integrity": "sha512-yOg9nMAi6U2HcAkhnFuWxfg53XDqpdbeBESo+7DfmlDpQX4RrEzBNV6szOjlm/OH0KiRgG5J0emvg/BS/gQmXQ==", 1471 | "dev": true, 1472 | "requires": { 1473 | "dateformat": "~1.0.4-1.2.3", 1474 | "dynamic-dedupe": "^0.3.0", 1475 | "filewatcher": "~3.0.0", 1476 | "minimist": "^1.1.3", 1477 | "mkdirp": "^0.5.1", 1478 | "node-notifier": "^5.4.0", 1479 | "resolve": "^1.0.0", 1480 | "rimraf": "^2.6.1", 1481 | "tree-kill": "^1.2.1", 1482 | "ts-node": "*", 1483 | "tsconfig": "^7.0.0" 1484 | } 1485 | }, 1486 | "tsconfig": { 1487 | "version": "7.0.0", 1488 | "resolved": "https://registry.npmjs.org/tsconfig/-/tsconfig-7.0.0.tgz", 1489 | "integrity": "sha512-vZXmzPrL+EmC4T/4rVlT2jNVMWCi/O4DIiSj3UHg1OE5kCKbk4mfrXc6dZksLgRM/TZlKnousKH9bbTazUWRRw==", 1490 | "dev": true, 1491 | "requires": { 1492 | "@types/strip-bom": "^3.0.0", 1493 | "@types/strip-json-comments": "0.0.30", 1494 | "strip-bom": "^3.0.0", 1495 | "strip-json-comments": "^2.0.0" 1496 | }, 1497 | "dependencies": { 1498 | "strip-bom": { 1499 | "version": "3.0.0", 1500 | "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", 1501 | "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", 1502 | "dev": true 1503 | } 1504 | } 1505 | }, 1506 | "tslib": { 1507 | "version": "1.9.3", 1508 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.9.3.tgz", 1509 | "integrity": "sha512-4krF8scpejhaOgqzBEcGM7yDIEfi0/8+8zDRZhNZZ2kjmHJ4hv3zCbQWxoJGz1iw5U0Jl0nma13xzHXcncMavQ==" 1510 | }, 1511 | "tslint": { 1512 | "version": "5.17.0", 1513 | "resolved": "https://registry.npmjs.org/tslint/-/tslint-5.17.0.tgz", 1514 | "integrity": "sha512-pflx87WfVoYepTet3xLfDOLDm9Jqi61UXIKePOuca0qoAZyrGWonDG9VTbji58Fy+8gciUn8Bt7y69+KEVjc/w==", 1515 | "dev": true, 1516 | "requires": { 1517 | "@babel/code-frame": "^7.0.0", 1518 | "builtin-modules": "^1.1.1", 1519 | "chalk": "^2.3.0", 1520 | "commander": "^2.12.1", 1521 | "diff": "^3.2.0", 1522 | "glob": "^7.1.1", 1523 | "js-yaml": "^3.13.1", 1524 | "minimatch": "^3.0.4", 1525 | "mkdirp": "^0.5.1", 1526 | "resolve": "^1.3.2", 1527 | "semver": "^5.3.0", 1528 | "tslib": "^1.8.0", 1529 | "tsutils": "^2.29.0" 1530 | }, 1531 | "dependencies": { 1532 | "diff": { 1533 | "version": "3.5.0", 1534 | "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", 1535 | "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", 1536 | "dev": true 1537 | } 1538 | } 1539 | }, 1540 | "tsutils": { 1541 | "version": "2.29.0", 1542 | "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", 1543 | "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", 1544 | "dev": true, 1545 | "requires": { 1546 | "tslib": "^1.8.1" 1547 | } 1548 | }, 1549 | "type-is": { 1550 | "version": "1.6.18", 1551 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 1552 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 1553 | "requires": { 1554 | "media-typer": "0.3.0", 1555 | "mime-types": "~2.1.24" 1556 | } 1557 | }, 1558 | "typescript": { 1559 | "version": "3.5.1", 1560 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.5.1.tgz", 1561 | "integrity": "sha512-64HkdiRv1yYZsSe4xC1WVgamNigVYjlssIoaH2HcZF0+ijsk5YK2g0G34w9wJkze8+5ow4STd22AynfO6ZYYLw==" 1562 | }, 1563 | "universalify": { 1564 | "version": "0.1.2", 1565 | "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", 1566 | "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==" 1567 | }, 1568 | "unpipe": { 1569 | "version": "1.0.0", 1570 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1571 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 1572 | }, 1573 | "url": { 1574 | "version": "0.10.3", 1575 | "resolved": "https://registry.npmjs.org/url/-/url-0.10.3.tgz", 1576 | "integrity": "sha1-Ah5NnHcF8hu/N9A861h2dAJ3TGQ=", 1577 | "requires": { 1578 | "punycode": "1.3.2", 1579 | "querystring": "0.2.0" 1580 | } 1581 | }, 1582 | "utils-merge": { 1583 | "version": "1.0.1", 1584 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 1585 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 1586 | }, 1587 | "uuid": { 1588 | "version": "3.3.2", 1589 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.3.2.tgz", 1590 | "integrity": "sha512-yXJmeNaw3DnnKAOKJE51sL/ZaYfWJRl1pK9dr19YFCu0ObS231AB1/LbqTKRAQ5kw8A90rA6fr4riOUpTZvQZA==" 1591 | }, 1592 | "validate-npm-package-license": { 1593 | "version": "3.0.4", 1594 | "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", 1595 | "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", 1596 | "dev": true, 1597 | "requires": { 1598 | "spdx-correct": "^3.0.0", 1599 | "spdx-expression-parse": "^3.0.0" 1600 | } 1601 | }, 1602 | "vary": { 1603 | "version": "1.1.2", 1604 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 1605 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 1606 | }, 1607 | "which": { 1608 | "version": "1.3.1", 1609 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", 1610 | "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", 1611 | "dev": true, 1612 | "requires": { 1613 | "isexe": "^2.0.0" 1614 | } 1615 | }, 1616 | "wrappy": { 1617 | "version": "1.0.2", 1618 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1619 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1620 | "dev": true 1621 | }, 1622 | "xml2js": { 1623 | "version": "0.4.19", 1624 | "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.19.tgz", 1625 | "integrity": "sha512-esZnJZJOiJR9wWKMyuvSE1y6Dq5LCuJanqhxslH2bxM6duahNZ+HMpCLhBQGZkbX6xRf8x1Y2eJlgt2q3qo49Q==", 1626 | "requires": { 1627 | "sax": ">=0.6.0", 1628 | "xmlbuilder": "~9.0.1" 1629 | } 1630 | }, 1631 | "xmlbuilder": { 1632 | "version": "9.0.7", 1633 | "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-9.0.7.tgz", 1634 | "integrity": "sha1-Ey7mPS7FVlxVfiD0wi35rKaGsQ0=" 1635 | }, 1636 | "xtend": { 1637 | "version": "4.0.1", 1638 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.1.tgz", 1639 | "integrity": "sha1-pcbVMr5lbiPbgg77lDofBJmNY68=", 1640 | "dev": true 1641 | }, 1642 | "yn": { 1643 | "version": "3.1.0", 1644 | "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.0.tgz", 1645 | "integrity": "sha512-kKfnnYkbTfrAdd0xICNFw7Atm8nKpLcLv9AZGEt+kczL/WQVai4e2V6ZN8U/O+iI6WrNuJjNNOyu4zfhl9D3Hg==" 1646 | } 1647 | } 1648 | } 1649 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "@algorithm-visualizer/server", 3 | "version": "2.0.0", 4 | "title": "Algorithm Visualizer", 5 | "description": "Algorithm Visualizer is an interactive online platform that visualizes algorithms from code.", 6 | "scripts": { 7 | "watch": "NODE_ENV=development NODE_PATH=src ts-node-dev --respawn --ignore-watch node_modules --no-notify src", 8 | "start": "NODE_ENV=production NODE_PATH=src ts-node --transpile-only src", 9 | "tslint": "tslint -c tslint.json -p tsconfig.json" 10 | }, 11 | "repository": { 12 | "type": "git", 13 | "url": "git+https://github.com/algorithm-visualizer/server.git" 14 | }, 15 | "license": "MIT", 16 | "devDependencies": { 17 | "@types/compression": "0.0.36", 18 | "@types/execa": "^0.9.0", 19 | "@types/express": "^4.16.1", 20 | "@types/fs-extra": "^7.0.0", 21 | "@types/morgan": "^1.7.35", 22 | "@types/node": "^12.0.0", 23 | "@types/node-cron": "^3.0.1", 24 | "@types/remove-markdown": "^0.1.1", 25 | "@types/uuid": "^3.4.4", 26 | "ts-node-dev": "^1.0.0-pre.39", 27 | "tslint": "^5.16.0" 28 | }, 29 | "dependencies": { 30 | "aws-sdk": "^2.814.0", 31 | "axios": "^0.21.2", 32 | "body-parser": "^1.18.2", 33 | "compression": "^1.7.3", 34 | "dotenv": "^8.0.0", 35 | "dotenv-flow": "^2.0.0", 36 | "express": "^4.16.4", 37 | "express-github-webhook": "^1.0.6", 38 | "fs-extra": "^6.0.1", 39 | "morgan": "^1.9.1", 40 | "node-cron": "^3.0.0", 41 | "remove-markdown": "^0.3.0", 42 | "ts-httpexceptions": "^4.1.0", 43 | "ts-node": "^8.1.0", 44 | "typescript": "^3.4.5", 45 | "uuid": "^3.3.2" 46 | } 47 | } 48 | -------------------------------------------------------------------------------- /pm2.json: -------------------------------------------------------------------------------- 1 | { 2 | "apps": [ 3 | { 4 | "name": "algorithm-visualizer", 5 | "script": "npm start" 6 | } 7 | ] 8 | } 9 | -------------------------------------------------------------------------------- /src/Server.ts: -------------------------------------------------------------------------------- 1 | import express, { NextFunction, Request, Response } from 'express'; 2 | import morgan from 'morgan'; 3 | import bodyParser from 'body-parser'; 4 | import * as Controllers from 'controllers'; 5 | import { NotFound } from 'ts-httpexceptions'; 6 | import compression from 'compression'; 7 | import { __PROD__, credentials, httpPort, httpsPort, webhookOptions } from 'config/environments'; 8 | import http from 'http'; 9 | import https from 'https'; 10 | import cron from 'node-cron'; 11 | import { Hierarchy } from 'models'; 12 | import * as Tracers from 'tracers'; 13 | import { errorHandlerMiddleware, frontendMiddleware, redirectMiddleware } from 'middlewares'; 14 | import { execute, issueHttpsCertificate, pull } from 'utils/misc'; 15 | import { frontendBuildDir, frontendBuiltDir, frontendDir, rootDir } from 'config/paths'; 16 | 17 | const Webhook = require('express-github-webhook'); 18 | 19 | export default class Server { 20 | readonly hierarchy = new Hierarchy(); 21 | readonly tracers = Object.values(Tracers).map(Tracer => new Tracer()); 22 | private readonly app = express(); 23 | private readonly webhook = webhookOptions && Webhook(webhookOptions); 24 | 25 | constructor() { 26 | this.app 27 | .use(compression()) 28 | .use(morgan(__PROD__ ? 'tiny' : 'dev')) 29 | .use(redirectMiddleware()) 30 | .use(bodyParser.json()) 31 | .use(bodyParser.urlencoded({ extended: true })) 32 | .use('/api', this.getApiRouter()) 33 | .use(frontendMiddleware(this)); 34 | if (this.webhook) { 35 | this.app.use(this.webhook); 36 | } 37 | this.app.use(errorHandlerMiddleware()); 38 | 39 | if (this.webhook) { 40 | this.webhook.on('push', async (repo: string, data: any) => { 41 | const { ref, head_commit } = data; 42 | if (ref !== 'refs/heads/master') return; 43 | if (!head_commit) throw new Error('The `head_commit` is empty.'); 44 | 45 | switch (repo) { 46 | case 'server': 47 | await this.update(head_commit.id); 48 | break; 49 | case 'algorithm-visualizer': 50 | await this.updateFrontend(head_commit.id); 51 | break; 52 | case 'algorithms': 53 | await this.hierarchy.update(head_commit.id); 54 | break; 55 | default: 56 | throw new Error(`Webhook from unknown repository '${repo}'.`); 57 | } 58 | }); 59 | 60 | this.webhook.on('release', async (repo: string, data: any) => { 61 | const tracer = this.tracers.find(tracer => repo === `tracers.${tracer.lang}`); 62 | if (!tracer) throw new Error(`Tracer not found for repository '${repo}'.`); 63 | await tracer.update(data.release); 64 | }); 65 | } 66 | 67 | if (credentials) { 68 | cron.schedule('0 0 1 * *', () => { 69 | issueHttpsCertificate(); 70 | }); 71 | } 72 | } 73 | 74 | getApiRouter() { 75 | const router = express.Router(); 76 | Object.values(Controllers).forEach(Controller => new Controller(this).route(router)); 77 | router.use((req: Request, res: Response, next: NextFunction) => { 78 | next(new NotFound('API not found.')); 79 | }); 80 | return router; 81 | }; 82 | 83 | async update(commit?: string) { 84 | await pull(rootDir, 'server', commit); 85 | await execute('npm install', { 86 | cwd: rootDir, 87 | stdout: process.stdout, 88 | stderr: process.stderr, 89 | }); 90 | process.exit(0); 91 | }; 92 | 93 | async updateFrontend(commit?: string) { 94 | await pull(frontendDir, 'algorithm-visualizer', commit); 95 | await execute([ 96 | 'npm install', 97 | 'npm run build', 98 | `rm -rf ${frontendBuiltDir}`, 99 | `mv ${frontendBuildDir} ${frontendBuiltDir}`, 100 | ].join(' && '), { 101 | cwd: frontendDir, 102 | stdout: process.stdout, 103 | stderr: process.stderr, 104 | }); 105 | } 106 | 107 | start() { 108 | const httpServer = http.createServer(this.app); 109 | httpServer.listen(httpPort); 110 | console.info(`http: listening on port ${httpPort}`); 111 | 112 | if (credentials) { 113 | const httpsServer = https.createServer(credentials, this.app); 114 | httpsServer.listen(httpsPort); 115 | console.info(`https: listening on port ${httpsPort}`); 116 | } 117 | } 118 | } 119 | -------------------------------------------------------------------------------- /src/config/constants.ts: -------------------------------------------------------------------------------- 1 | export const memoryLimit = 256; // in megabytes 2 | export const timeLimit = 5000; // in milliseconds 3 | -------------------------------------------------------------------------------- /src/config/environments.ts: -------------------------------------------------------------------------------- 1 | import fs from 'fs'; 2 | import { ServerOptions } from 'https'; 3 | import path from 'path'; 4 | import { issueHttpsCertificate } from '../utils/misc'; 5 | 6 | require('dotenv-flow').config(); 7 | 8 | const { 9 | NODE_ENV, 10 | 11 | HTTP_PORT, 12 | HTTPS_PORT, 13 | 14 | CREDENTIALS_ENABLED, 15 | CREDENTIALS_PATH, 16 | CREDENTIALS_CA, 17 | CREDENTIALS_KEY, 18 | CREDENTIALS_CERT, 19 | 20 | WEBHOOK_ENABLED, 21 | WEBHOOK_SECRET, 22 | 23 | GITHUB_CLIENT_ID, 24 | GITHUB_CLIENT_SECRET, 25 | 26 | AWS_ACCESS_KEY_ID, 27 | AWS_SECRET_ACCESS_KEY, 28 | } = process.env as { 29 | [key: string]: string, 30 | }; 31 | 32 | const isEnabled = (v: string) => v === '1'; 33 | 34 | const missingVars = [ 35 | 'NODE_ENV', 36 | 'HTTP_PORT', 37 | 'HTTPS_PORT', 38 | 'CREDENTIALS_ENABLED', 39 | ...(isEnabled(CREDENTIALS_ENABLED) ? [ 40 | 'CREDENTIALS_PATH', 41 | 'CREDENTIALS_CA', 42 | 'CREDENTIALS_KEY', 43 | 'CREDENTIALS_CERT', 44 | ] : []), 45 | 'WEBHOOK_ENABLED', 46 | ...(isEnabled(WEBHOOK_ENABLED) ? [ 47 | 'WEBHOOK_SECRET', 48 | ] : []), 49 | 'GITHUB_CLIENT_ID', 50 | 'GITHUB_CLIENT_SECRET', 51 | 'AWS_ACCESS_KEY_ID', 52 | 'AWS_SECRET_ACCESS_KEY', 53 | ].filter(variable => process.env[variable] === undefined); 54 | if (missingVars.length) throw new Error(`The following environment variables are missing: ${missingVars.join(', ')}`); 55 | 56 | export const __PROD__ = NODE_ENV === 'production'; 57 | export const __DEV__ = NODE_ENV === 'development'; 58 | 59 | export const httpPort = parseInt(HTTP_PORT); 60 | export const httpsPort = parseInt(HTTPS_PORT); 61 | 62 | export const webhookOptions = isEnabled(WEBHOOK_ENABLED) ? { 63 | path: '/webhook', 64 | secret: WEBHOOK_SECRET, 65 | } : undefined; 66 | 67 | export let credentials: ServerOptions | undefined; 68 | if (isEnabled(CREDENTIALS_ENABLED)) { 69 | if (fs.existsSync(CREDENTIALS_PATH)) { 70 | const readCredentials = (file: string) => fs.readFileSync(path.resolve(CREDENTIALS_PATH, file)); 71 | credentials = { 72 | ca: readCredentials(CREDENTIALS_CA), 73 | key: readCredentials(CREDENTIALS_KEY), 74 | cert: readCredentials(CREDENTIALS_CERT), 75 | }; 76 | } else { 77 | issueHttpsCertificate(); 78 | } 79 | } 80 | 81 | export const githubClientId = GITHUB_CLIENT_ID; 82 | export const githubClientSecret = GITHUB_CLIENT_SECRET; 83 | 84 | export const awsAccessKeyId = AWS_ACCESS_KEY_ID; 85 | export const awsSecretAccessKey = AWS_SECRET_ACCESS_KEY; 86 | -------------------------------------------------------------------------------- /src/config/paths.ts: -------------------------------------------------------------------------------- 1 | import path from 'path'; 2 | 3 | export const rootDir = path.resolve(__dirname, '..', '..'); 4 | export const publicDir = path.resolve(rootDir, 'public'); 5 | export const algorithmsDir = path.resolve(publicDir, 'algorithms'); 6 | export const codesDir = path.resolve(publicDir, 'codes'); 7 | export const visualizationsDir = path.resolve(publicDir, 'visualizations'); 8 | export const frontendDir = path.resolve(publicDir, 'frontend'); 9 | export const frontendBuildDir = path.resolve(frontendDir, 'build'); 10 | export const frontendBuiltDir = path.resolve(publicDir, 'frontend-built'); 11 | -------------------------------------------------------------------------------- /src/controllers/AlgorithmsController.ts: -------------------------------------------------------------------------------- 1 | import express from 'express'; 2 | import { Controller } from 'controllers/Controller'; 3 | import { NotFound } from 'ts-httpexceptions'; 4 | import Server from 'Server'; 5 | 6 | export class AlgorithmsController extends Controller { 7 | constructor(server: Server) { 8 | super(server); 9 | this.router 10 | .get('/', this.getHierarchy) 11 | .get('/:categoryKey/:algorithmKey', this.getAlgorithm) 12 | .get('/sitemap.txt', this.getSitemap); 13 | } 14 | 15 | route = (router: express.Router): void => { 16 | router.use('/algorithms', this.router); 17 | }; 18 | 19 | getHierarchy = (req: express.Request, res: express.Response) => { 20 | res.json(this.server.hierarchy); 21 | }; 22 | 23 | getAlgorithm = (req: express.Request, res: express.Response, next: express.NextFunction) => { 24 | const {categoryKey, algorithmKey} = req.params; 25 | const algorithm = this.server.hierarchy.find(categoryKey, algorithmKey); 26 | if (!algorithm) return next(new NotFound('Algorithm not found.')); 27 | res.json({algorithm}); 28 | }; 29 | 30 | getSitemap = (req: express.Request, res: express.Response) => { 31 | const urls: string[] = []; 32 | this.server.hierarchy.iterate((category, algorithm) => { 33 | urls.push(`https://algorithm-visualizer.org/${category.key}/${algorithm.key}`); 34 | }); 35 | res.set('Content-Type', 'text/plain'); 36 | res.send(urls.join('\n')); 37 | }; 38 | } 39 | -------------------------------------------------------------------------------- /src/controllers/AuthController.ts: -------------------------------------------------------------------------------- 1 | import express from 'express'; 2 | import { githubClientId } from 'config/environments'; 3 | import { GitHubApi } from 'utils/apis'; 4 | import { Controller } from 'controllers/Controller'; 5 | import Server from 'Server'; 6 | 7 | export class AuthController extends Controller { 8 | constructor(server: Server) { 9 | super(server); 10 | this.router 11 | .get('/request', this.request) 12 | .get('/response', this.response) 13 | .get('/destroy', this.destroy); 14 | } 15 | 16 | route = (router: express.Router): void => { 17 | router.use('/auth', this.router); 18 | }; 19 | 20 | request = (req: express.Request, res: express.Response) => { 21 | res.redirect(`https://github.com/login/oauth/authorize?client_id=${githubClientId}&scope=user,gist`); 22 | }; 23 | 24 | response = (req: express.Request, res: express.Response, next: express.NextFunction) => { 25 | const {code} = req.query; 26 | 27 | GitHubApi.getAccessToken(code).then(({data}) => { 28 | const {access_token} = data; 29 | res.send(``); 30 | }).catch(next); 31 | }; 32 | 33 | destroy = (req: express.Request, res: express.Response) => { 34 | res.send(``); 35 | }; 36 | } 37 | -------------------------------------------------------------------------------- /src/controllers/Controller.ts: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | import Server from 'Server'; 3 | 4 | export abstract class Controller { 5 | protected readonly router: express.Router; 6 | 7 | protected constructor(protected server: Server) { 8 | this.router = express.Router(); 9 | } 10 | 11 | abstract route(router: express.Router): void; 12 | } 13 | -------------------------------------------------------------------------------- /src/controllers/TracersController.ts: -------------------------------------------------------------------------------- 1 | import express from 'express'; 2 | import { Controller } from 'controllers/Controller'; 3 | import Server from 'Server'; 4 | 5 | export class TracersController extends Controller { 6 | constructor(server: Server) { 7 | super(server); 8 | this.server.tracers.forEach(tracer => tracer.route(this.router)); 9 | } 10 | 11 | route = (router: express.Router): void => { 12 | router.use('/tracers', this.router); 13 | }; 14 | } 15 | -------------------------------------------------------------------------------- /src/controllers/VisualizationsController.ts: -------------------------------------------------------------------------------- 1 | import express from 'express'; 2 | import path from 'path'; 3 | import uuid from 'uuid'; 4 | import fs from 'fs-extra'; 5 | import { Controller } from 'controllers/Controller'; 6 | import Server from 'Server'; 7 | import { visualizationsDir } from 'config/paths'; 8 | 9 | export class VisualizationsController extends Controller { 10 | constructor(server: Server) { 11 | super(server); 12 | this.router 13 | .post('/', this.uploadVisualization) 14 | .get('/:visualizationId', this.getVisualization); 15 | 16 | fs.remove(visualizationsDir).catch(console.error); 17 | } 18 | 19 | route = (router: express.Router): void => { 20 | router.use('/visualizations', this.router); 21 | }; 22 | 23 | uploadVisualization = (req: express.Request, res: express.Response, next: express.NextFunction) => { 24 | const {content} = req.body; 25 | const visualizationId = uuid.v4(); 26 | const visualizationPath = path.resolve(visualizationsDir, `${visualizationId}.json`); 27 | const url = `https://algorithm-visualizer.org/scratch-paper/new?visualizationId=${visualizationId}`; 28 | fs.outputFile(visualizationPath, content) 29 | .then(() => res.send(url)) 30 | .catch(next); 31 | }; 32 | 33 | getVisualization = (req: express.Request, res: express.Response, next: express.NextFunction) => { 34 | const {visualizationId} = req.params; 35 | const visualizationPath = path.resolve(visualizationsDir, `${visualizationId}.json`); 36 | res.sendFile(visualizationPath, err => { 37 | if (err) next(new Error('Visualization Expired')); 38 | fs.remove(visualizationPath); 39 | }); 40 | }; 41 | } 42 | -------------------------------------------------------------------------------- /src/controllers/index.ts: -------------------------------------------------------------------------------- 1 | export { AlgorithmsController } from './AlgorithmsController'; 2 | export { AuthController } from './AuthController'; 3 | export { TracersController } from './TracersController'; 4 | export { VisualizationsController } from './VisualizationsController'; 5 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | import Server from 'Server'; 2 | 3 | new Server().start(); 4 | -------------------------------------------------------------------------------- /src/middlewares/errorHandlerMiddleware.ts: -------------------------------------------------------------------------------- 1 | import { NextFunction, Request, Response } from 'express'; 2 | import { Exception, InternalServerError } from 'ts-httpexceptions'; 3 | 4 | export function errorHandlerMiddleware() { 5 | return (err: any, req: Request, res: Response, next: NextFunction) => { 6 | if (!(err instanceof Exception)) { 7 | console.error(err); 8 | err = new InternalServerError(err.message, err); 9 | } 10 | 11 | const {message, status} = err; 12 | res.status(status).send(message); 13 | }; 14 | } 15 | -------------------------------------------------------------------------------- /src/middlewares/frontendMiddleware.ts: -------------------------------------------------------------------------------- 1 | import express, { NextFunction, Request, Response } from 'express'; 2 | import path from 'path'; 3 | import fs from 'fs-extra'; 4 | import url from 'url'; 5 | import Server from 'Server'; 6 | import { frontendBuiltDir } from 'config/paths'; 7 | 8 | const packageJson = require('../../package.json'); 9 | 10 | export function frontendMiddleware(server: Server) { 11 | const staticMiddleware = express.static(frontendBuiltDir, {index: false}); 12 | 13 | if (!fs.pathExistsSync(frontendBuiltDir)) { 14 | server.updateFrontend().catch(console.error); 15 | } 16 | 17 | return (req: Request, res: Response, next: NextFunction) => { 18 | staticMiddleware(req, res, err => { 19 | if (err) return next(err); 20 | if (req.method !== 'GET') return next(); 21 | 22 | const filePath = path.resolve(frontendBuiltDir, 'index.html'); 23 | fs.readFile(filePath, 'utf8', (err, data) => { 24 | if (err) return next(err); 25 | 26 | const {pathname} = url.parse(req.originalUrl); 27 | if (!pathname) return next(new Error('Failed to get `pathname`.')); 28 | const [, categoryKey, algorithmKey] = pathname.split('/'); 29 | let {title, description} = packageJson; 30 | let algorithm = undefined; 31 | if (categoryKey && categoryKey !== 'scratch-paper') { 32 | algorithm = server.hierarchy.find(categoryKey, algorithmKey) || null; 33 | if (algorithm) { 34 | title = [algorithm.categoryName, algorithm.algorithmName].join(' - '); 35 | description = algorithm.description; 36 | } else { 37 | res.status(404); 38 | return; 39 | } 40 | } 41 | 42 | const indexFile = data 43 | .replace(/\$TITLE/g, title) 44 | .replace(/\$DESCRIPTION/g, description) 45 | .replace(/\$ALGORITHM/g, algorithm === undefined ? 'undefined' : 46 | JSON.stringify(algorithm).replace(/ { 6 | if (req.hostname === 'algo-visualizer.jasonpark.me') { 7 | res.redirect(301, 'https://algorithm-visualizer.org/'); 8 | } else if (credentials && !req.secure) { 9 | res.redirect(301, `https://${req.hostname}${req.url}`); 10 | } else { 11 | next(); 12 | } 13 | }; 14 | } 15 | -------------------------------------------------------------------------------- /src/models/Algorithm.ts: -------------------------------------------------------------------------------- 1 | import path from 'path'; 2 | import { createKey, listFiles } from 'utils/hierarchy'; 3 | import { File } from 'models'; 4 | import { getDescription } from 'utils/misc'; 5 | 6 | export class Algorithm { 7 | key: string; 8 | files!: File[]; 9 | description!: string; 10 | 11 | constructor(private path: string, public name: string) { 12 | this.key = createKey(name); 13 | this.refresh(); 14 | } 15 | 16 | refresh() { 17 | this.files = listFiles(this.path) 18 | .map(fileName => new File(path.resolve(this.path, fileName), fileName)); 19 | this.description = getDescription(this.files); 20 | } 21 | 22 | toJSON() { 23 | const {key, name} = this; 24 | return {key, name}; 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /src/models/Category.ts: -------------------------------------------------------------------------------- 1 | import path from 'path'; 2 | import { createKey, listDirectories } from 'utils/hierarchy'; 3 | import { Algorithm } from 'models'; 4 | 5 | export class Category { 6 | key: string; 7 | algorithms!: Algorithm[]; 8 | 9 | constructor(private path: string, public name: string) { 10 | this.key = createKey(name); 11 | this.refresh(); 12 | } 13 | 14 | refresh() { 15 | this.algorithms = listDirectories(this.path) 16 | .map(algorithmName => new Algorithm(path.resolve(this.path, algorithmName), algorithmName)); 17 | } 18 | 19 | toJSON() { 20 | const {key, name, algorithms} = this; 21 | return {key, name, algorithms}; 22 | } 23 | } 24 | 25 | export default Category; 26 | -------------------------------------------------------------------------------- /src/models/File.ts: -------------------------------------------------------------------------------- 1 | import fs from 'fs-extra'; 2 | 3 | export type Author = { 4 | login: string, 5 | avatar_url: string, 6 | } 7 | 8 | export class File { 9 | content!: string; 10 | contributors!: Author[]; 11 | 12 | constructor(public path: string, public name: string) { 13 | this.refresh(); 14 | } 15 | 16 | refresh() { 17 | this.content = fs.readFileSync(this.path, 'utf-8'); 18 | this.contributors = []; 19 | } 20 | 21 | toJSON() { 22 | const {name, content, contributors} = this; 23 | return {name, content, contributors}; 24 | } 25 | } 26 | -------------------------------------------------------------------------------- /src/models/Hierarchy.ts: -------------------------------------------------------------------------------- 1 | import path from 'path'; 2 | import { listDirectories } from 'utils/hierarchy'; 3 | import { GitHubApi } from 'utils/apis'; 4 | import { Algorithm, Category, File } from 'models'; 5 | import { Author } from 'models/File'; 6 | import { algorithmsDir } from 'config/paths'; 7 | import { execute, pull } from 'utils/misc'; 8 | 9 | type CommitAuthors = { 10 | [sha: string]: Author, 11 | } 12 | 13 | export class Hierarchy { 14 | private categories!: Category[]; 15 | readonly path: string = algorithmsDir; 16 | 17 | constructor() { 18 | this.refresh(); 19 | this.update().catch(console.error); 20 | } 21 | 22 | refresh() { 23 | this.categories = listDirectories(this.path) 24 | .map(categoryName => new Category(path.resolve(this.path, categoryName), categoryName)); 25 | 26 | const files: File[] = []; 27 | this.categories.forEach(category => category.algorithms.forEach(algorithm => files.push(...algorithm.files))); 28 | this.cacheCommitAuthors().then(commitAuthors => this.cacheContributors(files, commitAuthors)); 29 | } 30 | 31 | async update(commit?: string) { 32 | await pull(this.path, 'algorithms', commit); 33 | this.refresh(); 34 | }; 35 | 36 | async cacheCommitAuthors(page = 1, commitAuthors: CommitAuthors = {}): Promise { 37 | const per_page = 100; 38 | const {data} = await GitHubApi.listCommits('algorithm-visualizer', 'algorithms', {per_page, page}); 39 | const commits: any[] = data; 40 | for (const {sha, author} of commits) { 41 | if (!author) continue; 42 | const {login, avatar_url} = author; 43 | commitAuthors[sha] = {login, avatar_url}; 44 | } 45 | if (commits.length < per_page) { 46 | return commitAuthors; 47 | } else { 48 | return this.cacheCommitAuthors(page + 1, commitAuthors); 49 | } 50 | } 51 | 52 | async cacheContributors(files: File[], commitAuthors: CommitAuthors) { 53 | for (const file of files) { 54 | const stdout = await execute(`git --no-pager log --follow --no-merges --format="%H" -- "${path.relative(this.path, file.path)}"`, { 55 | cwd: this.path, 56 | }); 57 | const output = stdout.toString().replace(/\n$/, ''); 58 | const shas = output.split('\n').reverse(); 59 | const contributors: Author[] = []; 60 | for (const sha of shas) { 61 | const author = commitAuthors[sha]; 62 | if (author && !contributors.find(contributor => contributor.login === author.login)) { 63 | contributors.push(author); 64 | } 65 | } 66 | file.contributors = contributors; 67 | } 68 | } 69 | 70 | find(categoryKey: string, algorithmKey: string) { 71 | const category = this.categories.find(category => category.key === categoryKey); 72 | if (!category) return; 73 | const algorithm = category.algorithms.find(algorithm => algorithm.key === algorithmKey); 74 | if (!algorithm) return; 75 | 76 | const categoryName = category.name; 77 | const algorithmName = algorithm.name; 78 | const files = algorithm.files; 79 | const description = algorithm.description; 80 | 81 | return {categoryKey, categoryName, algorithmKey, algorithmName, files, description}; 82 | } 83 | 84 | iterate(callback: (category: Category, algorithm: Algorithm) => void) { 85 | this.categories.forEach(category => category.algorithms.forEach(algorithm => callback(category, algorithm))); 86 | } 87 | 88 | toJSON() { 89 | const {categories} = this; 90 | return {categories}; 91 | } 92 | } 93 | -------------------------------------------------------------------------------- /src/models/index.ts: -------------------------------------------------------------------------------- 1 | export { Algorithm } from './Algorithm'; 2 | export { Category } from './Category'; 3 | export { File } from './File'; 4 | export { Hierarchy } from './Hierarchy'; 5 | -------------------------------------------------------------------------------- /src/tracers/DockerTracer.ts: -------------------------------------------------------------------------------- 1 | import path from 'path'; 2 | import { Release, Tracer } from 'tracers/Tracer'; 3 | import express from 'express'; 4 | import uuid from 'uuid'; 5 | import fs from 'fs-extra'; 6 | import { memoryLimit, timeLimit } from 'config/constants'; 7 | import { codesDir } from 'config/paths'; 8 | import { execute } from 'utils/misc'; 9 | 10 | export class DockerTracer extends Tracer { 11 | private readonly directory: string; 12 | private readonly imageName: string; 13 | 14 | constructor(lang: string) { 15 | super(lang); 16 | this.directory = path.resolve(__dirname, lang); 17 | this.imageName = `tracer-${this.lang}`; 18 | } 19 | 20 | build(release: Release) { 21 | const {tag_name} = release; 22 | return execute(`docker build -t ${this.imageName} . --build-arg tag_name=${tag_name}`, { 23 | cwd: this.directory, 24 | stdout: process.stdout, 25 | stderr: process.stderr, 26 | }); 27 | } 28 | 29 | route(router: express.Router) { 30 | router.post(`/${this.lang}`, (req, res, next) => { 31 | const {code} = req.body; 32 | const tempPath = path.resolve(codesDir, uuid.v4()); 33 | fs.outputFile(path.resolve(tempPath, `Main.${this.lang}`), code) 34 | .then(() => { 35 | const containerName = uuid.v4(); 36 | let killed = false; 37 | const timer = setTimeout(() => { 38 | execute(`docker kill ${containerName}`).then(() => { 39 | killed = true; 40 | }); 41 | }, timeLimit); 42 | return execute([ 43 | 'docker run --rm', 44 | `--name=${containerName}`, 45 | '-w=/usr/visualization', 46 | `-v=${tempPath}:/usr/visualization:rw`, 47 | `-m=${memoryLimit}m`, 48 | '-e ALGORITHM_VISUALIZER=1', 49 | this.imageName, 50 | ].join(' ')).catch(error => { 51 | if (killed) throw new Error('Time Limit Exceeded'); 52 | throw error; 53 | }).finally(() => clearTimeout(timer)); 54 | }) 55 | .then(() => new Promise((resolve, reject) => { 56 | const visualizationPath = path.resolve(tempPath, 'visualization.json'); 57 | res.sendFile(visualizationPath, (err: any) => { 58 | if (err) return reject(new Error('Visualization Not Found')); 59 | resolve(); 60 | }); 61 | })) 62 | .catch(next) 63 | .finally(() => fs.remove(tempPath)); 64 | }); 65 | } 66 | } 67 | -------------------------------------------------------------------------------- /src/tracers/LambdaTracer.ts: -------------------------------------------------------------------------------- 1 | import AWS from 'aws-sdk'; 2 | import express from 'express'; 3 | import { Release, Tracer } from 'tracers/Tracer'; 4 | import { awsAccessKeyId, awsSecretAccessKey } from 'config/environments'; 5 | import { BadRequest } from 'ts-httpexceptions'; 6 | 7 | export class LambdaTracer extends Tracer { 8 | static lambda = new AWS.Lambda({ 9 | region: 'us-east-2', 10 | accessKeyId: awsAccessKeyId, 11 | secretAccessKey: awsSecretAccessKey, 12 | }); 13 | 14 | async build(release: Release) { 15 | } 16 | 17 | route(router: express.Router) { 18 | router.post(`/${this.lang}`, (req, res, next) => { 19 | const {code} = req.body; 20 | LambdaTracer.lambda.invoke({ 21 | FunctionName: `extractor-${this.lang}`, 22 | InvocationType: 'RequestResponse', 23 | Payload: JSON.stringify(code), 24 | }, function (err, data) { 25 | if (err) return next(err); 26 | if (typeof data.Payload !== 'string') return next(new Error('Unexpected Payload Type')); 27 | const payload = JSON.parse(data.Payload); 28 | if (!payload.success) return next(new BadRequest(payload.errorMessage)); 29 | res.send(payload.commands); 30 | }); 31 | }); 32 | } 33 | } 34 | -------------------------------------------------------------------------------- /src/tracers/Tracer.ts: -------------------------------------------------------------------------------- 1 | import express from 'express'; 2 | import { GitHubApi } from 'utils/apis'; 3 | 4 | export type Release = { 5 | tag_name: string; 6 | } 7 | 8 | export abstract class Tracer { 9 | protected constructor(public lang: string) { 10 | this.update().catch(console.error); 11 | } 12 | 13 | abstract build(release: Release): Promise; 14 | 15 | abstract route(router: express.Router): void; 16 | 17 | async update(release?: Release) { 18 | if (release) { 19 | return this.build(release); 20 | } 21 | const {data} = await GitHubApi.getLatestRelease('algorithm-visualizer', `tracers.${this.lang}`); 22 | return this.build(data); 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /src/tracers/cpp/CppTracer.ts: -------------------------------------------------------------------------------- 1 | import { DockerTracer } from 'tracers/DockerTracer'; 2 | 3 | export class CppTracer extends DockerTracer { 4 | constructor() { 5 | super('cpp'); 6 | } 7 | } 8 | -------------------------------------------------------------------------------- /src/tracers/cpp/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM rikorose/gcc-cmake 2 | 3 | ARG tag_name 4 | 5 | RUN curl --create-dirs -o /usr/local/include/nlohmann/json.hpp -L "https://github.com/nlohmann/json/releases/download/v3.1.2/json.hpp" \ 6 | && curl --create-dirs -o /usr/tmp/algorithm-visualizer.tar.gz -L "https://github.com/algorithm-visualizer/tracers.cpp/archive/${tag_name}.tar.gz" \ 7 | && cd /usr/tmp \ 8 | && mkdir algorithm-visualizer \ 9 | && tar xvzf algorithm-visualizer.tar.gz -C algorithm-visualizer --strip-components=1 \ 10 | && cd /usr/tmp/algorithm-visualizer \ 11 | && mkdir build \ 12 | && cd build \ 13 | && cmake .. \ 14 | && make install 15 | 16 | CMD g++ Main.cpp -o Main -O2 -std=c++11 -lcurl \ 17 | && ./Main 18 | -------------------------------------------------------------------------------- /src/tracers/index.ts: -------------------------------------------------------------------------------- 1 | export { CppTracer } from './cpp/CppTracer'; 2 | export { JavaTracer } from './java/JavaTracer'; 3 | export { JsTracer } from './js/JsTracer'; 4 | -------------------------------------------------------------------------------- /src/tracers/java/JavaTracer.ts: -------------------------------------------------------------------------------- 1 | import { LambdaTracer } from 'tracers/LambdaTracer'; 2 | 3 | export class JavaTracer extends LambdaTracer { 4 | constructor() { 5 | super('java'); 6 | } 7 | } 8 | -------------------------------------------------------------------------------- /src/tracers/js/JsTracer.ts: -------------------------------------------------------------------------------- 1 | import path from 'path'; 2 | import { Release, Tracer } from 'tracers/Tracer'; 3 | import express from 'express'; 4 | 5 | export class JsTracer extends Tracer { 6 | readonly workerPath: string; 7 | tagName?: string; 8 | 9 | constructor() { 10 | super('js'); 11 | this.workerPath = path.resolve(__dirname, 'worker.js'); 12 | } 13 | 14 | async build(release: Release) { 15 | const {tag_name} = release; 16 | this.tagName = tag_name; 17 | } 18 | 19 | route(router: express.Router) { 20 | router.get(`/${this.lang}`, (req, res) => { 21 | if (!this.tagName) throw new Error('JsTracer has not been built yet.'); 22 | const version = this.tagName.slice(1); 23 | res.redirect(`https://unpkg.com/algorithm-visualizer@${version}/dist/algorithm-visualizer.umd.js`); 24 | }); 25 | router.get(`/${this.lang}/worker`, (req, res) => res.sendFile(this.workerPath)); 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /src/tracers/js/worker.js: -------------------------------------------------------------------------------- 1 | const process = { env: { ALGORITHM_VISUALIZER: '1' } }; 2 | importScripts('/api/tracers/js'); 3 | 4 | const sandbox = code => { 5 | const require = name => ({ 'algorithm-visualizer': AlgorithmVisualizer }[name]); // fake require 6 | eval(code); 7 | }; 8 | 9 | onmessage = e => { 10 | const lines = e.data.split('\n').map((line, i) => line.replace(/(\.\s*delay\s*)\(\s*\)/g, `$1(${i})`)); 11 | const code = lines.join('\n'); 12 | sandbox(code); 13 | postMessage(AlgorithmVisualizer.Commander.commands); 14 | }; 15 | -------------------------------------------------------------------------------- /src/utils/apis.ts: -------------------------------------------------------------------------------- 1 | import axios, { AxiosResponse } from 'axios'; 2 | import { githubClientId, githubClientSecret } from 'config/environments'; 3 | 4 | const instance = axios.create(); 5 | 6 | instance.interceptors.request.use(request => { 7 | request.params = {client_id: githubClientId, client_secret: githubClientSecret, ...request.params}; 8 | return request; 9 | }); 10 | 11 | const request = (url: string, process: (mappedUrl: string, args: any[]) => Promise>) => { 12 | const tokens = url.split('/'); 13 | const baseURL = /^https?:\/\//i.test(url) ? '' : 'https://api.github.com'; 14 | return (...args: any[]) => { 15 | const mappedUrl = baseURL + tokens.map(token => token.startsWith(':') ? args.shift() : token).join('/'); 16 | return process(mappedUrl, args); 17 | }; 18 | }; 19 | 20 | const GET = (url: string) => { 21 | return request(url, (mappedUrl: string, args: any[]) => { 22 | const [params] = args; 23 | return instance.get(mappedUrl, {params}); 24 | }); 25 | }; 26 | 27 | const DELETE = (url: string) => { 28 | return request(url, (mappedUrl: string, args: any[]) => { 29 | const [params] = args; 30 | return instance.delete(mappedUrl, {params}); 31 | }); 32 | }; 33 | 34 | const POST = (url: string) => { 35 | return request(url, (mappedUrl: string, args: any[]) => { 36 | const [body, params] = args; 37 | return instance.post(mappedUrl, body, {params}); 38 | }); 39 | }; 40 | 41 | const PUT = (url: string) => { 42 | return request(url, (mappedUrl: string, args: any[]) => { 43 | const [body, params] = args; 44 | return instance.put(mappedUrl, body, {params}); 45 | }); 46 | }; 47 | 48 | const PATCH = (url: string) => { 49 | return request(url, (mappedUrl: string, args: any[]) => { 50 | const [body, params] = args; 51 | return instance.patch(mappedUrl, body, {params}); 52 | }); 53 | }; 54 | 55 | export const GitHubApi = { 56 | listCommits: GET('/repos/:owner/:repo/commits'), 57 | 58 | getAccessToken: (code: string) => instance.post('https://github.com/login/oauth/access_token', {code}, {headers: {Accept: 'application/json'}}), 59 | 60 | getLatestRelease: GET('/repos/:owner/:repo/releases/latest'), 61 | }; 62 | -------------------------------------------------------------------------------- /src/utils/hierarchy.ts: -------------------------------------------------------------------------------- 1 | import path from 'path'; 2 | import fs from 'fs-extra'; 3 | 4 | export function createKey(name: string) { 5 | return name.toLowerCase().trim().replace(/[^\w \-]/g, '').replace(/ /g, '-'); 6 | } 7 | 8 | export function isDirectory(dirPath: string) { 9 | return fs.lstatSync(dirPath).isDirectory(); 10 | } 11 | 12 | export function listFiles(dirPath: string) { 13 | return fs.pathExistsSync(dirPath) ? fs.readdirSync(dirPath).filter(fileName => !fileName.startsWith('.')) : []; 14 | } 15 | 16 | export function listDirectories(dirPath: string) { 17 | return listFiles(dirPath).filter(fileName => isDirectory(path.resolve(dirPath, fileName))); 18 | } 19 | -------------------------------------------------------------------------------- /src/utils/misc.ts: -------------------------------------------------------------------------------- 1 | import axios from 'axios'; 2 | import fs from 'fs-extra'; 3 | import { File } from 'models'; 4 | import removeMarkdown from 'remove-markdown'; 5 | import * as child_process from 'child_process'; 6 | import { ExecOptions, spawn } from 'child_process'; 7 | import { rootDir } from '../config/paths'; 8 | import path from 'path'; 9 | 10 | export function download(url: string, localPath: string) { 11 | return axios({ url, method: 'GET', responseType: 'stream' }) 12 | .then(response => new Promise((resolve, reject) => { 13 | const writer = fs.createWriteStream(localPath); 14 | writer.on('finish', resolve); 15 | writer.on('error', reject); 16 | response.data.pipe(writer); 17 | })); 18 | } 19 | 20 | export async function pull(dir: string, repo: string, commit = 'origin/master') { 21 | if (fs.pathExistsSync(dir)) { 22 | await execute(`git fetch`, { 23 | cwd: dir, 24 | stdout: process.stdout, 25 | stderr: process.stderr, 26 | }); 27 | } else { 28 | await execute(`git clone https://github.com/algorithm-visualizer/${repo}.git ${dir}`, { 29 | stdout: process.stdout, 30 | stderr: process.stderr, 31 | }); 32 | } 33 | await execute(`git reset --hard ${commit}`, { 34 | cwd: dir, 35 | stdout: process.stdout, 36 | stderr: process.stderr, 37 | }); 38 | } 39 | 40 | export function getDescription(files: File[]) { 41 | const readmeFile = files.find(file => file.name === 'README.md'); 42 | if (!readmeFile) return ''; 43 | const lines = readmeFile.content.split('\n'); 44 | lines.shift(); 45 | while (lines.length && !lines[0].trim()) lines.shift(); 46 | const descriptionLines = []; 47 | while (lines.length && lines[0].trim()) descriptionLines.push(lines.shift()); 48 | return removeMarkdown(descriptionLines.join(' ')); 49 | } 50 | 51 | type ExecuteOptions = ExecOptions & { 52 | stdout?: NodeJS.WriteStream; 53 | stderr?: NodeJS.WriteStream; 54 | }; 55 | 56 | export function execute(command: string, { stdout, stderr, ...options }: ExecuteOptions = {}): Promise { 57 | return new Promise((resolve, reject) => { 58 | const child = child_process.exec(command, options, (error, stdout, stderr) => { 59 | if (error) return reject(error.code ? new Error(stderr) : error); 60 | resolve(stdout); 61 | }); 62 | if (child.stdout && stdout) child.stdout.pipe(stdout); 63 | if (child.stderr && stderr) child.stderr.pipe(stderr); 64 | }); 65 | } 66 | 67 | export function issueHttpsCertificate() { 68 | const certbotIniPath = path.resolve(rootDir, 'certbot.ini'); 69 | const childProcess = spawn('certbot', ['certonly', '--non-interactive', '--agree-tos', '--config', certbotIniPath]); 70 | childProcess.stdout.pipe(process.stdout); 71 | childProcess.stderr.pipe(process.stderr); 72 | childProcess.on('error', console.error); 73 | childProcess.on('exit', code => { 74 | if (code === 0) { 75 | process.exit(0); 76 | } else { 77 | console.error(new Error(`certbot failed with exit code ${code}.`)); 78 | } 79 | }); 80 | } 81 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "es2015", 4 | "lib": [ 5 | "es2015", 6 | "esnext" 7 | ], 8 | "module": "commonjs", 9 | "moduleResolution": "node", 10 | "esModuleInterop": true, 11 | "experimentalDecorators": true, 12 | "emitDecoratorMetadata": true, 13 | "sourceMap": true, 14 | "declaration": false, 15 | "strict": true, 16 | "baseUrl": "src", 17 | "resolveJsonModule": true 18 | }, 19 | "exclude": [ 20 | "node_modules", 21 | "public" 22 | ] 23 | } 24 | -------------------------------------------------------------------------------- /tslint.json: -------------------------------------------------------------------------------- 1 | { 2 | "rules": { 3 | "class-name": true, 4 | "comment-format": [ 5 | true, 6 | "check-space" 7 | ], 8 | "indent": [ 9 | true, 10 | "spaces", 11 | 2 12 | ], 13 | "one-line": [ 14 | true, 15 | "check-open-brace", 16 | "check-whitespace" 17 | ], 18 | "no-var-keyword": true, 19 | "quotemark": [ 20 | true, 21 | "single", 22 | "avoid-escape" 23 | ], 24 | "semicolon": [ 25 | true, 26 | "always", 27 | "ignore-bound-class-methods" 28 | ], 29 | "whitespace": [ 30 | true, 31 | "check-branch", 32 | "check-decl", 33 | "check-operator", 34 | "check-module", 35 | "check-separator", 36 | "check-type" 37 | ], 38 | "typedef-whitespace": [ 39 | true, 40 | { 41 | "call-signature": "nospace", 42 | "index-signature": "nospace", 43 | "parameter": "nospace", 44 | "property-declaration": "nospace", 45 | "variable-declaration": "nospace" 46 | }, 47 | { 48 | "call-signature": "onespace", 49 | "index-signature": "onespace", 50 | "parameter": "onespace", 51 | "property-declaration": "onespace", 52 | "variable-declaration": "onespace" 53 | } 54 | ], 55 | "no-internal-module": true, 56 | "no-trailing-whitespace": true, 57 | "no-null-keyword": true, 58 | "prefer-const": true, 59 | "jsdoc-format": true 60 | } 61 | } 62 | --------------------------------------------------------------------------------