├── .eslintrc ├── .gitignore ├── README.md ├── RoomService.js ├── credentials.js ├── jsconfig.json ├── package-lock.json ├── package.json ├── public ├── .eslintrc ├── css │ └── main.css ├── index.html └── js │ └── video.js └── server.js /.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "rules": { 3 | "no-unused-vars": "warn", 4 | "no-trailing-spaces": "warn", 5 | "semi": ["warn", "always"], 6 | "space-before-blocks": "warn", 7 | "no-console": ["off"], 8 | "brace-style": ["warn", "1tbs"], 9 | "camelcase": ["warn", {"properties": "never"}], 10 | "keyword-spacing": "warn", 11 | "space-infix-ops": "warn", 12 | "no-const-assign": "warn", 13 | "no-this-before-super": "warn", 14 | "no-undef": "warn", 15 | "no-unreachable": "warn", 16 | "constructor-super": "warn", 17 | "valid-typeof": "warn" 18 | }, 19 | "env": { 20 | "node": true, 21 | "es6": true 22 | }, 23 | "parserOptions": { 24 | "ecmaVersion": 8, 25 | "ecmaFeatures": {}, 26 | "sourceType": "module" 27 | }, 28 | "extends": "eslint:recommended" 29 | } 30 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules 3 | .vscode 4 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Video Chat 2 | 3 | [Slides](http://slides.com/basscord/webrtc-video-streaming/) | [Demo](https://video-chat.basscord.co/your-clever-room-name) 4 | 5 | Simple app that allows peer to peer video chats. 6 | 7 | Create a room by going to [https://video-chat.basscord.co/](https://video-chat.basscord.co/)[room-name-goes-here] 8 | 9 | Send that url to a friend and video chatting will commence. 10 | -------------------------------------------------------------------------------- /RoomService.js: -------------------------------------------------------------------------------- 1 | /** @type {SocketIO.Server} */ 2 | let _io; 3 | const MAX_CLIENTS = 3; 4 | 5 | /** @param {SocketIO.Socket} socket */ 6 | function listen(socket) { 7 | const io = _io; 8 | const rooms = io.nsps['/'].adapter.rooms; 9 | 10 | socket.on('join', function(room) { 11 | 12 | let numClients = 0; 13 | if (rooms[room]) { 14 | numClients = rooms[room].length; 15 | } 16 | if (numClients < MAX_CLIENTS) { 17 | socket.on('ready', function() { 18 | socket.broadcast.to(room).emit('ready', socket.id); 19 | }); 20 | socket.on('offer', function (id, message) { 21 | socket.to(id).emit('offer', socket.id, message); 22 | }); 23 | socket.on('answer', function (id, message) { 24 | socket.to(id).emit('answer', socket.id, message); 25 | }); 26 | socket.on('candidate', function (id, message) { 27 | socket.to(id).emit('candidate', socket.id, message); 28 | }); 29 | socket.on('disconnect', function() { 30 | socket.broadcast.to(room).emit('bye', socket.id); 31 | }); 32 | socket.join(room); 33 | } else { 34 | socket.emit('full', room); 35 | } 36 | }); 37 | } 38 | 39 | /** @param {SocketIO.Server} io */ 40 | module.exports = function(io) { 41 | _io = io; 42 | return {listen}; 43 | }; -------------------------------------------------------------------------------- /credentials.js: -------------------------------------------------------------------------------- 1 | // const fs = require('fs'); 2 | module.exports = { 3 | key: null, 4 | cert: null 5 | // key: fs.readFileSync('privkey.pem', 'utf8'), 6 | // cert: fs.readFileSync('fullchain.pem', 'utf8') 7 | }; -------------------------------------------------------------------------------- /jsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "ES6", 4 | "module": "commonjs", 5 | "checkJs": true, 6 | "diagnostics": true 7 | }, 8 | "exclude": [ 9 | "node_modules" 10 | ] 11 | } -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "video-chat", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@types/node": { 8 | "version": "8.0.19", 9 | "resolved": "https://registry.npmjs.org/@types/node/-/node-8.0.19.tgz", 10 | "integrity": "sha512-VRQB+Q0L3YZWs45uRdpN9oWr82meL/8TrJ6faoKT5tp0uub2l/aRMhtm5fo68h7kjYKH60f9/bay1nF7ZpTW5g==", 11 | "dev": true 12 | }, 13 | "@types/socket.io-client": { 14 | "version": "1.4.29", 15 | "resolved": "https://registry.npmjs.org/@types/socket.io-client/-/socket.io-client-1.4.29.tgz", 16 | "integrity": "sha1-+HQwcM7pMXXjbgtqd6ivc+WMyzI=", 17 | "dev": true 18 | }, 19 | "@types/webrtc": { 20 | "version": "0.0.21", 21 | "resolved": "https://registry.npmjs.org/@types/webrtc/-/webrtc-0.0.21.tgz", 22 | "integrity": "sha1-vDz7fiKt0oR2Hl1j761mSw66i4Y=", 23 | "dev": true 24 | }, 25 | "accepts": { 26 | "version": "1.3.3", 27 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.3.tgz", 28 | "integrity": "sha1-w8p0NJOGSMPg2cHjKN1otiLChMo=", 29 | "requires": { 30 | "mime-types": "2.1.16", 31 | "negotiator": "0.6.1" 32 | } 33 | }, 34 | "acorn": { 35 | "version": "5.1.1", 36 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-5.1.1.tgz", 37 | "integrity": "sha512-vOk6uEMctu0vQrvuSqFdJyqj1Q0S5VTDL79qtjo+DhRr+1mmaD+tluFSCZqhvi/JUhXSzoZN2BhtstaPEeE8cw==", 38 | "dev": true 39 | }, 40 | "acorn-jsx": { 41 | "version": "3.0.1", 42 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-3.0.1.tgz", 43 | "integrity": "sha1-r9+UiPsezvyDSPb7IvRk4ypYs2s=", 44 | "dev": true, 45 | "requires": { 46 | "acorn": "3.3.0" 47 | }, 48 | "dependencies": { 49 | "acorn": { 50 | "version": "3.3.0", 51 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-3.3.0.tgz", 52 | "integrity": "sha1-ReN/s56No/JbruP/U2niu18iAXo=", 53 | "dev": true 54 | } 55 | } 56 | }, 57 | "after": { 58 | "version": "0.8.1", 59 | "resolved": "https://registry.npmjs.org/after/-/after-0.8.1.tgz", 60 | "integrity": "sha1-q11PuIP1loFtNRX495HAr0ht1ic=" 61 | }, 62 | "ajv": { 63 | "version": "5.2.2", 64 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-5.2.2.tgz", 65 | "integrity": "sha1-R8aNaehvXZUxA7AHSpQw3GPaXjk=", 66 | "dev": true, 67 | "requires": { 68 | "co": "4.6.0", 69 | "fast-deep-equal": "1.0.0", 70 | "json-schema-traverse": "0.3.1", 71 | "json-stable-stringify": "1.0.1" 72 | } 73 | }, 74 | "ajv-keywords": { 75 | "version": "1.5.1", 76 | "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-1.5.1.tgz", 77 | "integrity": "sha1-MU3QpLM2j609/NxU7eYXG4htrzw=", 78 | "dev": true 79 | }, 80 | "ansi-escapes": { 81 | "version": "2.0.0", 82 | "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-2.0.0.tgz", 83 | "integrity": "sha1-W65SvkJIeN2Xg+iRDj/Cki6DyBs=", 84 | "dev": true 85 | }, 86 | "ansi-regex": { 87 | "version": "2.1.1", 88 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", 89 | "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", 90 | "dev": true 91 | }, 92 | "ansi-styles": { 93 | "version": "2.2.1", 94 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", 95 | "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", 96 | "dev": true 97 | }, 98 | "argparse": { 99 | "version": "1.0.9", 100 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.9.tgz", 101 | "integrity": "sha1-c9g7wmP4bpf4zE9rrhsOkKfSLIY=", 102 | "dev": true, 103 | "requires": { 104 | "sprintf-js": "1.0.3" 105 | } 106 | }, 107 | "array-flatten": { 108 | "version": "1.1.1", 109 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 110 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 111 | }, 112 | "array-union": { 113 | "version": "1.0.2", 114 | "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", 115 | "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", 116 | "dev": true, 117 | "requires": { 118 | "array-uniq": "1.0.3" 119 | } 120 | }, 121 | "array-uniq": { 122 | "version": "1.0.3", 123 | "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", 124 | "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=", 125 | "dev": true 126 | }, 127 | "arraybuffer.slice": { 128 | "version": "0.0.6", 129 | "resolved": "https://registry.npmjs.org/arraybuffer.slice/-/arraybuffer.slice-0.0.6.tgz", 130 | "integrity": "sha1-8zshWfBTKj8xB6JywMz70a0peco=" 131 | }, 132 | "arrify": { 133 | "version": "1.0.1", 134 | "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", 135 | "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", 136 | "dev": true 137 | }, 138 | "babel-code-frame": { 139 | "version": "6.22.0", 140 | "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.22.0.tgz", 141 | "integrity": "sha1-AnYgvuVnqIwyVhV05/0IAdMxGOQ=", 142 | "dev": true, 143 | "requires": { 144 | "chalk": "1.1.3", 145 | "esutils": "2.0.2", 146 | "js-tokens": "3.0.2" 147 | } 148 | }, 149 | "backo2": { 150 | "version": "1.0.2", 151 | "resolved": "https://registry.npmjs.org/backo2/-/backo2-1.0.2.tgz", 152 | "integrity": "sha1-MasayLEpNjRj41s+u2n038+6eUc=" 153 | }, 154 | "balanced-match": { 155 | "version": "1.0.0", 156 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 157 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 158 | "dev": true 159 | }, 160 | "base64-arraybuffer": { 161 | "version": "0.1.2", 162 | "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-0.1.2.tgz", 163 | "integrity": "sha1-R030qfLaJOBd8xWMOx2zw81GoVQ=" 164 | }, 165 | "base64id": { 166 | "version": "0.1.0", 167 | "resolved": "https://registry.npmjs.org/base64id/-/base64id-0.1.0.tgz", 168 | "integrity": "sha1-As4P3u4M709ACA4ec+g08LG/zj8=" 169 | }, 170 | "benchmark": { 171 | "version": "1.0.0", 172 | "resolved": "https://registry.npmjs.org/benchmark/-/benchmark-1.0.0.tgz", 173 | "integrity": "sha1-Lx4vpMNZ8REiqhgwgiGOlX45DHM=" 174 | }, 175 | "better-assert": { 176 | "version": "1.0.2", 177 | "resolved": "https://registry.npmjs.org/better-assert/-/better-assert-1.0.2.tgz", 178 | "integrity": "sha1-QIZrnhueC1W0gYlDEeaPr/rrxSI=", 179 | "requires": { 180 | "callsite": "1.0.0" 181 | } 182 | }, 183 | "blob": { 184 | "version": "0.0.4", 185 | "resolved": "https://registry.npmjs.org/blob/-/blob-0.0.4.tgz", 186 | "integrity": "sha1-vPEwUspURj8w+fx+lbmkdjCpSSE=" 187 | }, 188 | "brace-expansion": { 189 | "version": "1.1.8", 190 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.8.tgz", 191 | "integrity": "sha1-wHshHHyVLsH479Uad+8NHTmQopI=", 192 | "dev": true, 193 | "requires": { 194 | "balanced-match": "1.0.0", 195 | "concat-map": "0.0.1" 196 | } 197 | }, 198 | "caller-path": { 199 | "version": "0.1.0", 200 | "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-0.1.0.tgz", 201 | "integrity": "sha1-lAhe9jWB7NPaqSREqP6U6CV3dR8=", 202 | "dev": true, 203 | "requires": { 204 | "callsites": "0.2.0" 205 | } 206 | }, 207 | "callsite": { 208 | "version": "1.0.0", 209 | "resolved": "https://registry.npmjs.org/callsite/-/callsite-1.0.0.tgz", 210 | "integrity": "sha1-KAOY5dZkvXQDi28JBRU+borxvCA=" 211 | }, 212 | "callsites": { 213 | "version": "0.2.0", 214 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-0.2.0.tgz", 215 | "integrity": "sha1-r6uWJikQp/M8GaV3WCXGnzTjUMo=", 216 | "dev": true 217 | }, 218 | "chalk": { 219 | "version": "1.1.3", 220 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", 221 | "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", 222 | "dev": true, 223 | "requires": { 224 | "ansi-styles": "2.2.1", 225 | "escape-string-regexp": "1.0.5", 226 | "has-ansi": "2.0.0", 227 | "strip-ansi": "3.0.1", 228 | "supports-color": "2.0.0" 229 | } 230 | }, 231 | "circular-json": { 232 | "version": "0.3.3", 233 | "resolved": "https://registry.npmjs.org/circular-json/-/circular-json-0.3.3.tgz", 234 | "integrity": "sha512-UZK3NBx2Mca+b5LsG7bY183pHWt5Y1xts4P3Pz7ENTwGVnJOUWbRb3ocjvX7hx9tq/yTAdclXm9sZ38gNuem4A==", 235 | "dev": true 236 | }, 237 | "cli-cursor": { 238 | "version": "2.1.0", 239 | "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", 240 | "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", 241 | "dev": true, 242 | "requires": { 243 | "restore-cursor": "2.0.0" 244 | } 245 | }, 246 | "cli-width": { 247 | "version": "2.1.0", 248 | "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.1.0.tgz", 249 | "integrity": "sha1-sjTKIJsp72b8UY2bmNWEewDt8Ao=", 250 | "dev": true 251 | }, 252 | "co": { 253 | "version": "4.6.0", 254 | "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", 255 | "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=", 256 | "dev": true 257 | }, 258 | "color-convert": { 259 | "version": "1.9.0", 260 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.0.tgz", 261 | "integrity": "sha1-Gsz5fdc5uYO/mU1W/sj5WFNkG3o=", 262 | "dev": true, 263 | "requires": { 264 | "color-name": "1.1.3" 265 | } 266 | }, 267 | "color-name": { 268 | "version": "1.1.3", 269 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 270 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 271 | "dev": true 272 | }, 273 | "component-bind": { 274 | "version": "1.0.0", 275 | "resolved": "https://registry.npmjs.org/component-bind/-/component-bind-1.0.0.tgz", 276 | "integrity": "sha1-AMYIq33Nk4l8AAllGx06jh5zu9E=" 277 | }, 278 | "component-emitter": { 279 | "version": "1.1.2", 280 | "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.1.2.tgz", 281 | "integrity": "sha1-KWWU8nU9qmOZbSrwjRWpURbJrsM=" 282 | }, 283 | "component-inherit": { 284 | "version": "0.0.3", 285 | "resolved": "https://registry.npmjs.org/component-inherit/-/component-inherit-0.0.3.tgz", 286 | "integrity": "sha1-ZF/ErfWLcrZJ1crmUTVhnbJv8UM=" 287 | }, 288 | "concat-map": { 289 | "version": "0.0.1", 290 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 291 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 292 | "dev": true 293 | }, 294 | "concat-stream": { 295 | "version": "1.6.0", 296 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.0.tgz", 297 | "integrity": "sha1-CqxmL9Ur54lk1VMvaUeE5wEQrPc=", 298 | "dev": true, 299 | "requires": { 300 | "inherits": "2.0.3", 301 | "readable-stream": "2.3.3", 302 | "typedarray": "0.0.6" 303 | } 304 | }, 305 | "content-disposition": { 306 | "version": "0.5.2", 307 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", 308 | "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ=" 309 | }, 310 | "content-type": { 311 | "version": "1.0.2", 312 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.2.tgz", 313 | "integrity": "sha1-t9ETrueo3Se9IRM8TcJSnfFyHu0=" 314 | }, 315 | "cookie": { 316 | "version": "0.3.1", 317 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", 318 | "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" 319 | }, 320 | "cookie-signature": { 321 | "version": "1.0.6", 322 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 323 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 324 | }, 325 | "core-util-is": { 326 | "version": "1.0.2", 327 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 328 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", 329 | "dev": true 330 | }, 331 | "cross-spawn": { 332 | "version": "5.1.0", 333 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz", 334 | "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=", 335 | "dev": true, 336 | "requires": { 337 | "lru-cache": "4.1.1", 338 | "shebang-command": "1.2.0", 339 | "which": "1.3.0" 340 | } 341 | }, 342 | "deep-is": { 343 | "version": "0.1.3", 344 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", 345 | "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", 346 | "dev": true 347 | }, 348 | "del": { 349 | "version": "2.2.2", 350 | "resolved": "https://registry.npmjs.org/del/-/del-2.2.2.tgz", 351 | "integrity": "sha1-wSyYHQZ4RshLyvhiz/kw2Qf/0ag=", 352 | "dev": true, 353 | "requires": { 354 | "globby": "5.0.0", 355 | "is-path-cwd": "1.0.0", 356 | "is-path-in-cwd": "1.0.0", 357 | "object-assign": "4.1.1", 358 | "pify": "2.3.0", 359 | "pinkie-promise": "2.0.1", 360 | "rimraf": "2.6.1" 361 | } 362 | }, 363 | "depd": { 364 | "version": "1.1.1", 365 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.1.tgz", 366 | "integrity": "sha1-V4O04cRZ8G+lyif5kfPQbnoxA1k=" 367 | }, 368 | "destroy": { 369 | "version": "1.0.4", 370 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 371 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 372 | }, 373 | "doctrine": { 374 | "version": "2.0.0", 375 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.0.0.tgz", 376 | "integrity": "sha1-xz2NKQnSIpHhoAejlYBNqLZl/mM=", 377 | "dev": true, 378 | "requires": { 379 | "esutils": "2.0.2", 380 | "isarray": "1.0.0" 381 | }, 382 | "dependencies": { 383 | "isarray": { 384 | "version": "1.0.0", 385 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 386 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", 387 | "dev": true 388 | } 389 | } 390 | }, 391 | "ee-first": { 392 | "version": "1.1.1", 393 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 394 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 395 | }, 396 | "encodeurl": { 397 | "version": "1.0.1", 398 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.1.tgz", 399 | "integrity": "sha1-eePVhlU0aQn+bw9Fpd5oEDspTSA=" 400 | }, 401 | "engine.io": { 402 | "version": "1.6.11", 403 | "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-1.6.11.tgz", 404 | "integrity": "sha1-JTOpemWHbED/z5U5e375tJXEI/4=", 405 | "requires": { 406 | "accepts": "1.1.4", 407 | "base64id": "0.1.0", 408 | "debug": "2.2.0", 409 | "engine.io-parser": "1.2.4", 410 | "ws": "1.1.0" 411 | }, 412 | "dependencies": { 413 | "accepts": { 414 | "version": "1.1.4", 415 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.1.4.tgz", 416 | "integrity": "sha1-1xyW99QdD+2iw4zRToonwEFY30o=", 417 | "requires": { 418 | "mime-types": "2.0.14", 419 | "negotiator": "0.4.9" 420 | } 421 | }, 422 | "debug": { 423 | "version": "2.2.0", 424 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.2.0.tgz", 425 | "integrity": "sha1-+HBX6ZWxofauaklgZkE3vFbwOdo=", 426 | "requires": { 427 | "ms": "0.7.1" 428 | } 429 | }, 430 | "mime-db": { 431 | "version": "1.12.0", 432 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.12.0.tgz", 433 | "integrity": "sha1-PQxjGA9FjrENMlqqN9fFiuMS6dc=" 434 | }, 435 | "mime-types": { 436 | "version": "2.0.14", 437 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.0.14.tgz", 438 | "integrity": "sha1-MQ4VnbI+B3+Lsit0jav6SVcUCqY=", 439 | "requires": { 440 | "mime-db": "1.12.0" 441 | } 442 | }, 443 | "ms": { 444 | "version": "0.7.1", 445 | "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.1.tgz", 446 | "integrity": "sha1-nNE8A62/8ltl7/3nzoZO6VIBcJg=" 447 | }, 448 | "negotiator": { 449 | "version": "0.4.9", 450 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.4.9.tgz", 451 | "integrity": "sha1-kuRrbbU8fkIe1koryU8IvnYw3z8=" 452 | } 453 | } 454 | }, 455 | "engine.io-client": { 456 | "version": "1.6.11", 457 | "resolved": "https://registry.npmjs.org/engine.io-client/-/engine.io-client-1.6.11.tgz", 458 | "integrity": "sha1-fSUNj6HCGBGezeUTkEWKV9UXE3Y=", 459 | "requires": { 460 | "component-emitter": "1.1.2", 461 | "component-inherit": "0.0.3", 462 | "debug": "2.2.0", 463 | "engine.io-parser": "1.2.4", 464 | "has-cors": "1.1.0", 465 | "indexof": "0.0.1", 466 | "parsejson": "0.0.1", 467 | "parseqs": "0.0.2", 468 | "parseuri": "0.0.4", 469 | "ws": "1.0.1", 470 | "xmlhttprequest-ssl": "1.5.1", 471 | "yeast": "0.1.2" 472 | }, 473 | "dependencies": { 474 | "debug": { 475 | "version": "2.2.0", 476 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.2.0.tgz", 477 | "integrity": "sha1-+HBX6ZWxofauaklgZkE3vFbwOdo=", 478 | "requires": { 479 | "ms": "0.7.1" 480 | } 481 | }, 482 | "ms": { 483 | "version": "0.7.1", 484 | "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.1.tgz", 485 | "integrity": "sha1-nNE8A62/8ltl7/3nzoZO6VIBcJg=" 486 | }, 487 | "ws": { 488 | "version": "1.0.1", 489 | "resolved": "https://registry.npmjs.org/ws/-/ws-1.0.1.tgz", 490 | "integrity": "sha1-fQsqLljN3YGQOcKcneZQReGzEOk=", 491 | "requires": { 492 | "options": "0.0.6", 493 | "ultron": "1.0.2" 494 | } 495 | } 496 | } 497 | }, 498 | "engine.io-parser": { 499 | "version": "1.2.4", 500 | "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-1.2.4.tgz", 501 | "integrity": "sha1-4Il7C/FOeS1M0qWVBVORnFaUjEI=", 502 | "requires": { 503 | "after": "0.8.1", 504 | "arraybuffer.slice": "0.0.6", 505 | "base64-arraybuffer": "0.1.2", 506 | "blob": "0.0.4", 507 | "has-binary": "0.1.6", 508 | "utf8": "2.1.0" 509 | }, 510 | "dependencies": { 511 | "has-binary": { 512 | "version": "0.1.6", 513 | "resolved": "https://registry.npmjs.org/has-binary/-/has-binary-0.1.6.tgz", 514 | "integrity": "sha1-JTJvOc+k9hath4eJTjryz7x7bhA=", 515 | "requires": { 516 | "isarray": "0.0.1" 517 | } 518 | } 519 | } 520 | }, 521 | "escape-html": { 522 | "version": "1.0.3", 523 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 524 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 525 | }, 526 | "escape-string-regexp": { 527 | "version": "1.0.5", 528 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 529 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 530 | "dev": true 531 | }, 532 | "eslint": { 533 | "version": "4.4.1", 534 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-4.4.1.tgz", 535 | "integrity": "sha1-mc1+r8/8ov+Zpcj18qR01jZLS9M=", 536 | "dev": true, 537 | "requires": { 538 | "ajv": "5.2.2", 539 | "babel-code-frame": "6.22.0", 540 | "chalk": "1.1.3", 541 | "concat-stream": "1.6.0", 542 | "cross-spawn": "5.1.0", 543 | "debug": "2.6.8", 544 | "doctrine": "2.0.0", 545 | "eslint-scope": "3.7.1", 546 | "espree": "3.5.0", 547 | "esquery": "1.0.0", 548 | "estraverse": "4.2.0", 549 | "esutils": "2.0.2", 550 | "file-entry-cache": "2.0.0", 551 | "functional-red-black-tree": "1.0.1", 552 | "glob": "7.1.2", 553 | "globals": "9.18.0", 554 | "ignore": "3.3.3", 555 | "imurmurhash": "0.1.4", 556 | "inquirer": "3.2.1", 557 | "is-resolvable": "1.0.0", 558 | "js-yaml": "3.9.1", 559 | "json-stable-stringify": "1.0.1", 560 | "levn": "0.3.0", 561 | "lodash": "4.17.4", 562 | "minimatch": "3.0.4", 563 | "mkdirp": "0.5.1", 564 | "natural-compare": "1.4.0", 565 | "optionator": "0.8.2", 566 | "path-is-inside": "1.0.2", 567 | "pluralize": "4.0.0", 568 | "progress": "2.0.0", 569 | "require-uncached": "1.0.3", 570 | "semver": "5.4.1", 571 | "strip-json-comments": "2.0.1", 572 | "table": "4.0.1", 573 | "text-table": "0.2.0" 574 | }, 575 | "dependencies": { 576 | "debug": { 577 | "version": "2.6.8", 578 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.8.tgz", 579 | "integrity": "sha1-5zFTHKLt4n0YgiJCfaF4IdaP9Pw=", 580 | "dev": true, 581 | "requires": { 582 | "ms": "2.0.0" 583 | } 584 | } 585 | } 586 | }, 587 | "eslint-scope": { 588 | "version": "3.7.1", 589 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-3.7.1.tgz", 590 | "integrity": "sha1-PWPD7f2gLgbgGkUq2IyqzHzctug=", 591 | "dev": true, 592 | "requires": { 593 | "esrecurse": "4.2.0", 594 | "estraverse": "4.2.0" 595 | } 596 | }, 597 | "espree": { 598 | "version": "3.5.0", 599 | "resolved": "https://registry.npmjs.org/espree/-/espree-3.5.0.tgz", 600 | "integrity": "sha1-mDWGJb3QVYYeon4oZ+pyn69GPY0=", 601 | "dev": true, 602 | "requires": { 603 | "acorn": "5.1.1", 604 | "acorn-jsx": "3.0.1" 605 | } 606 | }, 607 | "esprima": { 608 | "version": "4.0.0", 609 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.0.tgz", 610 | "integrity": "sha512-oftTcaMu/EGrEIu904mWteKIv8vMuOgGYo7EhVJJN00R/EED9DCua/xxHRdYnKtcECzVg7xOWhflvJMnqcFZjw==", 611 | "dev": true 612 | }, 613 | "esquery": { 614 | "version": "1.0.0", 615 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.0.0.tgz", 616 | "integrity": "sha1-z7qLV9f7qT8XKYqKAGoEzaE9gPo=", 617 | "dev": true, 618 | "requires": { 619 | "estraverse": "4.2.0" 620 | } 621 | }, 622 | "esrecurse": { 623 | "version": "4.2.0", 624 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.2.0.tgz", 625 | "integrity": "sha1-+pVo2Y04I/mkHZHpAtyrnqblsWM=", 626 | "dev": true, 627 | "requires": { 628 | "estraverse": "4.2.0", 629 | "object-assign": "4.1.1" 630 | } 631 | }, 632 | "estraverse": { 633 | "version": "4.2.0", 634 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.2.0.tgz", 635 | "integrity": "sha1-De4/7TH81GlhjOc0IJn8GvoL2xM=", 636 | "dev": true 637 | }, 638 | "esutils": { 639 | "version": "2.0.2", 640 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz", 641 | "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=", 642 | "dev": true 643 | }, 644 | "etag": { 645 | "version": "1.8.0", 646 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.0.tgz", 647 | "integrity": "sha1-b2Ma7zNtbEY2K1F2QETOIWvjwFE=" 648 | }, 649 | "express": { 650 | "version": "4.15.4", 651 | "resolved": "https://registry.npmjs.org/express/-/express-4.15.4.tgz", 652 | "integrity": "sha1-Ay4iU0ic+PzgJma+yj0R7XotrtE=", 653 | "requires": { 654 | "accepts": "1.3.3", 655 | "array-flatten": "1.1.1", 656 | "content-disposition": "0.5.2", 657 | "content-type": "1.0.2", 658 | "cookie": "0.3.1", 659 | "cookie-signature": "1.0.6", 660 | "debug": "2.6.8", 661 | "depd": "1.1.1", 662 | "encodeurl": "1.0.1", 663 | "escape-html": "1.0.3", 664 | "etag": "1.8.0", 665 | "finalhandler": "1.0.4", 666 | "fresh": "0.5.0", 667 | "merge-descriptors": "1.0.1", 668 | "methods": "1.1.2", 669 | "on-finished": "2.3.0", 670 | "parseurl": "1.3.1", 671 | "path-to-regexp": "0.1.7", 672 | "proxy-addr": "1.1.5", 673 | "qs": "6.5.0", 674 | "range-parser": "1.2.0", 675 | "send": "0.15.4", 676 | "serve-static": "1.12.4", 677 | "setprototypeof": "1.0.3", 678 | "statuses": "1.3.1", 679 | "type-is": "1.6.15", 680 | "utils-merge": "1.0.0", 681 | "vary": "1.1.1" 682 | }, 683 | "dependencies": { 684 | "debug": { 685 | "version": "2.6.8", 686 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.8.tgz", 687 | "integrity": "sha1-5zFTHKLt4n0YgiJCfaF4IdaP9Pw=", 688 | "requires": { 689 | "ms": "2.0.0" 690 | } 691 | }, 692 | "qs": { 693 | "version": "6.5.0", 694 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.0.tgz", 695 | "integrity": "sha512-fjVFjW9yhqMhVGwRExCXLhJKrLlkYSaxNWdyc9rmHlrVZbk35YHH312dFd7191uQeXkI3mKLZTIbSvIeFwFemg==" 696 | } 697 | } 698 | }, 699 | "external-editor": { 700 | "version": "2.0.4", 701 | "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-2.0.4.tgz", 702 | "integrity": "sha1-HtkZnanL/i7y96MbL96LDRI2iXI=", 703 | "dev": true, 704 | "requires": { 705 | "iconv-lite": "0.4.18", 706 | "jschardet": "1.5.1", 707 | "tmp": "0.0.31" 708 | }, 709 | "dependencies": { 710 | "iconv-lite": { 711 | "version": "0.4.18", 712 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.18.tgz", 713 | "integrity": "sha512-sr1ZQph3UwHTR0XftSbK85OvBbxe/abLGzEnPENCQwmHf7sck8Oyu4ob3LgBxWWxRoM+QszeUyl7jbqapu2TqA==", 714 | "dev": true 715 | } 716 | } 717 | }, 718 | "fast-deep-equal": { 719 | "version": "1.0.0", 720 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.0.0.tgz", 721 | "integrity": "sha1-liVqO8l1WV6zbYLpkp0GDYk0Of8=", 722 | "dev": true 723 | }, 724 | "fast-levenshtein": { 725 | "version": "2.0.6", 726 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", 727 | "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", 728 | "dev": true 729 | }, 730 | "figures": { 731 | "version": "2.0.0", 732 | "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", 733 | "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", 734 | "dev": true, 735 | "requires": { 736 | "escape-string-regexp": "1.0.5" 737 | } 738 | }, 739 | "file-entry-cache": { 740 | "version": "2.0.0", 741 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-2.0.0.tgz", 742 | "integrity": "sha1-w5KZDD5oR4PYOLjISkXYoEhFg2E=", 743 | "dev": true, 744 | "requires": { 745 | "flat-cache": "1.2.2", 746 | "object-assign": "4.1.1" 747 | } 748 | }, 749 | "finalhandler": { 750 | "version": "1.0.4", 751 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.0.4.tgz", 752 | "integrity": "sha512-16l/r8RgzlXKmFOhZpHBztvye+lAhC5SU7hXavnerC9UfZqZxxXl3BzL8MhffPT3kF61lj9Oav2LKEzh0ei7tg==", 753 | "requires": { 754 | "debug": "2.6.8", 755 | "encodeurl": "1.0.1", 756 | "escape-html": "1.0.3", 757 | "on-finished": "2.3.0", 758 | "parseurl": "1.3.1", 759 | "statuses": "1.3.1", 760 | "unpipe": "1.0.0" 761 | }, 762 | "dependencies": { 763 | "debug": { 764 | "version": "2.6.8", 765 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.8.tgz", 766 | "integrity": "sha1-5zFTHKLt4n0YgiJCfaF4IdaP9Pw=", 767 | "requires": { 768 | "ms": "2.0.0" 769 | } 770 | } 771 | } 772 | }, 773 | "flat-cache": { 774 | "version": "1.2.2", 775 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-1.2.2.tgz", 776 | "integrity": "sha1-+oZxTnLCHbiGAXYezy9VXRq8a5Y=", 777 | "dev": true, 778 | "requires": { 779 | "circular-json": "0.3.3", 780 | "del": "2.2.2", 781 | "graceful-fs": "4.1.11", 782 | "write": "0.2.1" 783 | } 784 | }, 785 | "forwarded": { 786 | "version": "0.1.0", 787 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.0.tgz", 788 | "integrity": "sha1-Ge+YdMSuHCl7zweP3mOgm2aoQ2M=" 789 | }, 790 | "fresh": { 791 | "version": "0.5.0", 792 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.0.tgz", 793 | "integrity": "sha1-9HTKXmqSRtb9jglTz6m5yAWvp44=" 794 | }, 795 | "fs.realpath": { 796 | "version": "1.0.0", 797 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 798 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", 799 | "dev": true 800 | }, 801 | "functional-red-black-tree": { 802 | "version": "1.0.1", 803 | "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", 804 | "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", 805 | "dev": true 806 | }, 807 | "glob": { 808 | "version": "7.1.2", 809 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz", 810 | "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==", 811 | "dev": true, 812 | "requires": { 813 | "fs.realpath": "1.0.0", 814 | "inflight": "1.0.6", 815 | "inherits": "2.0.3", 816 | "minimatch": "3.0.4", 817 | "once": "1.4.0", 818 | "path-is-absolute": "1.0.1" 819 | } 820 | }, 821 | "globals": { 822 | "version": "9.18.0", 823 | "resolved": "https://registry.npmjs.org/globals/-/globals-9.18.0.tgz", 824 | "integrity": "sha512-S0nG3CLEQiY/ILxqtztTWH/3iRRdyBLw6KMDxnKMchrtbj2OFmehVh0WUCfW3DUrIgx/qFrJPICrq4Z4sTR9UQ==", 825 | "dev": true 826 | }, 827 | "globby": { 828 | "version": "5.0.0", 829 | "resolved": "https://registry.npmjs.org/globby/-/globby-5.0.0.tgz", 830 | "integrity": "sha1-69hGZ8oNuzMLmbz8aOrCvFQ3Dg0=", 831 | "dev": true, 832 | "requires": { 833 | "array-union": "1.0.2", 834 | "arrify": "1.0.1", 835 | "glob": "7.1.2", 836 | "object-assign": "4.1.1", 837 | "pify": "2.3.0", 838 | "pinkie-promise": "2.0.1" 839 | } 840 | }, 841 | "graceful-fs": { 842 | "version": "4.1.11", 843 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz", 844 | "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=", 845 | "dev": true 846 | }, 847 | "has-ansi": { 848 | "version": "2.0.0", 849 | "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", 850 | "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", 851 | "dev": true, 852 | "requires": { 853 | "ansi-regex": "2.1.1" 854 | } 855 | }, 856 | "has-binary": { 857 | "version": "0.1.7", 858 | "resolved": "https://registry.npmjs.org/has-binary/-/has-binary-0.1.7.tgz", 859 | "integrity": "sha1-aOYesWIQyVRaClzOBqhzkS/h5ow=", 860 | "requires": { 861 | "isarray": "0.0.1" 862 | } 863 | }, 864 | "has-cors": { 865 | "version": "1.1.0", 866 | "resolved": "https://registry.npmjs.org/has-cors/-/has-cors-1.1.0.tgz", 867 | "integrity": "sha1-XkdHk/fqmEPRu5nCPu9J/xJv/zk=" 868 | }, 869 | "has-flag": { 870 | "version": "2.0.0", 871 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-2.0.0.tgz", 872 | "integrity": "sha1-6CB68cx7MNRGzHC3NLXovhj4jVE=", 873 | "dev": true 874 | }, 875 | "http-errors": { 876 | "version": "1.6.2", 877 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.2.tgz", 878 | "integrity": "sha1-CgAsyFcHGSp+eUbO7cERVfYOxzY=", 879 | "requires": { 880 | "depd": "1.1.1", 881 | "inherits": "2.0.3", 882 | "setprototypeof": "1.0.3", 883 | "statuses": "1.3.1" 884 | } 885 | }, 886 | "ignore": { 887 | "version": "3.3.3", 888 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.3.tgz", 889 | "integrity": "sha1-QyNS5XrM2HqzEQ6C0/6g5HgSFW0=", 890 | "dev": true 891 | }, 892 | "imurmurhash": { 893 | "version": "0.1.4", 894 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 895 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 896 | "dev": true 897 | }, 898 | "indexof": { 899 | "version": "0.0.1", 900 | "resolved": "https://registry.npmjs.org/indexof/-/indexof-0.0.1.tgz", 901 | "integrity": "sha1-gtwzbSMrkGIXnQWrMpOmYFn9Q10=" 902 | }, 903 | "inflight": { 904 | "version": "1.0.6", 905 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 906 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 907 | "dev": true, 908 | "requires": { 909 | "once": "1.4.0", 910 | "wrappy": "1.0.2" 911 | } 912 | }, 913 | "inherits": { 914 | "version": "2.0.3", 915 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 916 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 917 | }, 918 | "inquirer": { 919 | "version": "3.2.1", 920 | "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-3.2.1.tgz", 921 | "integrity": "sha512-QgW3eiPN8gpj/K5vVpHADJJgrrF0ho/dZGylikGX7iqAdRgC9FVKYKWFLx6hZDBFcOLEoSqINYrVPeFAeG/PdA==", 922 | "dev": true, 923 | "requires": { 924 | "ansi-escapes": "2.0.0", 925 | "chalk": "2.1.0", 926 | "cli-cursor": "2.1.0", 927 | "cli-width": "2.1.0", 928 | "external-editor": "2.0.4", 929 | "figures": "2.0.0", 930 | "lodash": "4.17.4", 931 | "mute-stream": "0.0.7", 932 | "run-async": "2.3.0", 933 | "rx-lite": "4.0.8", 934 | "rx-lite-aggregates": "4.0.8", 935 | "string-width": "2.1.1", 936 | "strip-ansi": "4.0.0", 937 | "through": "2.3.8" 938 | }, 939 | "dependencies": { 940 | "ansi-regex": { 941 | "version": "3.0.0", 942 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", 943 | "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", 944 | "dev": true 945 | }, 946 | "ansi-styles": { 947 | "version": "3.2.0", 948 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.0.tgz", 949 | "integrity": "sha512-NnSOmMEYtVR2JVMIGTzynRkkaxtiq1xnFBcdQD/DnNCYPoEPsVJhM98BDyaoNOQIi7p4okdi3E27eN7GQbsUug==", 950 | "dev": true, 951 | "requires": { 952 | "color-convert": "1.9.0" 953 | } 954 | }, 955 | "chalk": { 956 | "version": "2.1.0", 957 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.1.0.tgz", 958 | "integrity": "sha512-LUHGS/dge4ujbXMJrnihYMcL4AoOweGnw9Tp3kQuqy1Kx5c1qKjqvMJZ6nVJPMWJtKCTN72ZogH3oeSO9g9rXQ==", 959 | "dev": true, 960 | "requires": { 961 | "ansi-styles": "3.2.0", 962 | "escape-string-regexp": "1.0.5", 963 | "supports-color": "4.2.1" 964 | } 965 | }, 966 | "strip-ansi": { 967 | "version": "4.0.0", 968 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", 969 | "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", 970 | "dev": true, 971 | "requires": { 972 | "ansi-regex": "3.0.0" 973 | } 974 | }, 975 | "supports-color": { 976 | "version": "4.2.1", 977 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-4.2.1.tgz", 978 | "integrity": "sha512-qxzYsob3yv6U+xMzPrv170y8AwGP7i74g+pbixCfD6rgso8BscLT2qXIuz6TpOaiJZ3mFgT5O9lyT9nMU4LfaA==", 979 | "dev": true, 980 | "requires": { 981 | "has-flag": "2.0.0" 982 | } 983 | } 984 | } 985 | }, 986 | "ipaddr.js": { 987 | "version": "1.4.0", 988 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.4.0.tgz", 989 | "integrity": "sha1-KWrKh4qCGBbluF0KKFqZvP9FgvA=" 990 | }, 991 | "is-fullwidth-code-point": { 992 | "version": "2.0.0", 993 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 994 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 995 | "dev": true 996 | }, 997 | "is-path-cwd": { 998 | "version": "1.0.0", 999 | "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-1.0.0.tgz", 1000 | "integrity": "sha1-0iXsIxMuie3Tj9p2dHLmLmXxEG0=", 1001 | "dev": true 1002 | }, 1003 | "is-path-in-cwd": { 1004 | "version": "1.0.0", 1005 | "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-1.0.0.tgz", 1006 | "integrity": "sha1-ZHdYK4IU1gI0YJRWcAO+ip6sBNw=", 1007 | "dev": true, 1008 | "requires": { 1009 | "is-path-inside": "1.0.0" 1010 | } 1011 | }, 1012 | "is-path-inside": { 1013 | "version": "1.0.0", 1014 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.0.tgz", 1015 | "integrity": "sha1-/AbloWg/vaE95mev9xe7wQpI838=", 1016 | "dev": true, 1017 | "requires": { 1018 | "path-is-inside": "1.0.2" 1019 | } 1020 | }, 1021 | "is-promise": { 1022 | "version": "2.1.0", 1023 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz", 1024 | "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=", 1025 | "dev": true 1026 | }, 1027 | "is-resolvable": { 1028 | "version": "1.0.0", 1029 | "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.0.0.tgz", 1030 | "integrity": "sha1-jfV8YeouPFAUCNEA+wE8+NbgzGI=", 1031 | "dev": true, 1032 | "requires": { 1033 | "tryit": "1.0.3" 1034 | } 1035 | }, 1036 | "isarray": { 1037 | "version": "0.0.1", 1038 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", 1039 | "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" 1040 | }, 1041 | "isexe": { 1042 | "version": "2.0.0", 1043 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 1044 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 1045 | "dev": true 1046 | }, 1047 | "js-tokens": { 1048 | "version": "3.0.2", 1049 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", 1050 | "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls=", 1051 | "dev": true 1052 | }, 1053 | "js-yaml": { 1054 | "version": "3.9.1", 1055 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.9.1.tgz", 1056 | "integrity": "sha512-CbcG379L1e+mWBnLvHWWeLs8GyV/EMw862uLI3c+GxVyDHWZcjZinwuBd3iW2pgxgIlksW/1vNJa4to+RvDOww==", 1057 | "dev": true, 1058 | "requires": { 1059 | "argparse": "1.0.9", 1060 | "esprima": "4.0.0" 1061 | } 1062 | }, 1063 | "jschardet": { 1064 | "version": "1.5.1", 1065 | "resolved": "https://registry.npmjs.org/jschardet/-/jschardet-1.5.1.tgz", 1066 | "integrity": "sha512-vE2hT1D0HLZCLLclfBSfkfTTedhVj0fubHpJBHKwwUWX0nSbhPAfk+SG9rTX95BYNmau8rGFfCeaT6T5OW1C2A==", 1067 | "dev": true 1068 | }, 1069 | "json-schema-traverse": { 1070 | "version": "0.3.1", 1071 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz", 1072 | "integrity": "sha1-NJptRMU6Ud6JtAgFxdXlm0F9M0A=", 1073 | "dev": true 1074 | }, 1075 | "json-stable-stringify": { 1076 | "version": "1.0.1", 1077 | "resolved": "https://registry.npmjs.org/json-stable-stringify/-/json-stable-stringify-1.0.1.tgz", 1078 | "integrity": "sha1-mnWdOcXy/1A/1TAGRu1EX4jE+a8=", 1079 | "dev": true, 1080 | "requires": { 1081 | "jsonify": "0.0.0" 1082 | } 1083 | }, 1084 | "json3": { 1085 | "version": "3.2.6", 1086 | "resolved": "https://registry.npmjs.org/json3/-/json3-3.2.6.tgz", 1087 | "integrity": "sha1-9u/JPAagTemuxTBT3yVZuxniA4s=" 1088 | }, 1089 | "jsonify": { 1090 | "version": "0.0.0", 1091 | "resolved": "https://registry.npmjs.org/jsonify/-/jsonify-0.0.0.tgz", 1092 | "integrity": "sha1-LHS27kHZPKUbe1qu6PUDYx0lKnM=", 1093 | "dev": true 1094 | }, 1095 | "levn": { 1096 | "version": "0.3.0", 1097 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", 1098 | "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", 1099 | "dev": true, 1100 | "requires": { 1101 | "prelude-ls": "1.1.2", 1102 | "type-check": "0.3.2" 1103 | } 1104 | }, 1105 | "lodash": { 1106 | "version": "4.17.4", 1107 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz", 1108 | "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=", 1109 | "dev": true 1110 | }, 1111 | "lru-cache": { 1112 | "version": "4.1.1", 1113 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.1.tgz", 1114 | "integrity": "sha512-q4spe4KTfsAS1SUHLO0wz8Qiyf1+vMIAgpRYioFYDMNqKfHQbg+AVDH3i4fvpl71/P1L0dBl+fQi+P37UYf0ew==", 1115 | "dev": true, 1116 | "requires": { 1117 | "pseudomap": "1.0.2", 1118 | "yallist": "2.1.2" 1119 | } 1120 | }, 1121 | "media-typer": { 1122 | "version": "0.3.0", 1123 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 1124 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 1125 | }, 1126 | "merge-descriptors": { 1127 | "version": "1.0.1", 1128 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 1129 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 1130 | }, 1131 | "methods": { 1132 | "version": "1.1.2", 1133 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 1134 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 1135 | }, 1136 | "mime": { 1137 | "version": "1.3.4", 1138 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.3.4.tgz", 1139 | "integrity": "sha1-EV+eO2s9rylZmDyzjxSaLUDrXVM=" 1140 | }, 1141 | "mime-db": { 1142 | "version": "1.29.0", 1143 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.29.0.tgz", 1144 | "integrity": "sha1-SNJtI1WJZRcErFkWygYAGRQmaHg=" 1145 | }, 1146 | "mime-types": { 1147 | "version": "2.1.16", 1148 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.16.tgz", 1149 | "integrity": "sha1-K4WKUuXs1RbbiXrCvodIeDBpjiM=", 1150 | "requires": { 1151 | "mime-db": "1.29.0" 1152 | } 1153 | }, 1154 | "mimic-fn": { 1155 | "version": "1.1.0", 1156 | "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.1.0.tgz", 1157 | "integrity": "sha1-5md4PZLonb00KBi1IwudYqZyrRg=", 1158 | "dev": true 1159 | }, 1160 | "minimatch": { 1161 | "version": "3.0.4", 1162 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 1163 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 1164 | "dev": true, 1165 | "requires": { 1166 | "brace-expansion": "1.1.8" 1167 | } 1168 | }, 1169 | "minimist": { 1170 | "version": "0.0.8", 1171 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", 1172 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", 1173 | "dev": true 1174 | }, 1175 | "mkdirp": { 1176 | "version": "0.5.1", 1177 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", 1178 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", 1179 | "dev": true, 1180 | "requires": { 1181 | "minimist": "0.0.8" 1182 | } 1183 | }, 1184 | "ms": { 1185 | "version": "2.0.0", 1186 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1187 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 1188 | }, 1189 | "mute-stream": { 1190 | "version": "0.0.7", 1191 | "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", 1192 | "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=", 1193 | "dev": true 1194 | }, 1195 | "natural-compare": { 1196 | "version": "1.4.0", 1197 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", 1198 | "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", 1199 | "dev": true 1200 | }, 1201 | "negotiator": { 1202 | "version": "0.6.1", 1203 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.1.tgz", 1204 | "integrity": "sha1-KzJxhOiZIQEXeyhWP7XnECrNDKk=" 1205 | }, 1206 | "object-assign": { 1207 | "version": "4.1.1", 1208 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 1209 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", 1210 | "dev": true 1211 | }, 1212 | "object-component": { 1213 | "version": "0.0.3", 1214 | "resolved": "https://registry.npmjs.org/object-component/-/object-component-0.0.3.tgz", 1215 | "integrity": "sha1-8MaapQ78lbhmwYb0AKM3acsvEpE=" 1216 | }, 1217 | "on-finished": { 1218 | "version": "2.3.0", 1219 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 1220 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 1221 | "requires": { 1222 | "ee-first": "1.1.1" 1223 | } 1224 | }, 1225 | "once": { 1226 | "version": "1.4.0", 1227 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1228 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1229 | "dev": true, 1230 | "requires": { 1231 | "wrappy": "1.0.2" 1232 | } 1233 | }, 1234 | "onetime": { 1235 | "version": "2.0.1", 1236 | "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", 1237 | "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", 1238 | "dev": true, 1239 | "requires": { 1240 | "mimic-fn": "1.1.0" 1241 | } 1242 | }, 1243 | "optionator": { 1244 | "version": "0.8.2", 1245 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.2.tgz", 1246 | "integrity": "sha1-NkxeQJ0/TWMB1sC0wFu6UBgK62Q=", 1247 | "dev": true, 1248 | "requires": { 1249 | "deep-is": "0.1.3", 1250 | "fast-levenshtein": "2.0.6", 1251 | "levn": "0.3.0", 1252 | "prelude-ls": "1.1.2", 1253 | "type-check": "0.3.2", 1254 | "wordwrap": "1.0.0" 1255 | } 1256 | }, 1257 | "options": { 1258 | "version": "0.0.6", 1259 | "resolved": "https://registry.npmjs.org/options/-/options-0.0.6.tgz", 1260 | "integrity": "sha1-7CLTEoBrtT5zF3Pnza788cZDEo8=" 1261 | }, 1262 | "os-tmpdir": { 1263 | "version": "1.0.2", 1264 | "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", 1265 | "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", 1266 | "dev": true 1267 | }, 1268 | "parsejson": { 1269 | "version": "0.0.1", 1270 | "resolved": "https://registry.npmjs.org/parsejson/-/parsejson-0.0.1.tgz", 1271 | "integrity": "sha1-mxDGwNglq1ieaFFTgm3go7oni8w=", 1272 | "requires": { 1273 | "better-assert": "1.0.2" 1274 | } 1275 | }, 1276 | "parseqs": { 1277 | "version": "0.0.2", 1278 | "resolved": "https://registry.npmjs.org/parseqs/-/parseqs-0.0.2.tgz", 1279 | "integrity": "sha1-nf5wss3aw4i95PNbHyQPpYrb5sc=", 1280 | "requires": { 1281 | "better-assert": "1.0.2" 1282 | } 1283 | }, 1284 | "parseuri": { 1285 | "version": "0.0.4", 1286 | "resolved": "https://registry.npmjs.org/parseuri/-/parseuri-0.0.4.tgz", 1287 | "integrity": "sha1-gGWCo5iH4eoY3V4v4OAZAiaOk1A=", 1288 | "requires": { 1289 | "better-assert": "1.0.2" 1290 | } 1291 | }, 1292 | "parseurl": { 1293 | "version": "1.3.1", 1294 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.1.tgz", 1295 | "integrity": "sha1-yKuMkiO6NIiKpkopeyiFO+wY2lY=" 1296 | }, 1297 | "path-is-absolute": { 1298 | "version": "1.0.1", 1299 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1300 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 1301 | "dev": true 1302 | }, 1303 | "path-is-inside": { 1304 | "version": "1.0.2", 1305 | "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", 1306 | "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=", 1307 | "dev": true 1308 | }, 1309 | "path-to-regexp": { 1310 | "version": "0.1.7", 1311 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 1312 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 1313 | }, 1314 | "pify": { 1315 | "version": "2.3.0", 1316 | "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", 1317 | "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", 1318 | "dev": true 1319 | }, 1320 | "pinkie": { 1321 | "version": "2.0.4", 1322 | "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", 1323 | "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=", 1324 | "dev": true 1325 | }, 1326 | "pinkie-promise": { 1327 | "version": "2.0.1", 1328 | "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", 1329 | "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=", 1330 | "dev": true, 1331 | "requires": { 1332 | "pinkie": "2.0.4" 1333 | } 1334 | }, 1335 | "pluralize": { 1336 | "version": "4.0.0", 1337 | "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-4.0.0.tgz", 1338 | "integrity": "sha1-WbcIwcAZCi9pLxx2GMRGsFL9F2I=", 1339 | "dev": true 1340 | }, 1341 | "prelude-ls": { 1342 | "version": "1.1.2", 1343 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", 1344 | "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", 1345 | "dev": true 1346 | }, 1347 | "process-nextick-args": { 1348 | "version": "1.0.7", 1349 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-1.0.7.tgz", 1350 | "integrity": "sha1-FQ4gt1ZZCtP5EJPyWk8q2L/zC6M=", 1351 | "dev": true 1352 | }, 1353 | "progress": { 1354 | "version": "2.0.0", 1355 | "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.0.tgz", 1356 | "integrity": "sha1-ihvjZr+Pwj2yvSPxDG/pILQ4nR8=", 1357 | "dev": true 1358 | }, 1359 | "proxy-addr": { 1360 | "version": "1.1.5", 1361 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-1.1.5.tgz", 1362 | "integrity": "sha1-ccDuOxAt4/IC87ZPYI0XP8uhqRg=", 1363 | "requires": { 1364 | "forwarded": "0.1.0", 1365 | "ipaddr.js": "1.4.0" 1366 | } 1367 | }, 1368 | "pseudomap": { 1369 | "version": "1.0.2", 1370 | "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", 1371 | "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=", 1372 | "dev": true 1373 | }, 1374 | "range-parser": { 1375 | "version": "1.2.0", 1376 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz", 1377 | "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4=" 1378 | }, 1379 | "readable-stream": { 1380 | "version": "2.3.3", 1381 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.3.tgz", 1382 | "integrity": "sha512-m+qzzcn7KUxEmd1gMbchF+Y2eIUbieUaxkWtptyHywrX0rE8QEYqPC07Vuy4Wm32/xE16NcdBctb8S0Xe/5IeQ==", 1383 | "dev": true, 1384 | "requires": { 1385 | "core-util-is": "1.0.2", 1386 | "inherits": "2.0.3", 1387 | "isarray": "1.0.0", 1388 | "process-nextick-args": "1.0.7", 1389 | "safe-buffer": "5.1.1", 1390 | "string_decoder": "1.0.3", 1391 | "util-deprecate": "1.0.2" 1392 | }, 1393 | "dependencies": { 1394 | "isarray": { 1395 | "version": "1.0.0", 1396 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 1397 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", 1398 | "dev": true 1399 | } 1400 | } 1401 | }, 1402 | "require-uncached": { 1403 | "version": "1.0.3", 1404 | "resolved": "https://registry.npmjs.org/require-uncached/-/require-uncached-1.0.3.tgz", 1405 | "integrity": "sha1-Tg1W1slmL9MeQwEcS5WqSZVUIdM=", 1406 | "dev": true, 1407 | "requires": { 1408 | "caller-path": "0.1.0", 1409 | "resolve-from": "1.0.1" 1410 | } 1411 | }, 1412 | "resolve-from": { 1413 | "version": "1.0.1", 1414 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-1.0.1.tgz", 1415 | "integrity": "sha1-Jsv+k10a7uq7Kbw/5a6wHpPUQiY=", 1416 | "dev": true 1417 | }, 1418 | "restore-cursor": { 1419 | "version": "2.0.0", 1420 | "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", 1421 | "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", 1422 | "dev": true, 1423 | "requires": { 1424 | "onetime": "2.0.1", 1425 | "signal-exit": "3.0.2" 1426 | } 1427 | }, 1428 | "rimraf": { 1429 | "version": "2.6.1", 1430 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.1.tgz", 1431 | "integrity": "sha1-wjOOxkPfeht/5cVPqG9XQopV8z0=", 1432 | "dev": true, 1433 | "requires": { 1434 | "glob": "7.1.2" 1435 | } 1436 | }, 1437 | "run-async": { 1438 | "version": "2.3.0", 1439 | "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.3.0.tgz", 1440 | "integrity": "sha1-A3GrSuC91yDUFm19/aZP96RFpsA=", 1441 | "dev": true, 1442 | "requires": { 1443 | "is-promise": "2.1.0" 1444 | } 1445 | }, 1446 | "rx-lite": { 1447 | "version": "4.0.8", 1448 | "resolved": "https://registry.npmjs.org/rx-lite/-/rx-lite-4.0.8.tgz", 1449 | "integrity": "sha1-Cx4Rr4vESDbwSmQH6S2kJGe3lEQ=", 1450 | "dev": true 1451 | }, 1452 | "rx-lite-aggregates": { 1453 | "version": "4.0.8", 1454 | "resolved": "https://registry.npmjs.org/rx-lite-aggregates/-/rx-lite-aggregates-4.0.8.tgz", 1455 | "integrity": "sha1-dTuHqJoRyVRnxKwWJsTvxOBcZ74=", 1456 | "dev": true, 1457 | "requires": { 1458 | "rx-lite": "4.0.8" 1459 | } 1460 | }, 1461 | "safe-buffer": { 1462 | "version": "5.1.1", 1463 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz", 1464 | "integrity": "sha512-kKvNJn6Mm93gAczWVJg7wH+wGYWNrDHdWvpUmHyEsgCtIwwo3bqPtV4tR5tuPaUhTOo/kvhVwd8XwwOllGYkbg==", 1465 | "dev": true 1466 | }, 1467 | "semver": { 1468 | "version": "5.4.1", 1469 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.4.1.tgz", 1470 | "integrity": "sha512-WfG/X9+oATh81XtllIo/I8gOiY9EXRdv1cQdyykeXK17YcUW3EXUAi2To4pcH6nZtJPr7ZOpM5OMyWJZm+8Rsg==", 1471 | "dev": true 1472 | }, 1473 | "send": { 1474 | "version": "0.15.4", 1475 | "resolved": "https://registry.npmjs.org/send/-/send-0.15.4.tgz", 1476 | "integrity": "sha1-mF+qPihLAnPHkzZKNcZze9k5Bbk=", 1477 | "requires": { 1478 | "debug": "2.6.8", 1479 | "depd": "1.1.1", 1480 | "destroy": "1.0.4", 1481 | "encodeurl": "1.0.1", 1482 | "escape-html": "1.0.3", 1483 | "etag": "1.8.0", 1484 | "fresh": "0.5.0", 1485 | "http-errors": "1.6.2", 1486 | "mime": "1.3.4", 1487 | "ms": "2.0.0", 1488 | "on-finished": "2.3.0", 1489 | "range-parser": "1.2.0", 1490 | "statuses": "1.3.1" 1491 | }, 1492 | "dependencies": { 1493 | "debug": { 1494 | "version": "2.6.8", 1495 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.8.tgz", 1496 | "integrity": "sha1-5zFTHKLt4n0YgiJCfaF4IdaP9Pw=", 1497 | "requires": { 1498 | "ms": "2.0.0" 1499 | } 1500 | } 1501 | } 1502 | }, 1503 | "serve-static": { 1504 | "version": "1.12.4", 1505 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.12.4.tgz", 1506 | "integrity": "sha1-m2qpjutyU8Tu3Ewfb9vKYJkBqWE=", 1507 | "requires": { 1508 | "encodeurl": "1.0.1", 1509 | "escape-html": "1.0.3", 1510 | "parseurl": "1.3.1", 1511 | "send": "0.15.4" 1512 | } 1513 | }, 1514 | "setprototypeof": { 1515 | "version": "1.0.3", 1516 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.0.3.tgz", 1517 | "integrity": "sha1-ZlZ+NwQ+608E2RvWWMDL77VbjgQ=" 1518 | }, 1519 | "shebang-command": { 1520 | "version": "1.2.0", 1521 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", 1522 | "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", 1523 | "dev": true, 1524 | "requires": { 1525 | "shebang-regex": "1.0.0" 1526 | } 1527 | }, 1528 | "shebang-regex": { 1529 | "version": "1.0.0", 1530 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", 1531 | "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", 1532 | "dev": true 1533 | }, 1534 | "signal-exit": { 1535 | "version": "3.0.2", 1536 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", 1537 | "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", 1538 | "dev": true 1539 | }, 1540 | "slice-ansi": { 1541 | "version": "0.0.4", 1542 | "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-0.0.4.tgz", 1543 | "integrity": "sha1-7b+JA/ZvfOL46v1s7tZeJkyDGzU=", 1544 | "dev": true 1545 | }, 1546 | "socket.io": { 1547 | "version": "1.4.8", 1548 | "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-1.4.8.tgz", 1549 | "integrity": "sha1-5XbzMM0L7WTlWz/SbfmRFBiEhns=", 1550 | "requires": { 1551 | "debug": "2.2.0", 1552 | "engine.io": "1.6.11", 1553 | "has-binary": "0.1.7", 1554 | "socket.io-adapter": "0.4.0", 1555 | "socket.io-client": "1.4.8", 1556 | "socket.io-parser": "2.2.6" 1557 | }, 1558 | "dependencies": { 1559 | "debug": { 1560 | "version": "2.2.0", 1561 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.2.0.tgz", 1562 | "integrity": "sha1-+HBX6ZWxofauaklgZkE3vFbwOdo=", 1563 | "requires": { 1564 | "ms": "0.7.1" 1565 | } 1566 | }, 1567 | "ms": { 1568 | "version": "0.7.1", 1569 | "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.1.tgz", 1570 | "integrity": "sha1-nNE8A62/8ltl7/3nzoZO6VIBcJg=" 1571 | } 1572 | } 1573 | }, 1574 | "socket.io-adapter": { 1575 | "version": "0.4.0", 1576 | "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-0.4.0.tgz", 1577 | "integrity": "sha1-+5+CqxqmUpC/csNleVW5MKmRok8=", 1578 | "requires": { 1579 | "debug": "2.2.0", 1580 | "socket.io-parser": "2.2.2" 1581 | }, 1582 | "dependencies": { 1583 | "debug": { 1584 | "version": "2.2.0", 1585 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.2.0.tgz", 1586 | "integrity": "sha1-+HBX6ZWxofauaklgZkE3vFbwOdo=", 1587 | "requires": { 1588 | "ms": "0.7.1" 1589 | } 1590 | }, 1591 | "ms": { 1592 | "version": "0.7.1", 1593 | "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.1.tgz", 1594 | "integrity": "sha1-nNE8A62/8ltl7/3nzoZO6VIBcJg=" 1595 | }, 1596 | "socket.io-parser": { 1597 | "version": "2.2.2", 1598 | "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-2.2.2.tgz", 1599 | "integrity": "sha1-PXr2tkSX6Va32f53X5mXFgJ/lBc=", 1600 | "requires": { 1601 | "benchmark": "1.0.0", 1602 | "component-emitter": "1.1.2", 1603 | "debug": "0.7.4", 1604 | "isarray": "0.0.1", 1605 | "json3": "3.2.6" 1606 | }, 1607 | "dependencies": { 1608 | "debug": { 1609 | "version": "0.7.4", 1610 | "resolved": "https://registry.npmjs.org/debug/-/debug-0.7.4.tgz", 1611 | "integrity": "sha1-BuHqgILCyxTjmAbiLi9vdX+Srzk=" 1612 | } 1613 | } 1614 | } 1615 | } 1616 | }, 1617 | "socket.io-client": { 1618 | "version": "1.4.8", 1619 | "resolved": "https://registry.npmjs.org/socket.io-client/-/socket.io-client-1.4.8.tgz", 1620 | "integrity": "sha1-SBskHnPfFA6hpPsDSGqFrQl/VVg=", 1621 | "requires": { 1622 | "backo2": "1.0.2", 1623 | "component-bind": "1.0.0", 1624 | "component-emitter": "1.2.0", 1625 | "debug": "2.2.0", 1626 | "engine.io-client": "1.6.11", 1627 | "has-binary": "0.1.7", 1628 | "indexof": "0.0.1", 1629 | "object-component": "0.0.3", 1630 | "parseuri": "0.0.4", 1631 | "socket.io-parser": "2.2.6", 1632 | "to-array": "0.1.4" 1633 | }, 1634 | "dependencies": { 1635 | "component-emitter": { 1636 | "version": "1.2.0", 1637 | "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.0.tgz", 1638 | "integrity": "sha1-zNETqGOI0GSC0D3j/H35hSa6jv4=" 1639 | }, 1640 | "debug": { 1641 | "version": "2.2.0", 1642 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.2.0.tgz", 1643 | "integrity": "sha1-+HBX6ZWxofauaklgZkE3vFbwOdo=", 1644 | "requires": { 1645 | "ms": "0.7.1" 1646 | } 1647 | }, 1648 | "ms": { 1649 | "version": "0.7.1", 1650 | "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.1.tgz", 1651 | "integrity": "sha1-nNE8A62/8ltl7/3nzoZO6VIBcJg=" 1652 | } 1653 | } 1654 | }, 1655 | "socket.io-parser": { 1656 | "version": "2.2.6", 1657 | "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-2.2.6.tgz", 1658 | "integrity": "sha1-ON/WHfUNz4qx2eIJEyK/kCuii5k=", 1659 | "requires": { 1660 | "benchmark": "1.0.0", 1661 | "component-emitter": "1.1.2", 1662 | "debug": "2.2.0", 1663 | "isarray": "0.0.1", 1664 | "json3": "3.3.2" 1665 | }, 1666 | "dependencies": { 1667 | "debug": { 1668 | "version": "2.2.0", 1669 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.2.0.tgz", 1670 | "integrity": "sha1-+HBX6ZWxofauaklgZkE3vFbwOdo=", 1671 | "requires": { 1672 | "ms": "0.7.1" 1673 | } 1674 | }, 1675 | "json3": { 1676 | "version": "3.3.2", 1677 | "resolved": "https://registry.npmjs.org/json3/-/json3-3.3.2.tgz", 1678 | "integrity": "sha1-PAQ0dD35Pi9cQq7nsZvLSDV19OE=" 1679 | }, 1680 | "ms": { 1681 | "version": "0.7.1", 1682 | "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.1.tgz", 1683 | "integrity": "sha1-nNE8A62/8ltl7/3nzoZO6VIBcJg=" 1684 | } 1685 | } 1686 | }, 1687 | "sprintf-js": { 1688 | "version": "1.0.3", 1689 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 1690 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", 1691 | "dev": true 1692 | }, 1693 | "statuses": { 1694 | "version": "1.3.1", 1695 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.3.1.tgz", 1696 | "integrity": "sha1-+vUbnrdKrvOzrPStX2Gr8ky3uT4=" 1697 | }, 1698 | "string_decoder": { 1699 | "version": "1.0.3", 1700 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.0.3.tgz", 1701 | "integrity": "sha512-4AH6Z5fzNNBcH+6XDMfA/BTt87skxqJlO0lAh3Dker5zThcAxG6mKz+iGu308UKoPPQ8Dcqx/4JhujzltRa+hQ==", 1702 | "dev": true, 1703 | "requires": { 1704 | "safe-buffer": "5.1.1" 1705 | } 1706 | }, 1707 | "string-width": { 1708 | "version": "2.1.1", 1709 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", 1710 | "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", 1711 | "dev": true, 1712 | "requires": { 1713 | "is-fullwidth-code-point": "2.0.0", 1714 | "strip-ansi": "4.0.0" 1715 | }, 1716 | "dependencies": { 1717 | "ansi-regex": { 1718 | "version": "3.0.0", 1719 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", 1720 | "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", 1721 | "dev": true 1722 | }, 1723 | "strip-ansi": { 1724 | "version": "4.0.0", 1725 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", 1726 | "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", 1727 | "dev": true, 1728 | "requires": { 1729 | "ansi-regex": "3.0.0" 1730 | } 1731 | } 1732 | } 1733 | }, 1734 | "strip-ansi": { 1735 | "version": "3.0.1", 1736 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", 1737 | "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", 1738 | "dev": true, 1739 | "requires": { 1740 | "ansi-regex": "2.1.1" 1741 | } 1742 | }, 1743 | "strip-json-comments": { 1744 | "version": "2.0.1", 1745 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 1746 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 1747 | "dev": true 1748 | }, 1749 | "supports-color": { 1750 | "version": "2.0.0", 1751 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", 1752 | "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", 1753 | "dev": true 1754 | }, 1755 | "table": { 1756 | "version": "4.0.1", 1757 | "resolved": "https://registry.npmjs.org/table/-/table-4.0.1.tgz", 1758 | "integrity": "sha1-qBFsEz+sLGH0pCCrbN9cTWHw5DU=", 1759 | "dev": true, 1760 | "requires": { 1761 | "ajv": "4.11.8", 1762 | "ajv-keywords": "1.5.1", 1763 | "chalk": "1.1.3", 1764 | "lodash": "4.17.4", 1765 | "slice-ansi": "0.0.4", 1766 | "string-width": "2.1.1" 1767 | }, 1768 | "dependencies": { 1769 | "ajv": { 1770 | "version": "4.11.8", 1771 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-4.11.8.tgz", 1772 | "integrity": "sha1-gv+wKynmYq5TvcIK8VlHcGc5xTY=", 1773 | "dev": true, 1774 | "requires": { 1775 | "co": "4.6.0", 1776 | "json-stable-stringify": "1.0.1" 1777 | } 1778 | } 1779 | } 1780 | }, 1781 | "text-table": { 1782 | "version": "0.2.0", 1783 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", 1784 | "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", 1785 | "dev": true 1786 | }, 1787 | "through": { 1788 | "version": "2.3.8", 1789 | "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", 1790 | "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", 1791 | "dev": true 1792 | }, 1793 | "tmp": { 1794 | "version": "0.0.31", 1795 | "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.31.tgz", 1796 | "integrity": "sha1-jzirlDjhcxXl29izZX6L+yd65Kc=", 1797 | "dev": true, 1798 | "requires": { 1799 | "os-tmpdir": "1.0.2" 1800 | } 1801 | }, 1802 | "to-array": { 1803 | "version": "0.1.4", 1804 | "resolved": "https://registry.npmjs.org/to-array/-/to-array-0.1.4.tgz", 1805 | "integrity": "sha1-F+bBH3PdTz10zaek/zI46a2b+JA=" 1806 | }, 1807 | "tryit": { 1808 | "version": "1.0.3", 1809 | "resolved": "https://registry.npmjs.org/tryit/-/tryit-1.0.3.tgz", 1810 | "integrity": "sha1-OTvnMKlEb9Hq1tpZoBQwjzbCics=", 1811 | "dev": true 1812 | }, 1813 | "type-check": { 1814 | "version": "0.3.2", 1815 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", 1816 | "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", 1817 | "dev": true, 1818 | "requires": { 1819 | "prelude-ls": "1.1.2" 1820 | } 1821 | }, 1822 | "type-is": { 1823 | "version": "1.6.15", 1824 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.15.tgz", 1825 | "integrity": "sha1-yrEPtJCeRByChC6v4a1kbIGARBA=", 1826 | "requires": { 1827 | "media-typer": "0.3.0", 1828 | "mime-types": "2.1.16" 1829 | } 1830 | }, 1831 | "typedarray": { 1832 | "version": "0.0.6", 1833 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", 1834 | "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=", 1835 | "dev": true 1836 | }, 1837 | "ultron": { 1838 | "version": "1.0.2", 1839 | "resolved": "https://registry.npmjs.org/ultron/-/ultron-1.0.2.tgz", 1840 | "integrity": "sha1-rOEWq1V80Zc4ak6I9GhTeMiy5Po=" 1841 | }, 1842 | "unpipe": { 1843 | "version": "1.0.0", 1844 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1845 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 1846 | }, 1847 | "utf8": { 1848 | "version": "2.1.0", 1849 | "resolved": "https://registry.npmjs.org/utf8/-/utf8-2.1.0.tgz", 1850 | "integrity": "sha1-DP7FyAUtRKI+OqqQgQToB1+V39U=" 1851 | }, 1852 | "util-deprecate": { 1853 | "version": "1.0.2", 1854 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 1855 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", 1856 | "dev": true 1857 | }, 1858 | "utils-merge": { 1859 | "version": "1.0.0", 1860 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.0.tgz", 1861 | "integrity": "sha1-ApT7kiu5N1FTVBxPcJYjHyh8ivg=" 1862 | }, 1863 | "vary": { 1864 | "version": "1.1.1", 1865 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.1.tgz", 1866 | "integrity": "sha1-Z1Neu2lMHVIldFeYRmUyP1h+jTc=" 1867 | }, 1868 | "which": { 1869 | "version": "1.3.0", 1870 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.0.tgz", 1871 | "integrity": "sha512-xcJpopdamTuY5duC/KnTTNBraPK54YwpenP4lzxU8H91GudWpFv38u0CKjclE1Wi2EH2EDz5LRcHcKbCIzqGyg==", 1872 | "dev": true, 1873 | "requires": { 1874 | "isexe": "2.0.0" 1875 | } 1876 | }, 1877 | "wordwrap": { 1878 | "version": "1.0.0", 1879 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", 1880 | "integrity": "sha1-J1hIEIkUVqQXHI0CJkQa3pDLyus=", 1881 | "dev": true 1882 | }, 1883 | "wrappy": { 1884 | "version": "1.0.2", 1885 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1886 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1887 | "dev": true 1888 | }, 1889 | "write": { 1890 | "version": "0.2.1", 1891 | "resolved": "https://registry.npmjs.org/write/-/write-0.2.1.tgz", 1892 | "integrity": "sha1-X8A4KOJkzqP+kUVUdvejxWbLB1c=", 1893 | "dev": true, 1894 | "requires": { 1895 | "mkdirp": "0.5.1" 1896 | } 1897 | }, 1898 | "ws": { 1899 | "version": "1.1.0", 1900 | "resolved": "https://registry.npmjs.org/ws/-/ws-1.1.0.tgz", 1901 | "integrity": "sha1-wdb9FRXTzv8fCuJ1m/X9dwMKrR0=", 1902 | "requires": { 1903 | "options": "0.0.6", 1904 | "ultron": "1.0.2" 1905 | } 1906 | }, 1907 | "xmlhttprequest-ssl": { 1908 | "version": "1.5.1", 1909 | "resolved": "https://registry.npmjs.org/xmlhttprequest-ssl/-/xmlhttprequest-ssl-1.5.1.tgz", 1910 | "integrity": "sha1-O3dB/qSoZnWXbpCNKW1ERZYfqmc=" 1911 | }, 1912 | "yallist": { 1913 | "version": "2.1.2", 1914 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", 1915 | "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=", 1916 | "dev": true 1917 | }, 1918 | "yeast": { 1919 | "version": "0.1.2", 1920 | "resolved": "https://registry.npmjs.org/yeast/-/yeast-0.1.2.tgz", 1921 | "integrity": "sha1-AI4G2AlDIMNy28L47XagymyKxBk=" 1922 | } 1923 | } 1924 | } 1925 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "video-chat", 3 | "version": "1.0.0", 4 | "description": "Video Chat application.", 5 | "main": "server.js", 6 | "author": "Brian Mau", 7 | "scripts": { 8 | "start": "node server.js", 9 | "debug": "node --nolazy --inspect --debug-brk=5858 ./server.js" 10 | }, 11 | "dependencies": { 12 | "express": "^4.13.4", 13 | "socket.io": "1.4.*" 14 | }, 15 | "license": "ISC", 16 | "devDependencies": { 17 | "@types/node": "^8.0.10", 18 | "@types/socket.io-client": "^1.4.29", 19 | "@types/webrtc": "0.0.21", 20 | "eslint": "^4.2.0" 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /public/.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "globals": { 3 | "io": true 4 | }, 5 | "rules": { 6 | "no-unused-vars": "warn", 7 | "no-trailing-spaces": "warn", 8 | "semi": ["warn", "always"], 9 | "space-before-blocks": "warn", 10 | "no-console": ["off"], 11 | "brace-style": ["warn", "1tbs"], 12 | "camelcase": ["warn", {"properties": "never"}], 13 | "keyword-spacing": "warn", 14 | "space-infix-ops": "warn", 15 | "no-const-assign": "warn", 16 | "no-this-before-super": "warn", 17 | "no-undef": "warn", 18 | "no-unreachable": "warn", 19 | "constructor-super": "warn", 20 | "valid-typeof": "warn" 21 | }, 22 | "env": { 23 | "browser": true, 24 | "es6": true 25 | }, 26 | "parserOptions": { 27 | "ecmaVersion": 8, 28 | "ecmaFeatures": {} 29 | }, 30 | "extends": "eslint:recommended" 31 | } 32 | -------------------------------------------------------------------------------- /public/css/main.css: -------------------------------------------------------------------------------- 1 | 2 | html { 3 | overflow: hidden; 4 | height: 100%; 5 | } 6 | .one video { 7 | object-fit: cover; 8 | } 9 | .localVideo { 10 | width: 20%; 11 | position: absolute; 12 | z-index: 2; 13 | bottom: 10px; 14 | left: 10px; 15 | display: block; 16 | } 17 | .remoteVideos { 18 | width: 100%; 19 | height: 100%; 20 | position: absolute; 21 | top: 0; 22 | left: 0; 23 | display: flex; 24 | flex-direction: row; 25 | flex-wrap: nowrap; 26 | justify-content: flex-start; 27 | align-items: stretch; 28 | align-content: stretch; 29 | } 30 | .remoteVideos video { 31 | flex-grow: 1; 32 | flex-shrink: 1; 33 | min-width: 0px; 34 | min-height: 0px; 35 | } 36 | body { 37 | background-color: black; 38 | margin: 0; 39 | height: 100%; 40 | width: 100%; 41 | position: fixed; 42 | top: 0; 43 | left: 0; 44 | bottom: 0; 45 | right: 0; 46 | } 47 | 48 | -------------------------------------------------------------------------------- /public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Video Chat 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 |
14 | 15 | 16 | 17 | 18 | -------------------------------------------------------------------------------- /public/js/video.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | /** @type {SocketIOClient.Socket} */ 3 | const socket = io.connect(window.location.origin); 4 | const localVideo = document.querySelector('.localVideo'); 5 | const remoteVideos = document.querySelector('.remoteVideos'); 6 | const peerConnections = {}; 7 | 8 | let room = !location.pathname.substring(1) ? 'home' : location.pathname.substring(1); 9 | let getUserMediaAttempts = 5; 10 | let gettingUserMedia = false; 11 | 12 | /** @type {RTCConfiguration} */ 13 | const config = { 14 | 'iceServers': [{ 15 | 'urls': ['stun:stun.l.google.com:19302'] 16 | }] 17 | }; 18 | 19 | /** @type {MediaStreamConstraints} */ 20 | const constraints = { 21 | // audio: true, 22 | video: { facingMode: "user" } 23 | }; 24 | 25 | socket.on('full', function(room) { 26 | alert('Room ' + room + ' is full'); 27 | }); 28 | 29 | socket.on('bye', function(id) { 30 | handleRemoteHangup(id); 31 | }); 32 | 33 | if (room && !!room) { 34 | socket.emit('join', room); 35 | } 36 | 37 | window.onunload = window.onbeforeunload = function() { 38 | socket.close(); 39 | }; 40 | 41 | socket.on('ready', function (id) { 42 | if (!(localVideo instanceof HTMLVideoElement) || !localVideo.srcObject) { 43 | return; 44 | } 45 | const peerConnection = new RTCPeerConnection(config); 46 | peerConnections[id] = peerConnection; 47 | if (localVideo instanceof HTMLVideoElement) { 48 | peerConnection.addStream(localVideo.srcObject); 49 | } 50 | peerConnection.createOffer() 51 | .then(sdp => peerConnection.setLocalDescription(sdp)) 52 | .then(function () { 53 | socket.emit('offer', id, peerConnection.localDescription); 54 | }); 55 | peerConnection.onaddstream = event => handleRemoteStreamAdded(event.stream, id); 56 | peerConnection.onicecandidate = function(event) { 57 | if (event.candidate) { 58 | socket.emit('candidate', id, event.candidate); 59 | } 60 | }; 61 | }); 62 | 63 | socket.on('offer', function(id, description) { 64 | const peerConnection = new RTCPeerConnection(config); 65 | peerConnections[id] = peerConnection; 66 | if (localVideo instanceof HTMLVideoElement) { 67 | peerConnection.addStream(localVideo.srcObject); 68 | } 69 | peerConnection.setRemoteDescription(description) 70 | .then(() => peerConnection.createAnswer()) 71 | .then(sdp => peerConnection.setLocalDescription(sdp)) 72 | .then(function () { 73 | socket.emit('answer', id, peerConnection.localDescription); 74 | }); 75 | peerConnection.onaddstream = event => handleRemoteStreamAdded(event.stream, id); 76 | peerConnection.onicecandidate = function(event) { 77 | if (event.candidate) { 78 | socket.emit('candidate', id, event.candidate); 79 | } 80 | }; 81 | }); 82 | 83 | socket.on('candidate', function(id, candidate) { 84 | peerConnections[id].addIceCandidate(new RTCIceCandidate(candidate)) 85 | .catch(e => console.error(e)); 86 | }); 87 | 88 | socket.on('answer', function(id, description) { 89 | peerConnections[id].setRemoteDescription(description); 90 | }); 91 | 92 | function getUserMediaSuccess(stream) { 93 | gettingUserMedia = false; 94 | if (localVideo instanceof HTMLVideoElement) { 95 | !localVideo.srcObject && (localVideo.srcObject = stream); 96 | } 97 | socket.emit('ready'); 98 | } 99 | 100 | function handleRemoteStreamAdded(stream, id) { 101 | const remoteVideo = document.createElement('video'); 102 | remoteVideo.srcObject = stream; 103 | remoteVideo.setAttribute("id", id.replace(/[^a-zA-Z]+/g, "").toLowerCase()); 104 | remoteVideo.setAttribute("playsinline", "true"); 105 | remoteVideo.setAttribute("autoplay", "true"); 106 | remoteVideos.appendChild(remoteVideo); 107 | if (remoteVideos.querySelectorAll("video").length === 1) { 108 | remoteVideos.setAttribute("class", "one remoteVideos"); 109 | } else { 110 | remoteVideos.setAttribute("class", "remoteVideos"); 111 | } 112 | } 113 | 114 | function getUserMediaError(error) { 115 | console.error(error); 116 | gettingUserMedia = false; 117 | (--getUserMediaAttempts > 0) && setTimeout(getUserMediaDevices, 1000); 118 | } 119 | 120 | function getUserMediaDevices() { 121 | if (localVideo instanceof HTMLVideoElement) { 122 | if (localVideo.srcObject) { 123 | getUserMediaSuccess(localVideo.srcObject); 124 | } else if (!gettingUserMedia && !localVideo.srcObject) { 125 | gettingUserMedia = true; 126 | navigator.mediaDevices.getUserMedia(constraints) 127 | .then(getUserMediaSuccess) 128 | .catch(getUserMediaError); 129 | } 130 | } 131 | } 132 | 133 | function handleRemoteHangup(id) { 134 | peerConnections[id] && peerConnections[id].close(); 135 | delete peerConnections[id]; 136 | document.querySelector("#" + id.replace(/[^a-zA-Z]+/g, "").toLowerCase()).remove(); 137 | if (remoteVideos.querySelectorAll("video").length === 1) { 138 | remoteVideos.setAttribute("class", "one remoteVideos"); 139 | } else { 140 | remoteVideos.setAttribute("class", "remoteVideos"); 141 | } 142 | } 143 | 144 | getUserMediaDevices(); 145 | })(); -------------------------------------------------------------------------------- /server.js: -------------------------------------------------------------------------------- 1 | const credentials = require('./credentials'); 2 | const express = require('express'); 3 | const app = express(); 4 | let server; 5 | let port; 6 | if (credentials.key && credentials.cert) { 7 | const https = require('https'); 8 | server = https.createServer(credentials, app); 9 | port = 443; 10 | } else { 11 | const http = require('http'); 12 | server = http.createServer(app); 13 | port = 3000; 14 | } 15 | const io = require('socket.io')(server); 16 | const RoomService = require('./RoomService')(io); 17 | io.sockets.on('connection', RoomService.listen); 18 | io.sockets.on('error', e => console.log(e)); 19 | app.use(express.static(__dirname + '/public')); 20 | app.get('*', function(req, res) { 21 | res.sendFile(`${__dirname}/public/index.html`); 22 | }); 23 | server.listen(port, () => console.log(`Server is running on port ${port}`)); 24 | --------------------------------------------------------------------------------