├── .editorconfig ├── .gitignore ├── .prettierrc ├── http2 ├── exercise │ ├── backend │ │ └── server.js │ ├── frontend │ │ ├── http2-chat.js │ │ ├── index.html │ │ └── style.css │ ├── package-lock.json │ └── package.json └── push │ ├── backend │ └── server.js │ ├── frontend │ ├── http2-chat.js │ ├── index.html │ └── style.css │ ├── package-lock.json │ └── package.json ├── polling ├── backoff-and-retry │ ├── backend │ │ └── server.js │ ├── frontend │ │ ├── index.html │ │ ├── polling-chat.js │ │ └── style.css │ ├── package-lock.json │ └── package.json ├── exercise │ ├── backend │ │ └── server.js │ ├── frontend │ │ ├── index.html │ │ ├── polling-chat.js │ │ └── style.css │ ├── package-lock.json │ └── package.json ├── no-pause │ ├── backend │ │ └── server.js │ ├── frontend │ │ ├── index.html │ │ ├── polling-chat.js │ │ └── style.css │ ├── package-lock.json │ └── package.json └── pause-on-unfocus │ ├── backend │ └── server.js │ ├── frontend │ ├── index.html │ ├── polling-chat.js │ └── style.css │ ├── package-lock.json │ └── package.json └── websockets ├── exercise-raw ├── backend │ ├── generate-accept-value.js │ ├── obj-to-response.js │ ├── parse-message.js │ └── server.js ├── frontend │ ├── index.html │ ├── raw-chat.js │ └── style.css ├── package-lock.json └── package.json ├── exercise-socketio ├── backend │ └── server.js ├── frontend │ ├── index.html │ ├── socketio-chat.js │ └── style.css ├── package-lock.json └── package.json ├── raw ├── backend │ ├── generate-accept-value.js │ ├── obj-to-response.js │ ├── parse-message.js │ └── server.js ├── frontend │ ├── index.html │ ├── raw-chat.js │ └── style.css ├── package-lock.json └── package.json └── socketio ├── backend └── server.js ├── frontend ├── index.html ├── socketio-chat.js └── style.css ├── package-lock.json └── package.json /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [*] 4 | end_of_line = lf 5 | insert_final_newline = true 6 | charset = utf-8 7 | indent_style = space 8 | indent_size = 2 -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules/ 3 | .cache/ 4 | public/ 5 | *.pem 6 | *.crt 7 | -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | {} 2 | -------------------------------------------------------------------------------- /http2/exercise/backend/server.js: -------------------------------------------------------------------------------- 1 | import http2 from "http2"; 2 | import fs from "fs"; 3 | import path from "path"; 4 | import { fileURLToPath } from "url"; 5 | import handler from "serve-handler"; 6 | import nanobuffer from "nanobuffer"; 7 | 8 | let connections = []; 9 | 10 | const msg = new nanobuffer(50); 11 | const getMsgs = () => Array.from(msg).reverse(); 12 | 13 | msg.push({ 14 | user: "brian", 15 | text: "hi", 16 | time: Date.now(), 17 | }); 18 | 19 | // the two commands you'll have to run in the root directory of the project are 20 | // (not inside the backend folder) 21 | // 22 | // openssl req -new -newkey rsa:2048 -new -nodes -keyout key.pem -out csr.pem 23 | // openssl x509 -req -days 365 -in csr.pem -signkey key.pem -out server.crt 24 | // 25 | // http2 only works over HTTPS 26 | const __dirname = path.dirname(fileURLToPath(import.meta.url)); 27 | const server = http2.createSecureServer({ 28 | cert: fs.readFileSync(path.join(__dirname, "/../server.crt")), 29 | key: fs.readFileSync(path.join(__dirname, "/../key.pem")), 30 | }); 31 | 32 | /* 33 | * 34 | * Code goes here 35 | * 36 | */ 37 | 38 | server.on("request", async (req, res) => { 39 | const path = req.headers[":path"]; 40 | const method = req.headers[":method"]; 41 | 42 | if (path !== "/msgs") { 43 | // handle the static assets 44 | return handler(req, res, { 45 | public: "./frontend", 46 | }); 47 | } else if (method === "POST") { 48 | // get data out of post 49 | const buffers = []; 50 | for await (const chunk of req) { 51 | buffers.push(chunk); 52 | } 53 | const data = Buffer.concat(buffers).toString(); 54 | const { user, text } = JSON.parse(data); 55 | 56 | /* 57 | * 58 | * some code goes here 59 | * 60 | */ 61 | } 62 | }); 63 | 64 | // start listening 65 | const port = process.env.PORT || 8080; 66 | server.listen(port, () => 67 | console.log( 68 | `Server running at https://localhost:${port} - make sure you're on httpS, not http` 69 | ) 70 | ); 71 | -------------------------------------------------------------------------------- /http2/exercise/frontend/http2-chat.js: -------------------------------------------------------------------------------- 1 | const chat = document.getElementById("chat"); 2 | const msgs = document.getElementById("msgs"); 3 | const presence = document.getElementById("presence-indicator"); 4 | 5 | // this will hold all the most recent messages 6 | let allChat = []; 7 | 8 | chat.addEventListener("submit", function (e) { 9 | e.preventDefault(); 10 | postNewMsg(chat.elements.user.value, chat.elements.text.value); 11 | chat.elements.text.value = ""; 12 | }); 13 | 14 | async function postNewMsg(user, text) { 15 | const data = { 16 | user, 17 | text, 18 | }; 19 | 20 | // request options 21 | const options = { 22 | method: "POST", 23 | body: JSON.stringify(data), 24 | headers: { 25 | "Content-Type": "application/json", 26 | }, 27 | }; 28 | 29 | // send POST request 30 | // we're not sending any json back, but we could 31 | await fetch("/msgs", options); 32 | } 33 | 34 | async function getNewMsgs() { 35 | /* 36 | * 37 | * code goes here 38 | * 39 | */ 40 | } 41 | 42 | function render() { 43 | const html = allChat.map(({ user, text, time, id }) => 44 | template(user, text, time, id) 45 | ); 46 | msgs.innerHTML = html.join("\n"); 47 | } 48 | 49 | const template = (user, msg) => 50 | `
  • ${user}${msg}
  • `; 51 | 52 | getNewMsgs(); 53 | -------------------------------------------------------------------------------- /http2/exercise/frontend/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | HTTP/2 Chat 8 | 9 | 10 | 11 | 12 | 13 |
    🔴
    14 |

    Chat With Me

    15 |
    16 |
    17 |
    18 |
    19 | account_circle 20 | 21 | 22 |
    23 |
    24 | message 25 | 26 | 27 |
    28 |
    29 |
    30 | 33 |
    34 |
    35 |
    36 | 37 | 38 | 39 | 40 | 41 | 42 | -------------------------------------------------------------------------------- /http2/exercise/frontend/style.css: -------------------------------------------------------------------------------- 1 | .page-column { 2 | max-width: 600px; 3 | margin: 30px auto; 4 | } 5 | 6 | h1 { 7 | text-align: center; 8 | color: #26a69a; 9 | } 10 | 11 | .btn-box { 12 | text-align: center; 13 | } 14 | 15 | #presence-indicator { 16 | position: absolute; 17 | top: 5px; 18 | right: 15px; 19 | } 20 | -------------------------------------------------------------------------------- /http2/exercise/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "http2", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@sindresorhus/is": { 8 | "version": "0.14.0", 9 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz", 10 | "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==", 11 | "dev": true 12 | }, 13 | "@szmarczak/http-timer": { 14 | "version": "1.1.2", 15 | "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz", 16 | "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==", 17 | "dev": true, 18 | "requires": { 19 | "defer-to-connect": "^1.0.1" 20 | } 21 | }, 22 | "abbrev": { 23 | "version": "1.1.1", 24 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 25 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 26 | "dev": true 27 | }, 28 | "ansi-align": { 29 | "version": "3.0.0", 30 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz", 31 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==", 32 | "dev": true, 33 | "requires": { 34 | "string-width": "^3.0.0" 35 | }, 36 | "dependencies": { 37 | "string-width": { 38 | "version": "3.1.0", 39 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 40 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 41 | "dev": true, 42 | "requires": { 43 | "emoji-regex": "^7.0.1", 44 | "is-fullwidth-code-point": "^2.0.0", 45 | "strip-ansi": "^5.1.0" 46 | } 47 | } 48 | } 49 | }, 50 | "ansi-regex": { 51 | "version": "4.1.0", 52 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 53 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 54 | "dev": true 55 | }, 56 | "ansi-styles": { 57 | "version": "4.3.0", 58 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 59 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 60 | "dev": true, 61 | "requires": { 62 | "color-convert": "^2.0.1" 63 | } 64 | }, 65 | "anymatch": { 66 | "version": "3.1.2", 67 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 68 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 69 | "dev": true, 70 | "requires": { 71 | "normalize-path": "^3.0.0", 72 | "picomatch": "^2.0.4" 73 | } 74 | }, 75 | "balanced-match": { 76 | "version": "1.0.2", 77 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 78 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" 79 | }, 80 | "binary-extensions": { 81 | "version": "2.2.0", 82 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 83 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 84 | "dev": true 85 | }, 86 | "boxen": { 87 | "version": "4.2.0", 88 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz", 89 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==", 90 | "dev": true, 91 | "requires": { 92 | "ansi-align": "^3.0.0", 93 | "camelcase": "^5.3.1", 94 | "chalk": "^3.0.0", 95 | "cli-boxes": "^2.2.0", 96 | "string-width": "^4.1.0", 97 | "term-size": "^2.1.0", 98 | "type-fest": "^0.8.1", 99 | "widest-line": "^3.1.0" 100 | } 101 | }, 102 | "brace-expansion": { 103 | "version": "1.1.11", 104 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 105 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 106 | "requires": { 107 | "balanced-match": "^1.0.0", 108 | "concat-map": "0.0.1" 109 | } 110 | }, 111 | "braces": { 112 | "version": "3.0.2", 113 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 114 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 115 | "dev": true, 116 | "requires": { 117 | "fill-range": "^7.0.1" 118 | } 119 | }, 120 | "cacheable-request": { 121 | "version": "6.1.0", 122 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz", 123 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", 124 | "dev": true, 125 | "requires": { 126 | "clone-response": "^1.0.2", 127 | "get-stream": "^5.1.0", 128 | "http-cache-semantics": "^4.0.0", 129 | "keyv": "^3.0.0", 130 | "lowercase-keys": "^2.0.0", 131 | "normalize-url": "^4.1.0", 132 | "responselike": "^1.0.2" 133 | }, 134 | "dependencies": { 135 | "get-stream": { 136 | "version": "5.2.0", 137 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", 138 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", 139 | "dev": true, 140 | "requires": { 141 | "pump": "^3.0.0" 142 | } 143 | }, 144 | "lowercase-keys": { 145 | "version": "2.0.0", 146 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", 147 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", 148 | "dev": true 149 | } 150 | } 151 | }, 152 | "camelcase": { 153 | "version": "5.3.1", 154 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 155 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 156 | "dev": true 157 | }, 158 | "chalk": { 159 | "version": "3.0.0", 160 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", 161 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", 162 | "dev": true, 163 | "requires": { 164 | "ansi-styles": "^4.1.0", 165 | "supports-color": "^7.1.0" 166 | }, 167 | "dependencies": { 168 | "has-flag": { 169 | "version": "4.0.0", 170 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 171 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 172 | "dev": true 173 | }, 174 | "supports-color": { 175 | "version": "7.2.0", 176 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 177 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 178 | "dev": true, 179 | "requires": { 180 | "has-flag": "^4.0.0" 181 | } 182 | } 183 | } 184 | }, 185 | "chokidar": { 186 | "version": "3.5.2", 187 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.2.tgz", 188 | "integrity": "sha512-ekGhOnNVPgT77r4K/U3GDhu+FQ2S8TnK/s2KbIGXi0SZWuwkZ2QNyfWdZW+TVfn84DpEP7rLeCt2UI6bJ8GwbQ==", 189 | "dev": true, 190 | "requires": { 191 | "anymatch": "~3.1.2", 192 | "braces": "~3.0.2", 193 | "fsevents": "~2.3.2", 194 | "glob-parent": "~5.1.2", 195 | "is-binary-path": "~2.1.0", 196 | "is-glob": "~4.0.1", 197 | "normalize-path": "~3.0.0", 198 | "readdirp": "~3.6.0" 199 | } 200 | }, 201 | "ci-info": { 202 | "version": "2.0.0", 203 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz", 204 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", 205 | "dev": true 206 | }, 207 | "cli-boxes": { 208 | "version": "2.2.1", 209 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz", 210 | "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==", 211 | "dev": true 212 | }, 213 | "clone-response": { 214 | "version": "1.0.2", 215 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", 216 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", 217 | "dev": true, 218 | "requires": { 219 | "mimic-response": "^1.0.0" 220 | } 221 | }, 222 | "color-convert": { 223 | "version": "2.0.1", 224 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 225 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 226 | "dev": true, 227 | "requires": { 228 | "color-name": "~1.1.4" 229 | } 230 | }, 231 | "color-name": { 232 | "version": "1.1.4", 233 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 234 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 235 | "dev": true 236 | }, 237 | "concat-map": { 238 | "version": "0.0.1", 239 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 240 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" 241 | }, 242 | "configstore": { 243 | "version": "5.0.1", 244 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz", 245 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==", 246 | "dev": true, 247 | "requires": { 248 | "dot-prop": "^5.2.0", 249 | "graceful-fs": "^4.1.2", 250 | "make-dir": "^3.0.0", 251 | "unique-string": "^2.0.0", 252 | "write-file-atomic": "^3.0.0", 253 | "xdg-basedir": "^4.0.0" 254 | } 255 | }, 256 | "crypto-random-string": { 257 | "version": "2.0.0", 258 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", 259 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==", 260 | "dev": true 261 | }, 262 | "debug": { 263 | "version": "2.6.9", 264 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 265 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 266 | "dev": true, 267 | "requires": { 268 | "ms": "2.0.0" 269 | } 270 | }, 271 | "decompress-response": { 272 | "version": "3.3.0", 273 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", 274 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", 275 | "dev": true, 276 | "requires": { 277 | "mimic-response": "^1.0.0" 278 | } 279 | }, 280 | "deep-extend": { 281 | "version": "0.6.0", 282 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", 283 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", 284 | "dev": true 285 | }, 286 | "defer-to-connect": { 287 | "version": "1.1.3", 288 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz", 289 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==", 290 | "dev": true 291 | }, 292 | "dot-prop": { 293 | "version": "5.3.0", 294 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", 295 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", 296 | "dev": true, 297 | "requires": { 298 | "is-obj": "^2.0.0" 299 | } 300 | }, 301 | "duplexer3": { 302 | "version": "0.1.4", 303 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", 304 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", 305 | "dev": true 306 | }, 307 | "emoji-regex": { 308 | "version": "7.0.3", 309 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 310 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 311 | "dev": true 312 | }, 313 | "end-of-stream": { 314 | "version": "1.4.4", 315 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", 316 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", 317 | "dev": true, 318 | "requires": { 319 | "once": "^1.4.0" 320 | } 321 | }, 322 | "escape-goat": { 323 | "version": "2.1.1", 324 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz", 325 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==", 326 | "dev": true 327 | }, 328 | "fast-url-parser": { 329 | "version": "1.1.3", 330 | "resolved": "https://registry.npmjs.org/fast-url-parser/-/fast-url-parser-1.1.3.tgz", 331 | "integrity": "sha1-9K8+qfNNiicc9YrSs3WfQx8LMY0=", 332 | "requires": { 333 | "punycode": "^1.3.2" 334 | } 335 | }, 336 | "fill-range": { 337 | "version": "7.0.1", 338 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 339 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 340 | "dev": true, 341 | "requires": { 342 | "to-regex-range": "^5.0.1" 343 | } 344 | }, 345 | "fsevents": { 346 | "version": "2.3.2", 347 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 348 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 349 | "dev": true, 350 | "optional": true 351 | }, 352 | "get-stream": { 353 | "version": "4.1.0", 354 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", 355 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", 356 | "dev": true, 357 | "requires": { 358 | "pump": "^3.0.0" 359 | } 360 | }, 361 | "glob-parent": { 362 | "version": "5.1.2", 363 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 364 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 365 | "dev": true, 366 | "requires": { 367 | "is-glob": "^4.0.1" 368 | } 369 | }, 370 | "global-dirs": { 371 | "version": "2.1.0", 372 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.1.0.tgz", 373 | "integrity": "sha512-MG6kdOUh/xBnyo9cJFeIKkLEc1AyFq42QTU4XiX51i2NEdxLxLWXIjEjmqKeSuKR7pAZjTqUVoT2b2huxVLgYQ==", 374 | "dev": true, 375 | "requires": { 376 | "ini": "1.3.7" 377 | } 378 | }, 379 | "got": { 380 | "version": "9.6.0", 381 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz", 382 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", 383 | "dev": true, 384 | "requires": { 385 | "@sindresorhus/is": "^0.14.0", 386 | "@szmarczak/http-timer": "^1.1.2", 387 | "cacheable-request": "^6.0.0", 388 | "decompress-response": "^3.3.0", 389 | "duplexer3": "^0.1.4", 390 | "get-stream": "^4.1.0", 391 | "lowercase-keys": "^1.0.1", 392 | "mimic-response": "^1.0.1", 393 | "p-cancelable": "^1.0.0", 394 | "to-readable-stream": "^1.0.0", 395 | "url-parse-lax": "^3.0.0" 396 | } 397 | }, 398 | "graceful-fs": { 399 | "version": "4.2.6", 400 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.6.tgz", 401 | "integrity": "sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ==", 402 | "dev": true 403 | }, 404 | "has-flag": { 405 | "version": "3.0.0", 406 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 407 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 408 | "dev": true 409 | }, 410 | "has-yarn": { 411 | "version": "2.1.0", 412 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz", 413 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==", 414 | "dev": true 415 | }, 416 | "http-cache-semantics": { 417 | "version": "4.1.0", 418 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", 419 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", 420 | "dev": true 421 | }, 422 | "ignore-by-default": { 423 | "version": "1.0.1", 424 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 425 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", 426 | "dev": true 427 | }, 428 | "import-lazy": { 429 | "version": "2.1.0", 430 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", 431 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", 432 | "dev": true 433 | }, 434 | "imurmurhash": { 435 | "version": "0.1.4", 436 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 437 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 438 | "dev": true 439 | }, 440 | "ini": { 441 | "version": "1.3.7", 442 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.7.tgz", 443 | "integrity": "sha512-iKpRpXP+CrP2jyrxvg1kMUpXDyRUFDWurxbnVT1vQPx+Wz9uCYsMIqYuSBLV+PAaZG/d7kRLKRFc9oDMsH+mFQ==", 444 | "dev": true 445 | }, 446 | "is-binary-path": { 447 | "version": "2.1.0", 448 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 449 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 450 | "dev": true, 451 | "requires": { 452 | "binary-extensions": "^2.0.0" 453 | } 454 | }, 455 | "is-ci": { 456 | "version": "2.0.0", 457 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", 458 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", 459 | "dev": true, 460 | "requires": { 461 | "ci-info": "^2.0.0" 462 | } 463 | }, 464 | "is-extglob": { 465 | "version": "2.1.1", 466 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 467 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 468 | "dev": true 469 | }, 470 | "is-fullwidth-code-point": { 471 | "version": "2.0.0", 472 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 473 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 474 | "dev": true 475 | }, 476 | "is-glob": { 477 | "version": "4.0.1", 478 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 479 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 480 | "dev": true, 481 | "requires": { 482 | "is-extglob": "^2.1.1" 483 | } 484 | }, 485 | "is-installed-globally": { 486 | "version": "0.3.2", 487 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz", 488 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==", 489 | "dev": true, 490 | "requires": { 491 | "global-dirs": "^2.0.1", 492 | "is-path-inside": "^3.0.1" 493 | } 494 | }, 495 | "is-npm": { 496 | "version": "4.0.0", 497 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz", 498 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==", 499 | "dev": true 500 | }, 501 | "is-number": { 502 | "version": "7.0.0", 503 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 504 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 505 | "dev": true 506 | }, 507 | "is-obj": { 508 | "version": "2.0.0", 509 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", 510 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", 511 | "dev": true 512 | }, 513 | "is-path-inside": { 514 | "version": "3.0.3", 515 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", 516 | "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", 517 | "dev": true 518 | }, 519 | "is-typedarray": { 520 | "version": "1.0.0", 521 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 522 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", 523 | "dev": true 524 | }, 525 | "is-yarn-global": { 526 | "version": "0.3.0", 527 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", 528 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==", 529 | "dev": true 530 | }, 531 | "json-buffer": { 532 | "version": "3.0.0", 533 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", 534 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", 535 | "dev": true 536 | }, 537 | "keyv": { 538 | "version": "3.1.0", 539 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", 540 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==", 541 | "dev": true, 542 | "requires": { 543 | "json-buffer": "3.0.0" 544 | } 545 | }, 546 | "latest-version": { 547 | "version": "5.1.0", 548 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz", 549 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==", 550 | "dev": true, 551 | "requires": { 552 | "package-json": "^6.3.0" 553 | } 554 | }, 555 | "lowercase-keys": { 556 | "version": "1.0.1", 557 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", 558 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", 559 | "dev": true 560 | }, 561 | "make-dir": { 562 | "version": "3.1.0", 563 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 564 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 565 | "dev": true, 566 | "requires": { 567 | "semver": "^6.0.0" 568 | }, 569 | "dependencies": { 570 | "semver": { 571 | "version": "6.3.0", 572 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 573 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 574 | "dev": true 575 | } 576 | } 577 | }, 578 | "mimic-response": { 579 | "version": "1.0.1", 580 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", 581 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", 582 | "dev": true 583 | }, 584 | "minimatch": { 585 | "version": "3.0.4", 586 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 587 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 588 | "requires": { 589 | "brace-expansion": "^1.1.7" 590 | } 591 | }, 592 | "minimist": { 593 | "version": "1.2.5", 594 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 595 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", 596 | "dev": true 597 | }, 598 | "ms": { 599 | "version": "2.0.0", 600 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 601 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", 602 | "dev": true 603 | }, 604 | "nanobuffer": { 605 | "version": "2.0.0", 606 | "resolved": "https://registry.npmjs.org/nanobuffer/-/nanobuffer-2.0.0.tgz", 607 | "integrity": "sha512-5f7M0eixdz8SFuqseZXpK8Xkb7X70GBDrDWO+VXKiQtysiYRKtZmLhyr/I/25KCnmrsCzhAa8zJbGfGdqr7YIQ==" 608 | }, 609 | "nodemon": { 610 | "version": "2.0.12", 611 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.12.tgz", 612 | "integrity": "sha512-egCTmNZdObdBxUBw6ZNwvZ/xzk24CKRs5K6d+5zbmrMr7rOpPmfPeF6OxM3DDpaRx331CQRFEktn+wrFFfBSOA==", 613 | "dev": true, 614 | "requires": { 615 | "chokidar": "^3.2.2", 616 | "debug": "^3.2.6", 617 | "ignore-by-default": "^1.0.1", 618 | "minimatch": "^3.0.4", 619 | "pstree.remy": "^1.1.7", 620 | "semver": "^5.7.1", 621 | "supports-color": "^5.5.0", 622 | "touch": "^3.1.0", 623 | "undefsafe": "^2.0.3", 624 | "update-notifier": "^4.1.0" 625 | }, 626 | "dependencies": { 627 | "debug": { 628 | "version": "3.2.7", 629 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", 630 | "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", 631 | "dev": true, 632 | "requires": { 633 | "ms": "^2.1.1" 634 | } 635 | }, 636 | "ms": { 637 | "version": "2.1.3", 638 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 639 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 640 | "dev": true 641 | } 642 | } 643 | }, 644 | "nopt": { 645 | "version": "1.0.10", 646 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", 647 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", 648 | "dev": true, 649 | "requires": { 650 | "abbrev": "1" 651 | } 652 | }, 653 | "normalize-path": { 654 | "version": "3.0.0", 655 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 656 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 657 | "dev": true 658 | }, 659 | "normalize-url": { 660 | "version": "4.5.1", 661 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.1.tgz", 662 | "integrity": "sha512-9UZCFRHQdNrfTpGg8+1INIg93B6zE0aXMVFkw1WFwvO4SlZywU6aLg5Of0Ap/PgcbSw4LNxvMWXMeugwMCX0AA==", 663 | "dev": true 664 | }, 665 | "once": { 666 | "version": "1.4.0", 667 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 668 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 669 | "dev": true, 670 | "requires": { 671 | "wrappy": "1" 672 | } 673 | }, 674 | "p-cancelable": { 675 | "version": "1.1.0", 676 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz", 677 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==", 678 | "dev": true 679 | }, 680 | "package-json": { 681 | "version": "6.5.0", 682 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz", 683 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==", 684 | "dev": true, 685 | "requires": { 686 | "got": "^9.6.0", 687 | "registry-auth-token": "^4.0.0", 688 | "registry-url": "^5.0.0", 689 | "semver": "^6.2.0" 690 | }, 691 | "dependencies": { 692 | "semver": { 693 | "version": "6.3.0", 694 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 695 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 696 | "dev": true 697 | } 698 | } 699 | }, 700 | "path-is-inside": { 701 | "version": "1.0.2", 702 | "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", 703 | "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=" 704 | }, 705 | "picomatch": { 706 | "version": "2.3.0", 707 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", 708 | "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", 709 | "dev": true 710 | }, 711 | "prepend-http": { 712 | "version": "2.0.0", 713 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", 714 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", 715 | "dev": true 716 | }, 717 | "pstree.remy": { 718 | "version": "1.1.8", 719 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 720 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 721 | "dev": true 722 | }, 723 | "pump": { 724 | "version": "3.0.0", 725 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", 726 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", 727 | "dev": true, 728 | "requires": { 729 | "end-of-stream": "^1.1.0", 730 | "once": "^1.3.1" 731 | } 732 | }, 733 | "punycode": { 734 | "version": "1.4.1", 735 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", 736 | "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=" 737 | }, 738 | "pupa": { 739 | "version": "2.1.1", 740 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz", 741 | "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==", 742 | "dev": true, 743 | "requires": { 744 | "escape-goat": "^2.0.0" 745 | } 746 | }, 747 | "rc": { 748 | "version": "1.2.8", 749 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", 750 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", 751 | "dev": true, 752 | "requires": { 753 | "deep-extend": "^0.6.0", 754 | "ini": "~1.3.0", 755 | "minimist": "^1.2.0", 756 | "strip-json-comments": "~2.0.1" 757 | } 758 | }, 759 | "readdirp": { 760 | "version": "3.6.0", 761 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 762 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 763 | "dev": true, 764 | "requires": { 765 | "picomatch": "^2.2.1" 766 | } 767 | }, 768 | "registry-auth-token": { 769 | "version": "4.2.1", 770 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz", 771 | "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==", 772 | "dev": true, 773 | "requires": { 774 | "rc": "^1.2.8" 775 | } 776 | }, 777 | "registry-url": { 778 | "version": "5.1.0", 779 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz", 780 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==", 781 | "dev": true, 782 | "requires": { 783 | "rc": "^1.2.8" 784 | } 785 | }, 786 | "responselike": { 787 | "version": "1.0.2", 788 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", 789 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", 790 | "dev": true, 791 | "requires": { 792 | "lowercase-keys": "^1.0.0" 793 | } 794 | }, 795 | "semver": { 796 | "version": "5.7.1", 797 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 798 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 799 | "dev": true 800 | }, 801 | "semver-diff": { 802 | "version": "3.1.1", 803 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz", 804 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==", 805 | "dev": true, 806 | "requires": { 807 | "semver": "^6.3.0" 808 | }, 809 | "dependencies": { 810 | "semver": { 811 | "version": "6.3.0", 812 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 813 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 814 | "dev": true 815 | } 816 | } 817 | }, 818 | "serve-handler": { 819 | "version": "6.1.3", 820 | "resolved": "https://registry.npmjs.org/serve-handler/-/serve-handler-6.1.3.tgz", 821 | "integrity": "sha512-FosMqFBNrLyeiIDvP1zgO6YoTzFYHxLDEIavhlmQ+knB2Z7l1t+kGLHkZIDN7UVWqQAmKI3D20A6F6jo3nDd4w==", 822 | "requires": { 823 | "bytes": "3.0.0", 824 | "content-disposition": "0.5.2", 825 | "fast-url-parser": "1.1.3", 826 | "mime-types": "2.1.18", 827 | "minimatch": "3.0.4", 828 | "path-is-inside": "1.0.2", 829 | "path-to-regexp": "2.2.1", 830 | "range-parser": "1.2.0" 831 | }, 832 | "dependencies": { 833 | "bytes": { 834 | "version": "3.0.0", 835 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", 836 | "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" 837 | }, 838 | "content-disposition": { 839 | "version": "0.5.2", 840 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", 841 | "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ=" 842 | }, 843 | "mime-db": { 844 | "version": "1.33.0", 845 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.33.0.tgz", 846 | "integrity": "sha512-BHJ/EKruNIqJf/QahvxwQZXKygOQ256myeN/Ew+THcAa5q+PjyTTMMeNQC4DZw5AwfvelsUrA6B67NKMqXDbzQ==" 847 | }, 848 | "mime-types": { 849 | "version": "2.1.18", 850 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.18.tgz", 851 | "integrity": "sha512-lc/aahn+t4/SWV/qcmumYjymLsWfN3ELhpmVuUFjgsORruuZPVSwAQryq+HHGvO/SI2KVX26bx+En+zhM8g8hQ==", 852 | "requires": { 853 | "mime-db": "~1.33.0" 854 | } 855 | }, 856 | "path-to-regexp": { 857 | "version": "2.2.1", 858 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-2.2.1.tgz", 859 | "integrity": "sha512-gu9bD6Ta5bwGrrU8muHzVOBFFREpp2iRkVfhBJahwJ6p6Xw20SjT0MxLnwkjOibQmGSYhiUnf2FLe7k+jcFmGQ==" 860 | }, 861 | "range-parser": { 862 | "version": "1.2.0", 863 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz", 864 | "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4=" 865 | } 866 | } 867 | }, 868 | "signal-exit": { 869 | "version": "3.0.3", 870 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", 871 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", 872 | "dev": true 873 | }, 874 | "string-width": { 875 | "version": "4.2.2", 876 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", 877 | "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", 878 | "dev": true, 879 | "requires": { 880 | "emoji-regex": "^8.0.0", 881 | "is-fullwidth-code-point": "^3.0.0", 882 | "strip-ansi": "^6.0.0" 883 | }, 884 | "dependencies": { 885 | "ansi-regex": { 886 | "version": "5.0.0", 887 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 888 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 889 | "dev": true 890 | }, 891 | "emoji-regex": { 892 | "version": "8.0.0", 893 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 894 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 895 | "dev": true 896 | }, 897 | "is-fullwidth-code-point": { 898 | "version": "3.0.0", 899 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 900 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 901 | "dev": true 902 | }, 903 | "strip-ansi": { 904 | "version": "6.0.0", 905 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 906 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 907 | "dev": true, 908 | "requires": { 909 | "ansi-regex": "^5.0.0" 910 | } 911 | } 912 | } 913 | }, 914 | "strip-ansi": { 915 | "version": "5.2.0", 916 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 917 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 918 | "dev": true, 919 | "requires": { 920 | "ansi-regex": "^4.1.0" 921 | } 922 | }, 923 | "strip-json-comments": { 924 | "version": "2.0.1", 925 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 926 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 927 | "dev": true 928 | }, 929 | "supports-color": { 930 | "version": "5.5.0", 931 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 932 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 933 | "dev": true, 934 | "requires": { 935 | "has-flag": "^3.0.0" 936 | } 937 | }, 938 | "term-size": { 939 | "version": "2.2.1", 940 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz", 941 | "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==", 942 | "dev": true 943 | }, 944 | "to-readable-stream": { 945 | "version": "1.0.0", 946 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz", 947 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==", 948 | "dev": true 949 | }, 950 | "to-regex-range": { 951 | "version": "5.0.1", 952 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 953 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 954 | "dev": true, 955 | "requires": { 956 | "is-number": "^7.0.0" 957 | } 958 | }, 959 | "touch": { 960 | "version": "3.1.0", 961 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", 962 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", 963 | "dev": true, 964 | "requires": { 965 | "nopt": "~1.0.10" 966 | } 967 | }, 968 | "type-fest": { 969 | "version": "0.8.1", 970 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 971 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", 972 | "dev": true 973 | }, 974 | "typedarray-to-buffer": { 975 | "version": "3.1.5", 976 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", 977 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", 978 | "dev": true, 979 | "requires": { 980 | "is-typedarray": "^1.0.0" 981 | } 982 | }, 983 | "undefsafe": { 984 | "version": "2.0.3", 985 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz", 986 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==", 987 | "dev": true, 988 | "requires": { 989 | "debug": "^2.2.0" 990 | } 991 | }, 992 | "unique-string": { 993 | "version": "2.0.0", 994 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz", 995 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==", 996 | "dev": true, 997 | "requires": { 998 | "crypto-random-string": "^2.0.0" 999 | } 1000 | }, 1001 | "update-notifier": { 1002 | "version": "4.1.3", 1003 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz", 1004 | "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==", 1005 | "dev": true, 1006 | "requires": { 1007 | "boxen": "^4.2.0", 1008 | "chalk": "^3.0.0", 1009 | "configstore": "^5.0.1", 1010 | "has-yarn": "^2.1.0", 1011 | "import-lazy": "^2.1.0", 1012 | "is-ci": "^2.0.0", 1013 | "is-installed-globally": "^0.3.1", 1014 | "is-npm": "^4.0.0", 1015 | "is-yarn-global": "^0.3.0", 1016 | "latest-version": "^5.0.0", 1017 | "pupa": "^2.0.1", 1018 | "semver-diff": "^3.1.1", 1019 | "xdg-basedir": "^4.0.0" 1020 | } 1021 | }, 1022 | "url-parse-lax": { 1023 | "version": "3.0.0", 1024 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", 1025 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", 1026 | "dev": true, 1027 | "requires": { 1028 | "prepend-http": "^2.0.0" 1029 | } 1030 | }, 1031 | "widest-line": { 1032 | "version": "3.1.0", 1033 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz", 1034 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==", 1035 | "dev": true, 1036 | "requires": { 1037 | "string-width": "^4.0.0" 1038 | } 1039 | }, 1040 | "wrappy": { 1041 | "version": "1.0.2", 1042 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1043 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1044 | "dev": true 1045 | }, 1046 | "write-file-atomic": { 1047 | "version": "3.0.3", 1048 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", 1049 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", 1050 | "dev": true, 1051 | "requires": { 1052 | "imurmurhash": "^0.1.4", 1053 | "is-typedarray": "^1.0.0", 1054 | "signal-exit": "^3.0.2", 1055 | "typedarray-to-buffer": "^3.1.5" 1056 | } 1057 | }, 1058 | "xdg-basedir": { 1059 | "version": "4.0.0", 1060 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz", 1061 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==", 1062 | "dev": true 1063 | } 1064 | } 1065 | } 1066 | -------------------------------------------------------------------------------- /http2/exercise/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "http2", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node backend/server.js", 8 | "dev": "nodemon backend/server.js" 9 | }, 10 | "keywords": [], 11 | "type": "module", 12 | "author": "", 13 | "license": "ISC", 14 | "dependencies": { 15 | "nanobuffer": "^2.0.0", 16 | "serve-handler": "^6.1.3" 17 | }, 18 | "devDependencies": { 19 | "nodemon": "^2.0.12" 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /http2/push/backend/server.js: -------------------------------------------------------------------------------- 1 | import http2 from "http2"; 2 | import fs from "fs"; 3 | import path from "path"; 4 | import { fileURLToPath } from "url"; 5 | import handler from "serve-handler"; 6 | import nanobuffer from "nanobuffer"; 7 | 8 | let connections = []; 9 | 10 | const msg = new nanobuffer(50); 11 | const getMsgs = () => Array.from(msg).reverse(); 12 | 13 | msg.push({ 14 | user: "brian?", 15 | text: "hi", 16 | time: Date.now(), 17 | }); 18 | 19 | // openssl req -new -newkey rsa:2048 -new -nodes -keyout key.pem -out csr.pem 20 | // openssl x509 -req -days 365 -in csr.pem -signkey key.pem -out server.crt 21 | // http2 only works over HTTPS 22 | const __dirname = path.dirname(fileURLToPath(import.meta.url)); 23 | const server = http2.createSecureServer({ 24 | cert: fs.readFileSync(path.join(__dirname, "/../server.crt")), 25 | key: fs.readFileSync(path.join(__dirname, "/../key.pem")), 26 | }); 27 | 28 | server.on("stream", (stream, headers) => { 29 | const method = headers[":method"]; 30 | const path = headers[":path"]; 31 | 32 | // streams will open for everything, we want just GETs on /msgs 33 | if (path === "/msgs" && method === "GET") { 34 | // immediately respond with 200 OK and encoding 35 | stream.respond({ 36 | ":status": 200, 37 | "content-type": "text/plain; charset=utf-8", 38 | }); 39 | 40 | // write the first response 41 | stream.write(JSON.stringify({ msg: getMsgs() })); 42 | 43 | // keep track of the connection 44 | connections.push(stream); 45 | 46 | // when the connection closes, stop keeping track of it 47 | stream.on("close", () => { 48 | connections = connections.filter((s) => s !== stream); 49 | }); 50 | } 51 | }); 52 | 53 | server.on("request", async (req, res) => { 54 | const path = req.headers[":path"]; 55 | const method = req.headers[":method"]; 56 | 57 | if (path !== "/msgs") { 58 | // handle the static assets 59 | return handler(req, res, { 60 | public: "./frontend", 61 | }); 62 | } else if (method === "POST") { 63 | // get data out of post 64 | const buffers = []; 65 | for await (const chunk of req) { 66 | buffers.push(chunk); 67 | } 68 | const data = Buffer.concat(buffers).toString(); 69 | const { user, text } = JSON.parse(data); 70 | msg.push({ 71 | user, 72 | text, 73 | time: Date.now(), 74 | }); 75 | 76 | // all done with the request 77 | res.end(); 78 | 79 | // notify all connected users 80 | connections.forEach((stream) => { 81 | stream.write(JSON.stringify({ msg: getMsgs() })); 82 | }); 83 | } 84 | }); 85 | 86 | // start listening 87 | const port = process.env.PORT || 8080; 88 | server.listen(port, () => 89 | console.log( 90 | `Server running at https://localhost:${port} - make sure you're on httpS, not http` 91 | ) 92 | ); 93 | -------------------------------------------------------------------------------- /http2/push/frontend/http2-chat.js: -------------------------------------------------------------------------------- 1 | const chat = document.getElementById("chat"); 2 | const msgs = document.getElementById("msgs"); 3 | const presence = document.getElementById("presence-indicator"); 4 | 5 | // this will hold all the most recent messages 6 | let allChat = []; 7 | 8 | chat.addEventListener("submit", function (e) { 9 | e.preventDefault(); 10 | postNewMsg(chat.elements.user.value, chat.elements.text.value); 11 | chat.elements.text.value = ""; 12 | }); 13 | 14 | async function postNewMsg(user, text) { 15 | const data = { 16 | user, 17 | text, 18 | }; 19 | 20 | // request options 21 | const options = { 22 | method: "POST", 23 | body: JSON.stringify(data), 24 | headers: { 25 | "Content-Type": "application/json", 26 | }, 27 | }; 28 | 29 | // send POST request 30 | // we're not sending any json back, but we could 31 | await fetch("/msgs", options); 32 | } 33 | 34 | async function getNewMsgs() { 35 | let reader; 36 | const utf8Decoder = new TextDecoder("utf-8"); 37 | try { 38 | const res = await fetch("/msgs"); 39 | reader = res.body.getReader(); 40 | } catch (e) { 41 | console.log("connection error", e); 42 | } 43 | let done; 44 | presence.innerText = "🟢"; 45 | do { 46 | let readerResponse; 47 | try { 48 | readerResponse = await reader.read(); 49 | } catch (e) { 50 | console.error("reader failed", e); 51 | presence.innerText = "🔴"; 52 | return; 53 | } 54 | done = readerResponse.done; 55 | const chunk = utf8Decoder.decode(readerResponse.value, { stream: true }); 56 | if (chunk) { 57 | try { 58 | const json = JSON.parse(chunk); 59 | allChat = json.msg; 60 | render(); 61 | } catch (e) { 62 | console.error("parse error", e); 63 | } 64 | } 65 | console.log("done", done); 66 | } while (!done); 67 | // in theory, if our http2 connection closed, `done` would come back 68 | // as true and we'd no longer be connected 69 | presence.innerText = "🔴"; 70 | } 71 | 72 | function render() { 73 | const html = allChat.map(({ user, text, time, id }) => 74 | template(user, text, time, id) 75 | ); 76 | msgs.innerHTML = html.join("\n"); 77 | } 78 | 79 | const template = (user, msg) => 80 | `
  • ${user}${msg}
  • `; 81 | 82 | getNewMsgs(); 83 | -------------------------------------------------------------------------------- /http2/push/frontend/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | HTTP/2 Chat 8 | 9 | 10 | 11 | 12 | 13 |
    🔴
    14 |

    Chat With Me

    15 |
    16 |
    17 |
    18 |
    19 | account_circle 20 | 21 | 22 |
    23 |
    24 | message 25 | 26 | 27 |
    28 |
    29 |
    30 | 33 |
    34 |
    35 |
    36 | 37 | 38 | 39 | 40 | 41 | 42 | -------------------------------------------------------------------------------- /http2/push/frontend/style.css: -------------------------------------------------------------------------------- 1 | .page-column { 2 | max-width: 600px; 3 | margin: 30px auto; 4 | } 5 | 6 | h1 { 7 | text-align: center; 8 | color: #26a69a; 9 | } 10 | 11 | .btn-box { 12 | text-align: center; 13 | } 14 | 15 | #presence-indicator { 16 | position: absolute; 17 | top: 5px; 18 | right: 15px; 19 | } 20 | -------------------------------------------------------------------------------- /http2/push/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "http2", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@sindresorhus/is": { 8 | "version": "0.14.0", 9 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz", 10 | "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==", 11 | "dev": true 12 | }, 13 | "@szmarczak/http-timer": { 14 | "version": "1.1.2", 15 | "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz", 16 | "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==", 17 | "dev": true, 18 | "requires": { 19 | "defer-to-connect": "^1.0.1" 20 | } 21 | }, 22 | "abbrev": { 23 | "version": "1.1.1", 24 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 25 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 26 | "dev": true 27 | }, 28 | "ansi-align": { 29 | "version": "3.0.0", 30 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz", 31 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==", 32 | "dev": true, 33 | "requires": { 34 | "string-width": "^3.0.0" 35 | }, 36 | "dependencies": { 37 | "string-width": { 38 | "version": "3.1.0", 39 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 40 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 41 | "dev": true, 42 | "requires": { 43 | "emoji-regex": "^7.0.1", 44 | "is-fullwidth-code-point": "^2.0.0", 45 | "strip-ansi": "^5.1.0" 46 | } 47 | } 48 | } 49 | }, 50 | "ansi-regex": { 51 | "version": "4.1.0", 52 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 53 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 54 | "dev": true 55 | }, 56 | "ansi-styles": { 57 | "version": "4.3.0", 58 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 59 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 60 | "dev": true, 61 | "requires": { 62 | "color-convert": "^2.0.1" 63 | } 64 | }, 65 | "anymatch": { 66 | "version": "3.1.2", 67 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 68 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 69 | "dev": true, 70 | "requires": { 71 | "normalize-path": "^3.0.0", 72 | "picomatch": "^2.0.4" 73 | } 74 | }, 75 | "balanced-match": { 76 | "version": "1.0.2", 77 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 78 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" 79 | }, 80 | "binary-extensions": { 81 | "version": "2.2.0", 82 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 83 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 84 | "dev": true 85 | }, 86 | "boxen": { 87 | "version": "4.2.0", 88 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz", 89 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==", 90 | "dev": true, 91 | "requires": { 92 | "ansi-align": "^3.0.0", 93 | "camelcase": "^5.3.1", 94 | "chalk": "^3.0.0", 95 | "cli-boxes": "^2.2.0", 96 | "string-width": "^4.1.0", 97 | "term-size": "^2.1.0", 98 | "type-fest": "^0.8.1", 99 | "widest-line": "^3.1.0" 100 | } 101 | }, 102 | "brace-expansion": { 103 | "version": "1.1.11", 104 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 105 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 106 | "requires": { 107 | "balanced-match": "^1.0.0", 108 | "concat-map": "0.0.1" 109 | } 110 | }, 111 | "braces": { 112 | "version": "3.0.2", 113 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 114 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 115 | "dev": true, 116 | "requires": { 117 | "fill-range": "^7.0.1" 118 | } 119 | }, 120 | "cacheable-request": { 121 | "version": "6.1.0", 122 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz", 123 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", 124 | "dev": true, 125 | "requires": { 126 | "clone-response": "^1.0.2", 127 | "get-stream": "^5.1.0", 128 | "http-cache-semantics": "^4.0.0", 129 | "keyv": "^3.0.0", 130 | "lowercase-keys": "^2.0.0", 131 | "normalize-url": "^4.1.0", 132 | "responselike": "^1.0.2" 133 | }, 134 | "dependencies": { 135 | "get-stream": { 136 | "version": "5.2.0", 137 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", 138 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", 139 | "dev": true, 140 | "requires": { 141 | "pump": "^3.0.0" 142 | } 143 | }, 144 | "lowercase-keys": { 145 | "version": "2.0.0", 146 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", 147 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", 148 | "dev": true 149 | } 150 | } 151 | }, 152 | "camelcase": { 153 | "version": "5.3.1", 154 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 155 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 156 | "dev": true 157 | }, 158 | "chalk": { 159 | "version": "3.0.0", 160 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", 161 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", 162 | "dev": true, 163 | "requires": { 164 | "ansi-styles": "^4.1.0", 165 | "supports-color": "^7.1.0" 166 | }, 167 | "dependencies": { 168 | "has-flag": { 169 | "version": "4.0.0", 170 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 171 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 172 | "dev": true 173 | }, 174 | "supports-color": { 175 | "version": "7.2.0", 176 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 177 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 178 | "dev": true, 179 | "requires": { 180 | "has-flag": "^4.0.0" 181 | } 182 | } 183 | } 184 | }, 185 | "chokidar": { 186 | "version": "3.5.2", 187 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.2.tgz", 188 | "integrity": "sha512-ekGhOnNVPgT77r4K/U3GDhu+FQ2S8TnK/s2KbIGXi0SZWuwkZ2QNyfWdZW+TVfn84DpEP7rLeCt2UI6bJ8GwbQ==", 189 | "dev": true, 190 | "requires": { 191 | "anymatch": "~3.1.2", 192 | "braces": "~3.0.2", 193 | "fsevents": "~2.3.2", 194 | "glob-parent": "~5.1.2", 195 | "is-binary-path": "~2.1.0", 196 | "is-glob": "~4.0.1", 197 | "normalize-path": "~3.0.0", 198 | "readdirp": "~3.6.0" 199 | } 200 | }, 201 | "ci-info": { 202 | "version": "2.0.0", 203 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz", 204 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", 205 | "dev": true 206 | }, 207 | "cli-boxes": { 208 | "version": "2.2.1", 209 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz", 210 | "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==", 211 | "dev": true 212 | }, 213 | "clone-response": { 214 | "version": "1.0.2", 215 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", 216 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", 217 | "dev": true, 218 | "requires": { 219 | "mimic-response": "^1.0.0" 220 | } 221 | }, 222 | "color-convert": { 223 | "version": "2.0.1", 224 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 225 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 226 | "dev": true, 227 | "requires": { 228 | "color-name": "~1.1.4" 229 | } 230 | }, 231 | "color-name": { 232 | "version": "1.1.4", 233 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 234 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 235 | "dev": true 236 | }, 237 | "concat-map": { 238 | "version": "0.0.1", 239 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 240 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" 241 | }, 242 | "configstore": { 243 | "version": "5.0.1", 244 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz", 245 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==", 246 | "dev": true, 247 | "requires": { 248 | "dot-prop": "^5.2.0", 249 | "graceful-fs": "^4.1.2", 250 | "make-dir": "^3.0.0", 251 | "unique-string": "^2.0.0", 252 | "write-file-atomic": "^3.0.0", 253 | "xdg-basedir": "^4.0.0" 254 | } 255 | }, 256 | "crypto-random-string": { 257 | "version": "2.0.0", 258 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", 259 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==", 260 | "dev": true 261 | }, 262 | "debug": { 263 | "version": "2.6.9", 264 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 265 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 266 | "dev": true, 267 | "requires": { 268 | "ms": "2.0.0" 269 | } 270 | }, 271 | "decompress-response": { 272 | "version": "3.3.0", 273 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", 274 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", 275 | "dev": true, 276 | "requires": { 277 | "mimic-response": "^1.0.0" 278 | } 279 | }, 280 | "deep-extend": { 281 | "version": "0.6.0", 282 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", 283 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", 284 | "dev": true 285 | }, 286 | "defer-to-connect": { 287 | "version": "1.1.3", 288 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz", 289 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==", 290 | "dev": true 291 | }, 292 | "dot-prop": { 293 | "version": "5.3.0", 294 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", 295 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", 296 | "dev": true, 297 | "requires": { 298 | "is-obj": "^2.0.0" 299 | } 300 | }, 301 | "duplexer3": { 302 | "version": "0.1.4", 303 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", 304 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", 305 | "dev": true 306 | }, 307 | "emoji-regex": { 308 | "version": "7.0.3", 309 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 310 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 311 | "dev": true 312 | }, 313 | "end-of-stream": { 314 | "version": "1.4.4", 315 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", 316 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", 317 | "dev": true, 318 | "requires": { 319 | "once": "^1.4.0" 320 | } 321 | }, 322 | "escape-goat": { 323 | "version": "2.1.1", 324 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz", 325 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==", 326 | "dev": true 327 | }, 328 | "fast-url-parser": { 329 | "version": "1.1.3", 330 | "resolved": "https://registry.npmjs.org/fast-url-parser/-/fast-url-parser-1.1.3.tgz", 331 | "integrity": "sha1-9K8+qfNNiicc9YrSs3WfQx8LMY0=", 332 | "requires": { 333 | "punycode": "^1.3.2" 334 | } 335 | }, 336 | "fill-range": { 337 | "version": "7.0.1", 338 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 339 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 340 | "dev": true, 341 | "requires": { 342 | "to-regex-range": "^5.0.1" 343 | } 344 | }, 345 | "fsevents": { 346 | "version": "2.3.2", 347 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 348 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 349 | "dev": true, 350 | "optional": true 351 | }, 352 | "get-stream": { 353 | "version": "4.1.0", 354 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", 355 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", 356 | "dev": true, 357 | "requires": { 358 | "pump": "^3.0.0" 359 | } 360 | }, 361 | "glob-parent": { 362 | "version": "5.1.2", 363 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 364 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 365 | "dev": true, 366 | "requires": { 367 | "is-glob": "^4.0.1" 368 | } 369 | }, 370 | "global-dirs": { 371 | "version": "2.1.0", 372 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.1.0.tgz", 373 | "integrity": "sha512-MG6kdOUh/xBnyo9cJFeIKkLEc1AyFq42QTU4XiX51i2NEdxLxLWXIjEjmqKeSuKR7pAZjTqUVoT2b2huxVLgYQ==", 374 | "dev": true, 375 | "requires": { 376 | "ini": "1.3.7" 377 | } 378 | }, 379 | "got": { 380 | "version": "9.6.0", 381 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz", 382 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", 383 | "dev": true, 384 | "requires": { 385 | "@sindresorhus/is": "^0.14.0", 386 | "@szmarczak/http-timer": "^1.1.2", 387 | "cacheable-request": "^6.0.0", 388 | "decompress-response": "^3.3.0", 389 | "duplexer3": "^0.1.4", 390 | "get-stream": "^4.1.0", 391 | "lowercase-keys": "^1.0.1", 392 | "mimic-response": "^1.0.1", 393 | "p-cancelable": "^1.0.0", 394 | "to-readable-stream": "^1.0.0", 395 | "url-parse-lax": "^3.0.0" 396 | } 397 | }, 398 | "graceful-fs": { 399 | "version": "4.2.6", 400 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.6.tgz", 401 | "integrity": "sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ==", 402 | "dev": true 403 | }, 404 | "has-flag": { 405 | "version": "3.0.0", 406 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 407 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 408 | "dev": true 409 | }, 410 | "has-yarn": { 411 | "version": "2.1.0", 412 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz", 413 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==", 414 | "dev": true 415 | }, 416 | "http-cache-semantics": { 417 | "version": "4.1.0", 418 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", 419 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", 420 | "dev": true 421 | }, 422 | "ignore-by-default": { 423 | "version": "1.0.1", 424 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 425 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", 426 | "dev": true 427 | }, 428 | "import-lazy": { 429 | "version": "2.1.0", 430 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", 431 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", 432 | "dev": true 433 | }, 434 | "imurmurhash": { 435 | "version": "0.1.4", 436 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 437 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 438 | "dev": true 439 | }, 440 | "ini": { 441 | "version": "1.3.7", 442 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.7.tgz", 443 | "integrity": "sha512-iKpRpXP+CrP2jyrxvg1kMUpXDyRUFDWurxbnVT1vQPx+Wz9uCYsMIqYuSBLV+PAaZG/d7kRLKRFc9oDMsH+mFQ==", 444 | "dev": true 445 | }, 446 | "is-binary-path": { 447 | "version": "2.1.0", 448 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 449 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 450 | "dev": true, 451 | "requires": { 452 | "binary-extensions": "^2.0.0" 453 | } 454 | }, 455 | "is-ci": { 456 | "version": "2.0.0", 457 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", 458 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", 459 | "dev": true, 460 | "requires": { 461 | "ci-info": "^2.0.0" 462 | } 463 | }, 464 | "is-extglob": { 465 | "version": "2.1.1", 466 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 467 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 468 | "dev": true 469 | }, 470 | "is-fullwidth-code-point": { 471 | "version": "2.0.0", 472 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 473 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 474 | "dev": true 475 | }, 476 | "is-glob": { 477 | "version": "4.0.1", 478 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 479 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 480 | "dev": true, 481 | "requires": { 482 | "is-extglob": "^2.1.1" 483 | } 484 | }, 485 | "is-installed-globally": { 486 | "version": "0.3.2", 487 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz", 488 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==", 489 | "dev": true, 490 | "requires": { 491 | "global-dirs": "^2.0.1", 492 | "is-path-inside": "^3.0.1" 493 | } 494 | }, 495 | "is-npm": { 496 | "version": "4.0.0", 497 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz", 498 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==", 499 | "dev": true 500 | }, 501 | "is-number": { 502 | "version": "7.0.0", 503 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 504 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 505 | "dev": true 506 | }, 507 | "is-obj": { 508 | "version": "2.0.0", 509 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", 510 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", 511 | "dev": true 512 | }, 513 | "is-path-inside": { 514 | "version": "3.0.3", 515 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", 516 | "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", 517 | "dev": true 518 | }, 519 | "is-typedarray": { 520 | "version": "1.0.0", 521 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 522 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", 523 | "dev": true 524 | }, 525 | "is-yarn-global": { 526 | "version": "0.3.0", 527 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", 528 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==", 529 | "dev": true 530 | }, 531 | "json-buffer": { 532 | "version": "3.0.0", 533 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", 534 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", 535 | "dev": true 536 | }, 537 | "keyv": { 538 | "version": "3.1.0", 539 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", 540 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==", 541 | "dev": true, 542 | "requires": { 543 | "json-buffer": "3.0.0" 544 | } 545 | }, 546 | "latest-version": { 547 | "version": "5.1.0", 548 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz", 549 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==", 550 | "dev": true, 551 | "requires": { 552 | "package-json": "^6.3.0" 553 | } 554 | }, 555 | "lowercase-keys": { 556 | "version": "1.0.1", 557 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", 558 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", 559 | "dev": true 560 | }, 561 | "make-dir": { 562 | "version": "3.1.0", 563 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 564 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 565 | "dev": true, 566 | "requires": { 567 | "semver": "^6.0.0" 568 | }, 569 | "dependencies": { 570 | "semver": { 571 | "version": "6.3.0", 572 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 573 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 574 | "dev": true 575 | } 576 | } 577 | }, 578 | "mimic-response": { 579 | "version": "1.0.1", 580 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", 581 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", 582 | "dev": true 583 | }, 584 | "minimatch": { 585 | "version": "3.0.4", 586 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 587 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 588 | "requires": { 589 | "brace-expansion": "^1.1.7" 590 | } 591 | }, 592 | "minimist": { 593 | "version": "1.2.5", 594 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 595 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", 596 | "dev": true 597 | }, 598 | "ms": { 599 | "version": "2.0.0", 600 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 601 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", 602 | "dev": true 603 | }, 604 | "nanobuffer": { 605 | "version": "2.0.0", 606 | "resolved": "https://registry.npmjs.org/nanobuffer/-/nanobuffer-2.0.0.tgz", 607 | "integrity": "sha512-5f7M0eixdz8SFuqseZXpK8Xkb7X70GBDrDWO+VXKiQtysiYRKtZmLhyr/I/25KCnmrsCzhAa8zJbGfGdqr7YIQ==" 608 | }, 609 | "nodemon": { 610 | "version": "2.0.12", 611 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.12.tgz", 612 | "integrity": "sha512-egCTmNZdObdBxUBw6ZNwvZ/xzk24CKRs5K6d+5zbmrMr7rOpPmfPeF6OxM3DDpaRx331CQRFEktn+wrFFfBSOA==", 613 | "dev": true, 614 | "requires": { 615 | "chokidar": "^3.2.2", 616 | "debug": "^3.2.6", 617 | "ignore-by-default": "^1.0.1", 618 | "minimatch": "^3.0.4", 619 | "pstree.remy": "^1.1.7", 620 | "semver": "^5.7.1", 621 | "supports-color": "^5.5.0", 622 | "touch": "^3.1.0", 623 | "undefsafe": "^2.0.3", 624 | "update-notifier": "^4.1.0" 625 | }, 626 | "dependencies": { 627 | "debug": { 628 | "version": "3.2.7", 629 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", 630 | "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", 631 | "dev": true, 632 | "requires": { 633 | "ms": "^2.1.1" 634 | } 635 | }, 636 | "ms": { 637 | "version": "2.1.3", 638 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 639 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 640 | "dev": true 641 | } 642 | } 643 | }, 644 | "nopt": { 645 | "version": "1.0.10", 646 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", 647 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", 648 | "dev": true, 649 | "requires": { 650 | "abbrev": "1" 651 | } 652 | }, 653 | "normalize-path": { 654 | "version": "3.0.0", 655 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 656 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 657 | "dev": true 658 | }, 659 | "normalize-url": { 660 | "version": "4.5.1", 661 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.1.tgz", 662 | "integrity": "sha512-9UZCFRHQdNrfTpGg8+1INIg93B6zE0aXMVFkw1WFwvO4SlZywU6aLg5Of0Ap/PgcbSw4LNxvMWXMeugwMCX0AA==", 663 | "dev": true 664 | }, 665 | "once": { 666 | "version": "1.4.0", 667 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 668 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 669 | "dev": true, 670 | "requires": { 671 | "wrappy": "1" 672 | } 673 | }, 674 | "p-cancelable": { 675 | "version": "1.1.0", 676 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz", 677 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==", 678 | "dev": true 679 | }, 680 | "package-json": { 681 | "version": "6.5.0", 682 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz", 683 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==", 684 | "dev": true, 685 | "requires": { 686 | "got": "^9.6.0", 687 | "registry-auth-token": "^4.0.0", 688 | "registry-url": "^5.0.0", 689 | "semver": "^6.2.0" 690 | }, 691 | "dependencies": { 692 | "semver": { 693 | "version": "6.3.0", 694 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 695 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 696 | "dev": true 697 | } 698 | } 699 | }, 700 | "path-is-inside": { 701 | "version": "1.0.2", 702 | "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", 703 | "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=" 704 | }, 705 | "picomatch": { 706 | "version": "2.3.0", 707 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", 708 | "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", 709 | "dev": true 710 | }, 711 | "prepend-http": { 712 | "version": "2.0.0", 713 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", 714 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", 715 | "dev": true 716 | }, 717 | "pstree.remy": { 718 | "version": "1.1.8", 719 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 720 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 721 | "dev": true 722 | }, 723 | "pump": { 724 | "version": "3.0.0", 725 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", 726 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", 727 | "dev": true, 728 | "requires": { 729 | "end-of-stream": "^1.1.0", 730 | "once": "^1.3.1" 731 | } 732 | }, 733 | "punycode": { 734 | "version": "1.4.1", 735 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", 736 | "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=" 737 | }, 738 | "pupa": { 739 | "version": "2.1.1", 740 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz", 741 | "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==", 742 | "dev": true, 743 | "requires": { 744 | "escape-goat": "^2.0.0" 745 | } 746 | }, 747 | "rc": { 748 | "version": "1.2.8", 749 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", 750 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", 751 | "dev": true, 752 | "requires": { 753 | "deep-extend": "^0.6.0", 754 | "ini": "~1.3.0", 755 | "minimist": "^1.2.0", 756 | "strip-json-comments": "~2.0.1" 757 | } 758 | }, 759 | "readdirp": { 760 | "version": "3.6.0", 761 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 762 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 763 | "dev": true, 764 | "requires": { 765 | "picomatch": "^2.2.1" 766 | } 767 | }, 768 | "registry-auth-token": { 769 | "version": "4.2.1", 770 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz", 771 | "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==", 772 | "dev": true, 773 | "requires": { 774 | "rc": "^1.2.8" 775 | } 776 | }, 777 | "registry-url": { 778 | "version": "5.1.0", 779 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz", 780 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==", 781 | "dev": true, 782 | "requires": { 783 | "rc": "^1.2.8" 784 | } 785 | }, 786 | "responselike": { 787 | "version": "1.0.2", 788 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", 789 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", 790 | "dev": true, 791 | "requires": { 792 | "lowercase-keys": "^1.0.0" 793 | } 794 | }, 795 | "semver": { 796 | "version": "5.7.1", 797 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 798 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 799 | "dev": true 800 | }, 801 | "semver-diff": { 802 | "version": "3.1.1", 803 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz", 804 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==", 805 | "dev": true, 806 | "requires": { 807 | "semver": "^6.3.0" 808 | }, 809 | "dependencies": { 810 | "semver": { 811 | "version": "6.3.0", 812 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 813 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 814 | "dev": true 815 | } 816 | } 817 | }, 818 | "serve-handler": { 819 | "version": "6.1.3", 820 | "resolved": "https://registry.npmjs.org/serve-handler/-/serve-handler-6.1.3.tgz", 821 | "integrity": "sha512-FosMqFBNrLyeiIDvP1zgO6YoTzFYHxLDEIavhlmQ+knB2Z7l1t+kGLHkZIDN7UVWqQAmKI3D20A6F6jo3nDd4w==", 822 | "requires": { 823 | "bytes": "3.0.0", 824 | "content-disposition": "0.5.2", 825 | "fast-url-parser": "1.1.3", 826 | "mime-types": "2.1.18", 827 | "minimatch": "3.0.4", 828 | "path-is-inside": "1.0.2", 829 | "path-to-regexp": "2.2.1", 830 | "range-parser": "1.2.0" 831 | }, 832 | "dependencies": { 833 | "bytes": { 834 | "version": "3.0.0", 835 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", 836 | "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" 837 | }, 838 | "content-disposition": { 839 | "version": "0.5.2", 840 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", 841 | "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ=" 842 | }, 843 | "mime-db": { 844 | "version": "1.33.0", 845 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.33.0.tgz", 846 | "integrity": "sha512-BHJ/EKruNIqJf/QahvxwQZXKygOQ256myeN/Ew+THcAa5q+PjyTTMMeNQC4DZw5AwfvelsUrA6B67NKMqXDbzQ==" 847 | }, 848 | "mime-types": { 849 | "version": "2.1.18", 850 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.18.tgz", 851 | "integrity": "sha512-lc/aahn+t4/SWV/qcmumYjymLsWfN3ELhpmVuUFjgsORruuZPVSwAQryq+HHGvO/SI2KVX26bx+En+zhM8g8hQ==", 852 | "requires": { 853 | "mime-db": "~1.33.0" 854 | } 855 | }, 856 | "path-to-regexp": { 857 | "version": "2.2.1", 858 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-2.2.1.tgz", 859 | "integrity": "sha512-gu9bD6Ta5bwGrrU8muHzVOBFFREpp2iRkVfhBJahwJ6p6Xw20SjT0MxLnwkjOibQmGSYhiUnf2FLe7k+jcFmGQ==" 860 | }, 861 | "range-parser": { 862 | "version": "1.2.0", 863 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz", 864 | "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4=" 865 | } 866 | } 867 | }, 868 | "signal-exit": { 869 | "version": "3.0.3", 870 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", 871 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", 872 | "dev": true 873 | }, 874 | "string-width": { 875 | "version": "4.2.2", 876 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", 877 | "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", 878 | "dev": true, 879 | "requires": { 880 | "emoji-regex": "^8.0.0", 881 | "is-fullwidth-code-point": "^3.0.0", 882 | "strip-ansi": "^6.0.0" 883 | }, 884 | "dependencies": { 885 | "ansi-regex": { 886 | "version": "5.0.0", 887 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 888 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 889 | "dev": true 890 | }, 891 | "emoji-regex": { 892 | "version": "8.0.0", 893 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 894 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 895 | "dev": true 896 | }, 897 | "is-fullwidth-code-point": { 898 | "version": "3.0.0", 899 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 900 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 901 | "dev": true 902 | }, 903 | "strip-ansi": { 904 | "version": "6.0.0", 905 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 906 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 907 | "dev": true, 908 | "requires": { 909 | "ansi-regex": "^5.0.0" 910 | } 911 | } 912 | } 913 | }, 914 | "strip-ansi": { 915 | "version": "5.2.0", 916 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 917 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 918 | "dev": true, 919 | "requires": { 920 | "ansi-regex": "^4.1.0" 921 | } 922 | }, 923 | "strip-json-comments": { 924 | "version": "2.0.1", 925 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 926 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 927 | "dev": true 928 | }, 929 | "supports-color": { 930 | "version": "5.5.0", 931 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 932 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 933 | "dev": true, 934 | "requires": { 935 | "has-flag": "^3.0.0" 936 | } 937 | }, 938 | "term-size": { 939 | "version": "2.2.1", 940 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz", 941 | "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==", 942 | "dev": true 943 | }, 944 | "to-readable-stream": { 945 | "version": "1.0.0", 946 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz", 947 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==", 948 | "dev": true 949 | }, 950 | "to-regex-range": { 951 | "version": "5.0.1", 952 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 953 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 954 | "dev": true, 955 | "requires": { 956 | "is-number": "^7.0.0" 957 | } 958 | }, 959 | "touch": { 960 | "version": "3.1.0", 961 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", 962 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", 963 | "dev": true, 964 | "requires": { 965 | "nopt": "~1.0.10" 966 | } 967 | }, 968 | "type-fest": { 969 | "version": "0.8.1", 970 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 971 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", 972 | "dev": true 973 | }, 974 | "typedarray-to-buffer": { 975 | "version": "3.1.5", 976 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", 977 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", 978 | "dev": true, 979 | "requires": { 980 | "is-typedarray": "^1.0.0" 981 | } 982 | }, 983 | "undefsafe": { 984 | "version": "2.0.3", 985 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz", 986 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==", 987 | "dev": true, 988 | "requires": { 989 | "debug": "^2.2.0" 990 | } 991 | }, 992 | "unique-string": { 993 | "version": "2.0.0", 994 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz", 995 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==", 996 | "dev": true, 997 | "requires": { 998 | "crypto-random-string": "^2.0.0" 999 | } 1000 | }, 1001 | "update-notifier": { 1002 | "version": "4.1.3", 1003 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz", 1004 | "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==", 1005 | "dev": true, 1006 | "requires": { 1007 | "boxen": "^4.2.0", 1008 | "chalk": "^3.0.0", 1009 | "configstore": "^5.0.1", 1010 | "has-yarn": "^2.1.0", 1011 | "import-lazy": "^2.1.0", 1012 | "is-ci": "^2.0.0", 1013 | "is-installed-globally": "^0.3.1", 1014 | "is-npm": "^4.0.0", 1015 | "is-yarn-global": "^0.3.0", 1016 | "latest-version": "^5.0.0", 1017 | "pupa": "^2.0.1", 1018 | "semver-diff": "^3.1.1", 1019 | "xdg-basedir": "^4.0.0" 1020 | } 1021 | }, 1022 | "url-parse-lax": { 1023 | "version": "3.0.0", 1024 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", 1025 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", 1026 | "dev": true, 1027 | "requires": { 1028 | "prepend-http": "^2.0.0" 1029 | } 1030 | }, 1031 | "widest-line": { 1032 | "version": "3.1.0", 1033 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz", 1034 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==", 1035 | "dev": true, 1036 | "requires": { 1037 | "string-width": "^4.0.0" 1038 | } 1039 | }, 1040 | "wrappy": { 1041 | "version": "1.0.2", 1042 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1043 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1044 | "dev": true 1045 | }, 1046 | "write-file-atomic": { 1047 | "version": "3.0.3", 1048 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", 1049 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", 1050 | "dev": true, 1051 | "requires": { 1052 | "imurmurhash": "^0.1.4", 1053 | "is-typedarray": "^1.0.0", 1054 | "signal-exit": "^3.0.2", 1055 | "typedarray-to-buffer": "^3.1.5" 1056 | } 1057 | }, 1058 | "xdg-basedir": { 1059 | "version": "4.0.0", 1060 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz", 1061 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==", 1062 | "dev": true 1063 | } 1064 | } 1065 | } 1066 | -------------------------------------------------------------------------------- /http2/push/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "http2", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node backend/server.js", 8 | "dev": "nodemon backend/server.js" 9 | }, 10 | "keywords": [], 11 | "type": "module", 12 | "author": "", 13 | "license": "ISC", 14 | "dependencies": { 15 | "nanobuffer": "^2.0.0", 16 | "serve-handler": "^6.1.3" 17 | }, 18 | "devDependencies": { 19 | "nodemon": "^2.0.12" 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /polling/backoff-and-retry/backend/server.js: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | import bodyParser from "body-parser"; 3 | import nanobuffer from "nanobuffer"; 4 | import morgan from "morgan"; 5 | 6 | const app = express(); 7 | const msg = new nanobuffer(50); 8 | const getMsgs = () => Array.from(msg).reverse(); 9 | 10 | msg.push({ 11 | user: "brian", 12 | text: "hi", 13 | time: Date.now(), 14 | }); 15 | 16 | app.use(morgan("dev")); 17 | app.use(bodyParser.json()); 18 | app.use(express.static("frontend")); 19 | 20 | app.get("/poll", function (req, res) { 21 | // this intenitonally causes failures 22 | res.status(Math.random() > 0.5 ? 200 : 500).json({ 23 | msg: getMsgs(), 24 | }); 25 | }); 26 | 27 | app.post("/poll", function (req, res) { 28 | const { user, text } = req.body; 29 | 30 | msg.push({ 31 | user, 32 | text, 33 | time: Date.now(), 34 | }); 35 | 36 | res.json({ 37 | status: "ok", 38 | }); 39 | }); 40 | 41 | const port = process.env.PORT || 3000; 42 | app.listen(port); 43 | console.log(`listening on http://localhost:${port}`); 44 | -------------------------------------------------------------------------------- /polling/backoff-and-retry/frontend/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Polling Chat 8 | 9 | 10 | 11 | 12 | 13 |

    Chat With Me

    14 |
    15 |
    16 |
    17 |
    18 | account_circle 19 | 20 | 21 |
    22 |
    23 | message 24 | 25 | 26 |
    27 |
    28 |
    29 | 32 |
    33 |
    34 |
    35 | 36 | 37 | 38 | 39 | 40 | 41 | -------------------------------------------------------------------------------- /polling/backoff-and-retry/frontend/polling-chat.js: -------------------------------------------------------------------------------- 1 | const chat = document.getElementById("chat"); 2 | const msgs = document.getElementById("msgs"); 3 | 4 | let allChat = []; 5 | 6 | const INTERVAL = 3000; 7 | 8 | chat.addEventListener("submit", function (e) { 9 | e.preventDefault(); 10 | postNewMsg(chat.elements.user.value, chat.elements.text.value); 11 | chat.elements.text.value = ""; 12 | }); 13 | 14 | async function postNewMsg(user, text) { 15 | const data = { 16 | user, 17 | text, 18 | }; 19 | 20 | // request options 21 | const options = { 22 | method: "POST", 23 | body: JSON.stringify(data), 24 | headers: { 25 | "Content-Type": "application/json", 26 | }, 27 | }; 28 | 29 | // send POST request 30 | await fetch("/poll", options); 31 | } 32 | 33 | async function getNewMsgs() { 34 | try { 35 | const res = await fetch("/poll"); 36 | const json = await res.json(); 37 | 38 | if (res.status >= 400) { 39 | throw new Error("request did not succeed: " + res.status); 40 | } 41 | 42 | allChat = json.msg; 43 | render(); 44 | failedTries = 0; 45 | } catch (e) { 46 | // back off 47 | failedTries++; 48 | } 49 | } 50 | 51 | function render() { 52 | const html = allChat.map(({ user, text }) => template(user, text)); 53 | msgs.innerHTML = html.join("\n"); 54 | } 55 | 56 | const template = (user, msg) => 57 | `
  • ${user}${msg}
  • `; 58 | 59 | const BACKOFF = 5000; 60 | let timeToMakeNextRequest = 0; 61 | let failedTries = 0; 62 | async function rafTimer(time) { 63 | if (timeToMakeNextRequest <= time) { 64 | await getNewMsgs(); 65 | timeToMakeNextRequest = time + INTERVAL + failedTries * BACKOFF; 66 | } 67 | requestAnimationFrame(rafTimer); 68 | } 69 | 70 | requestAnimationFrame(rafTimer); 71 | -------------------------------------------------------------------------------- /polling/backoff-and-retry/frontend/style.css: -------------------------------------------------------------------------------- 1 | .page-column { 2 | max-width: 600px; 3 | margin: 30px auto; 4 | } 5 | 6 | h1 { 7 | text-align: center; 8 | color: #26a69a; 9 | } 10 | 11 | .btn-box { 12 | text-align: center; 13 | } 14 | -------------------------------------------------------------------------------- /polling/backoff-and-retry/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "polling", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node backend/server.js", 8 | "dev": "nodemon backend/server.js" 9 | }, 10 | "keywords": [], 11 | "type": "module", 12 | "author": "", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "express": "^4.17.1", 17 | "morgan": "^1.10.0", 18 | "nanobuffer": "^2.0.0" 19 | }, 20 | "devDependencies": { 21 | "nodemon": "^2.0.12" 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /polling/exercise/backend/server.js: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | import bodyParser from "body-parser"; 3 | import nanobuffer from "nanobuffer"; 4 | import morgan from "morgan"; 5 | 6 | // set up a limited array 7 | const msg = new nanobuffer(50); 8 | const getMsgs = () => Array.from(msg).reverse(); 9 | 10 | // feel free to take out, this just seeds the server with at least one message 11 | msg.push({ 12 | user: "brian", 13 | text: "hi", 14 | time: Date.now(), 15 | }); 16 | 17 | // get express ready to run 18 | const app = express(); 19 | app.use(morgan("dev")); 20 | app.use(bodyParser.json()); 21 | app.use(express.static("frontend")); 22 | 23 | app.get("/poll", function (req, res) { 24 | // use getMsgs to get messages to send back 25 | // write code here 26 | }); 27 | 28 | app.post("/poll", function (req, res) { 29 | // add a new message to the server 30 | // write code here 31 | }); 32 | 33 | // start the server 34 | const port = process.env.PORT || 3000; 35 | app.listen(port); 36 | console.log(`listening on http://localhost:${port}`); 37 | -------------------------------------------------------------------------------- /polling/exercise/frontend/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Polling Chat 8 | 9 | 10 | 11 | 12 | 13 | 14 |

    Chat With Me

    15 | 16 |
    17 | 18 |
    19 |
    20 | 21 |
    22 | account_circle 23 | 24 | 25 |
    26 | 27 |
    28 | message 29 | 30 | 31 |
    32 |
    33 | 34 |
    35 | 38 |
    39 | 40 |
    41 |
    42 | 43 | 46 | 47 | 48 | 49 | 50 | 51 | -------------------------------------------------------------------------------- /polling/exercise/frontend/polling-chat.js: -------------------------------------------------------------------------------- 1 | const chat = document.getElementById("chat"); 2 | const msgs = document.getElementById("msgs"); 3 | 4 | // let's store all current messages here 5 | let allChat = []; 6 | 7 | // the interval to poll at in milliseconds 8 | const INTERVAL = 3000; 9 | 10 | // a submit listener on the form in the HTML 11 | chat.addEventListener("submit", function (e) { 12 | e.preventDefault(); 13 | postNewMsg(chat.elements.user.value, chat.elements.text.value); 14 | chat.elements.text.value = ""; 15 | }); 16 | 17 | async function postNewMsg(user, text) { 18 | // post to /poll a new message 19 | // write code here 20 | } 21 | 22 | async function getNewMsgs() { 23 | // poll the server 24 | // write code here 25 | } 26 | 27 | function render() { 28 | // as long as allChat is holding all current messages, this will render them 29 | // into the ui. yes, it's inefficent. yes, it's fine for this example 30 | const html = allChat.map(({ user, text, time, id }) => 31 | template(user, text, time, id) 32 | ); 33 | msgs.innerHTML = html.join("\n"); 34 | } 35 | 36 | // given a user and a msg, it returns an HTML string to render to the UI 37 | const template = (user, msg) => 38 | `
  • ${user}${msg}
  • `; 39 | 40 | // make the first request 41 | getNewMsgs(); 42 | -------------------------------------------------------------------------------- /polling/exercise/frontend/style.css: -------------------------------------------------------------------------------- 1 | .page-column { 2 | max-width: 600px; 3 | margin: 30px auto; 4 | } 5 | 6 | h1 { 7 | text-align: center; 8 | color: #26a69a; 9 | } 10 | 11 | .btn-box { 12 | text-align: center; 13 | } 14 | -------------------------------------------------------------------------------- /polling/exercise/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "polling", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@sindresorhus/is": { 8 | "version": "0.14.0", 9 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz", 10 | "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==", 11 | "dev": true 12 | }, 13 | "@szmarczak/http-timer": { 14 | "version": "1.1.2", 15 | "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz", 16 | "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==", 17 | "dev": true, 18 | "requires": { 19 | "defer-to-connect": "^1.0.1" 20 | } 21 | }, 22 | "abbrev": { 23 | "version": "1.1.1", 24 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 25 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 26 | "dev": true 27 | }, 28 | "accepts": { 29 | "version": "1.3.7", 30 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 31 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 32 | "requires": { 33 | "mime-types": "~2.1.24", 34 | "negotiator": "0.6.2" 35 | } 36 | }, 37 | "ansi-align": { 38 | "version": "3.0.0", 39 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz", 40 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==", 41 | "dev": true, 42 | "requires": { 43 | "string-width": "^3.0.0" 44 | }, 45 | "dependencies": { 46 | "string-width": { 47 | "version": "3.1.0", 48 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 49 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 50 | "dev": true, 51 | "requires": { 52 | "emoji-regex": "^7.0.1", 53 | "is-fullwidth-code-point": "^2.0.0", 54 | "strip-ansi": "^5.1.0" 55 | } 56 | } 57 | } 58 | }, 59 | "ansi-regex": { 60 | "version": "4.1.0", 61 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 62 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 63 | "dev": true 64 | }, 65 | "ansi-styles": { 66 | "version": "4.3.0", 67 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 68 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 69 | "dev": true, 70 | "requires": { 71 | "color-convert": "^2.0.1" 72 | } 73 | }, 74 | "anymatch": { 75 | "version": "3.1.2", 76 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 77 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 78 | "dev": true, 79 | "requires": { 80 | "normalize-path": "^3.0.0", 81 | "picomatch": "^2.0.4" 82 | } 83 | }, 84 | "array-flatten": { 85 | "version": "1.1.1", 86 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 87 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 88 | }, 89 | "balanced-match": { 90 | "version": "1.0.2", 91 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 92 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 93 | "dev": true 94 | }, 95 | "basic-auth": { 96 | "version": "2.0.1", 97 | "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz", 98 | "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==", 99 | "requires": { 100 | "safe-buffer": "5.1.2" 101 | } 102 | }, 103 | "binary-extensions": { 104 | "version": "2.2.0", 105 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 106 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 107 | "dev": true 108 | }, 109 | "body-parser": { 110 | "version": "1.19.0", 111 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", 112 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", 113 | "requires": { 114 | "bytes": "3.1.0", 115 | "content-type": "~1.0.4", 116 | "debug": "2.6.9", 117 | "depd": "~1.1.2", 118 | "http-errors": "1.7.2", 119 | "iconv-lite": "0.4.24", 120 | "on-finished": "~2.3.0", 121 | "qs": "6.7.0", 122 | "raw-body": "2.4.0", 123 | "type-is": "~1.6.17" 124 | } 125 | }, 126 | "boxen": { 127 | "version": "4.2.0", 128 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz", 129 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==", 130 | "dev": true, 131 | "requires": { 132 | "ansi-align": "^3.0.0", 133 | "camelcase": "^5.3.1", 134 | "chalk": "^3.0.0", 135 | "cli-boxes": "^2.2.0", 136 | "string-width": "^4.1.0", 137 | "term-size": "^2.1.0", 138 | "type-fest": "^0.8.1", 139 | "widest-line": "^3.1.0" 140 | } 141 | }, 142 | "brace-expansion": { 143 | "version": "1.1.11", 144 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 145 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 146 | "dev": true, 147 | "requires": { 148 | "balanced-match": "^1.0.0", 149 | "concat-map": "0.0.1" 150 | } 151 | }, 152 | "braces": { 153 | "version": "3.0.2", 154 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 155 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 156 | "dev": true, 157 | "requires": { 158 | "fill-range": "^7.0.1" 159 | } 160 | }, 161 | "bytes": { 162 | "version": "3.1.0", 163 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", 164 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" 165 | }, 166 | "cacheable-request": { 167 | "version": "6.1.0", 168 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz", 169 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", 170 | "dev": true, 171 | "requires": { 172 | "clone-response": "^1.0.2", 173 | "get-stream": "^5.1.0", 174 | "http-cache-semantics": "^4.0.0", 175 | "keyv": "^3.0.0", 176 | "lowercase-keys": "^2.0.0", 177 | "normalize-url": "^4.1.0", 178 | "responselike": "^1.0.2" 179 | }, 180 | "dependencies": { 181 | "get-stream": { 182 | "version": "5.2.0", 183 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", 184 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", 185 | "dev": true, 186 | "requires": { 187 | "pump": "^3.0.0" 188 | } 189 | }, 190 | "lowercase-keys": { 191 | "version": "2.0.0", 192 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", 193 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", 194 | "dev": true 195 | } 196 | } 197 | }, 198 | "camelcase": { 199 | "version": "5.3.1", 200 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 201 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 202 | "dev": true 203 | }, 204 | "chalk": { 205 | "version": "3.0.0", 206 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", 207 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", 208 | "dev": true, 209 | "requires": { 210 | "ansi-styles": "^4.1.0", 211 | "supports-color": "^7.1.0" 212 | }, 213 | "dependencies": { 214 | "has-flag": { 215 | "version": "4.0.0", 216 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 217 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 218 | "dev": true 219 | }, 220 | "supports-color": { 221 | "version": "7.2.0", 222 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 223 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 224 | "dev": true, 225 | "requires": { 226 | "has-flag": "^4.0.0" 227 | } 228 | } 229 | } 230 | }, 231 | "chokidar": { 232 | "version": "3.5.2", 233 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.2.tgz", 234 | "integrity": "sha512-ekGhOnNVPgT77r4K/U3GDhu+FQ2S8TnK/s2KbIGXi0SZWuwkZ2QNyfWdZW+TVfn84DpEP7rLeCt2UI6bJ8GwbQ==", 235 | "dev": true, 236 | "requires": { 237 | "anymatch": "~3.1.2", 238 | "braces": "~3.0.2", 239 | "fsevents": "~2.3.2", 240 | "glob-parent": "~5.1.2", 241 | "is-binary-path": "~2.1.0", 242 | "is-glob": "~4.0.1", 243 | "normalize-path": "~3.0.0", 244 | "readdirp": "~3.6.0" 245 | } 246 | }, 247 | "ci-info": { 248 | "version": "2.0.0", 249 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz", 250 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", 251 | "dev": true 252 | }, 253 | "cli-boxes": { 254 | "version": "2.2.1", 255 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz", 256 | "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==", 257 | "dev": true 258 | }, 259 | "clone-response": { 260 | "version": "1.0.2", 261 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", 262 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", 263 | "dev": true, 264 | "requires": { 265 | "mimic-response": "^1.0.0" 266 | } 267 | }, 268 | "color-convert": { 269 | "version": "2.0.1", 270 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 271 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 272 | "dev": true, 273 | "requires": { 274 | "color-name": "~1.1.4" 275 | } 276 | }, 277 | "color-name": { 278 | "version": "1.1.4", 279 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 280 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 281 | "dev": true 282 | }, 283 | "concat-map": { 284 | "version": "0.0.1", 285 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 286 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 287 | "dev": true 288 | }, 289 | "configstore": { 290 | "version": "5.0.1", 291 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz", 292 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==", 293 | "dev": true, 294 | "requires": { 295 | "dot-prop": "^5.2.0", 296 | "graceful-fs": "^4.1.2", 297 | "make-dir": "^3.0.0", 298 | "unique-string": "^2.0.0", 299 | "write-file-atomic": "^3.0.0", 300 | "xdg-basedir": "^4.0.0" 301 | } 302 | }, 303 | "content-disposition": { 304 | "version": "0.5.3", 305 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", 306 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", 307 | "requires": { 308 | "safe-buffer": "5.1.2" 309 | } 310 | }, 311 | "content-type": { 312 | "version": "1.0.4", 313 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 314 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 315 | }, 316 | "cookie": { 317 | "version": "0.4.0", 318 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 319 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 320 | }, 321 | "cookie-signature": { 322 | "version": "1.0.6", 323 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 324 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 325 | }, 326 | "crypto-random-string": { 327 | "version": "2.0.0", 328 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", 329 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==", 330 | "dev": true 331 | }, 332 | "debug": { 333 | "version": "2.6.9", 334 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 335 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 336 | "requires": { 337 | "ms": "2.0.0" 338 | } 339 | }, 340 | "decompress-response": { 341 | "version": "3.3.0", 342 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", 343 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", 344 | "dev": true, 345 | "requires": { 346 | "mimic-response": "^1.0.0" 347 | } 348 | }, 349 | "deep-extend": { 350 | "version": "0.6.0", 351 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", 352 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", 353 | "dev": true 354 | }, 355 | "defer-to-connect": { 356 | "version": "1.1.3", 357 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz", 358 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==", 359 | "dev": true 360 | }, 361 | "depd": { 362 | "version": "1.1.2", 363 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 364 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 365 | }, 366 | "destroy": { 367 | "version": "1.0.4", 368 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 369 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 370 | }, 371 | "dot-prop": { 372 | "version": "5.3.0", 373 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", 374 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", 375 | "dev": true, 376 | "requires": { 377 | "is-obj": "^2.0.0" 378 | } 379 | }, 380 | "duplexer3": { 381 | "version": "0.1.4", 382 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", 383 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", 384 | "dev": true 385 | }, 386 | "ee-first": { 387 | "version": "1.1.1", 388 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 389 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 390 | }, 391 | "emoji-regex": { 392 | "version": "7.0.3", 393 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 394 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 395 | "dev": true 396 | }, 397 | "encodeurl": { 398 | "version": "1.0.2", 399 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 400 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 401 | }, 402 | "end-of-stream": { 403 | "version": "1.4.4", 404 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", 405 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", 406 | "dev": true, 407 | "requires": { 408 | "once": "^1.4.0" 409 | } 410 | }, 411 | "escape-goat": { 412 | "version": "2.1.1", 413 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz", 414 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==", 415 | "dev": true 416 | }, 417 | "escape-html": { 418 | "version": "1.0.3", 419 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 420 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 421 | }, 422 | "etag": { 423 | "version": "1.8.1", 424 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 425 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 426 | }, 427 | "express": { 428 | "version": "4.17.1", 429 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", 430 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", 431 | "requires": { 432 | "accepts": "~1.3.7", 433 | "array-flatten": "1.1.1", 434 | "body-parser": "1.19.0", 435 | "content-disposition": "0.5.3", 436 | "content-type": "~1.0.4", 437 | "cookie": "0.4.0", 438 | "cookie-signature": "1.0.6", 439 | "debug": "2.6.9", 440 | "depd": "~1.1.2", 441 | "encodeurl": "~1.0.2", 442 | "escape-html": "~1.0.3", 443 | "etag": "~1.8.1", 444 | "finalhandler": "~1.1.2", 445 | "fresh": "0.5.2", 446 | "merge-descriptors": "1.0.1", 447 | "methods": "~1.1.2", 448 | "on-finished": "~2.3.0", 449 | "parseurl": "~1.3.3", 450 | "path-to-regexp": "0.1.7", 451 | "proxy-addr": "~2.0.5", 452 | "qs": "6.7.0", 453 | "range-parser": "~1.2.1", 454 | "safe-buffer": "5.1.2", 455 | "send": "0.17.1", 456 | "serve-static": "1.14.1", 457 | "setprototypeof": "1.1.1", 458 | "statuses": "~1.5.0", 459 | "type-is": "~1.6.18", 460 | "utils-merge": "1.0.1", 461 | "vary": "~1.1.2" 462 | } 463 | }, 464 | "fill-range": { 465 | "version": "7.0.1", 466 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 467 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 468 | "dev": true, 469 | "requires": { 470 | "to-regex-range": "^5.0.1" 471 | } 472 | }, 473 | "finalhandler": { 474 | "version": "1.1.2", 475 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", 476 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", 477 | "requires": { 478 | "debug": "2.6.9", 479 | "encodeurl": "~1.0.2", 480 | "escape-html": "~1.0.3", 481 | "on-finished": "~2.3.0", 482 | "parseurl": "~1.3.3", 483 | "statuses": "~1.5.0", 484 | "unpipe": "~1.0.0" 485 | } 486 | }, 487 | "forwarded": { 488 | "version": "0.2.0", 489 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 490 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==" 491 | }, 492 | "fresh": { 493 | "version": "0.5.2", 494 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 495 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 496 | }, 497 | "fsevents": { 498 | "version": "2.3.2", 499 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 500 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 501 | "dev": true, 502 | "optional": true 503 | }, 504 | "get-stream": { 505 | "version": "4.1.0", 506 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", 507 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", 508 | "dev": true, 509 | "requires": { 510 | "pump": "^3.0.0" 511 | } 512 | }, 513 | "glob-parent": { 514 | "version": "5.1.2", 515 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 516 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 517 | "dev": true, 518 | "requires": { 519 | "is-glob": "^4.0.1" 520 | } 521 | }, 522 | "global-dirs": { 523 | "version": "2.1.0", 524 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.1.0.tgz", 525 | "integrity": "sha512-MG6kdOUh/xBnyo9cJFeIKkLEc1AyFq42QTU4XiX51i2NEdxLxLWXIjEjmqKeSuKR7pAZjTqUVoT2b2huxVLgYQ==", 526 | "dev": true, 527 | "requires": { 528 | "ini": "1.3.7" 529 | } 530 | }, 531 | "got": { 532 | "version": "9.6.0", 533 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz", 534 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", 535 | "dev": true, 536 | "requires": { 537 | "@sindresorhus/is": "^0.14.0", 538 | "@szmarczak/http-timer": "^1.1.2", 539 | "cacheable-request": "^6.0.0", 540 | "decompress-response": "^3.3.0", 541 | "duplexer3": "^0.1.4", 542 | "get-stream": "^4.1.0", 543 | "lowercase-keys": "^1.0.1", 544 | "mimic-response": "^1.0.1", 545 | "p-cancelable": "^1.0.0", 546 | "to-readable-stream": "^1.0.0", 547 | "url-parse-lax": "^3.0.0" 548 | } 549 | }, 550 | "graceful-fs": { 551 | "version": "4.2.6", 552 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.6.tgz", 553 | "integrity": "sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ==", 554 | "dev": true 555 | }, 556 | "has-flag": { 557 | "version": "3.0.0", 558 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 559 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 560 | "dev": true 561 | }, 562 | "has-yarn": { 563 | "version": "2.1.0", 564 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz", 565 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==", 566 | "dev": true 567 | }, 568 | "http-cache-semantics": { 569 | "version": "4.1.0", 570 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", 571 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", 572 | "dev": true 573 | }, 574 | "http-errors": { 575 | "version": "1.7.2", 576 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 577 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 578 | "requires": { 579 | "depd": "~1.1.2", 580 | "inherits": "2.0.3", 581 | "setprototypeof": "1.1.1", 582 | "statuses": ">= 1.5.0 < 2", 583 | "toidentifier": "1.0.0" 584 | } 585 | }, 586 | "iconv-lite": { 587 | "version": "0.4.24", 588 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 589 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 590 | "requires": { 591 | "safer-buffer": ">= 2.1.2 < 3" 592 | } 593 | }, 594 | "ignore-by-default": { 595 | "version": "1.0.1", 596 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 597 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", 598 | "dev": true 599 | }, 600 | "import-lazy": { 601 | "version": "2.1.0", 602 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", 603 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", 604 | "dev": true 605 | }, 606 | "imurmurhash": { 607 | "version": "0.1.4", 608 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 609 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 610 | "dev": true 611 | }, 612 | "inherits": { 613 | "version": "2.0.3", 614 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 615 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 616 | }, 617 | "ini": { 618 | "version": "1.3.7", 619 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.7.tgz", 620 | "integrity": "sha512-iKpRpXP+CrP2jyrxvg1kMUpXDyRUFDWurxbnVT1vQPx+Wz9uCYsMIqYuSBLV+PAaZG/d7kRLKRFc9oDMsH+mFQ==", 621 | "dev": true 622 | }, 623 | "ipaddr.js": { 624 | "version": "1.9.1", 625 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 626 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" 627 | }, 628 | "is-binary-path": { 629 | "version": "2.1.0", 630 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 631 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 632 | "dev": true, 633 | "requires": { 634 | "binary-extensions": "^2.0.0" 635 | } 636 | }, 637 | "is-ci": { 638 | "version": "2.0.0", 639 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", 640 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", 641 | "dev": true, 642 | "requires": { 643 | "ci-info": "^2.0.0" 644 | } 645 | }, 646 | "is-extglob": { 647 | "version": "2.1.1", 648 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 649 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 650 | "dev": true 651 | }, 652 | "is-fullwidth-code-point": { 653 | "version": "2.0.0", 654 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 655 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 656 | "dev": true 657 | }, 658 | "is-glob": { 659 | "version": "4.0.1", 660 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 661 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 662 | "dev": true, 663 | "requires": { 664 | "is-extglob": "^2.1.1" 665 | } 666 | }, 667 | "is-installed-globally": { 668 | "version": "0.3.2", 669 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz", 670 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==", 671 | "dev": true, 672 | "requires": { 673 | "global-dirs": "^2.0.1", 674 | "is-path-inside": "^3.0.1" 675 | } 676 | }, 677 | "is-npm": { 678 | "version": "4.0.0", 679 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz", 680 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==", 681 | "dev": true 682 | }, 683 | "is-number": { 684 | "version": "7.0.0", 685 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 686 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 687 | "dev": true 688 | }, 689 | "is-obj": { 690 | "version": "2.0.0", 691 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", 692 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", 693 | "dev": true 694 | }, 695 | "is-path-inside": { 696 | "version": "3.0.3", 697 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", 698 | "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", 699 | "dev": true 700 | }, 701 | "is-typedarray": { 702 | "version": "1.0.0", 703 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 704 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", 705 | "dev": true 706 | }, 707 | "is-yarn-global": { 708 | "version": "0.3.0", 709 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", 710 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==", 711 | "dev": true 712 | }, 713 | "json-buffer": { 714 | "version": "3.0.0", 715 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", 716 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", 717 | "dev": true 718 | }, 719 | "keyv": { 720 | "version": "3.1.0", 721 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", 722 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==", 723 | "dev": true, 724 | "requires": { 725 | "json-buffer": "3.0.0" 726 | } 727 | }, 728 | "latest-version": { 729 | "version": "5.1.0", 730 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz", 731 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==", 732 | "dev": true, 733 | "requires": { 734 | "package-json": "^6.3.0" 735 | } 736 | }, 737 | "lowercase-keys": { 738 | "version": "1.0.1", 739 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", 740 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", 741 | "dev": true 742 | }, 743 | "make-dir": { 744 | "version": "3.1.0", 745 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 746 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 747 | "dev": true, 748 | "requires": { 749 | "semver": "^6.0.0" 750 | }, 751 | "dependencies": { 752 | "semver": { 753 | "version": "6.3.0", 754 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 755 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 756 | "dev": true 757 | } 758 | } 759 | }, 760 | "media-typer": { 761 | "version": "0.3.0", 762 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 763 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 764 | }, 765 | "merge-descriptors": { 766 | "version": "1.0.1", 767 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 768 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 769 | }, 770 | "methods": { 771 | "version": "1.1.2", 772 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 773 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 774 | }, 775 | "mime": { 776 | "version": "1.6.0", 777 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 778 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 779 | }, 780 | "mime-db": { 781 | "version": "1.48.0", 782 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.48.0.tgz", 783 | "integrity": "sha512-FM3QwxV+TnZYQ2aRqhlKBMHxk10lTbMt3bBkMAp54ddrNeVSfcQYOOKuGuy3Ddrm38I04If834fOUSq1yzslJQ==" 784 | }, 785 | "mime-types": { 786 | "version": "2.1.31", 787 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.31.tgz", 788 | "integrity": "sha512-XGZnNzm3QvgKxa8dpzyhFTHmpP3l5YNusmne07VUOXxou9CqUqYa/HBy124RqtVh/O2pECas/MOcsDgpilPOPg==", 789 | "requires": { 790 | "mime-db": "1.48.0" 791 | } 792 | }, 793 | "mimic-response": { 794 | "version": "1.0.1", 795 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", 796 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", 797 | "dev": true 798 | }, 799 | "minimatch": { 800 | "version": "3.0.4", 801 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 802 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 803 | "dev": true, 804 | "requires": { 805 | "brace-expansion": "^1.1.7" 806 | } 807 | }, 808 | "minimist": { 809 | "version": "1.2.5", 810 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 811 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", 812 | "dev": true 813 | }, 814 | "morgan": { 815 | "version": "1.10.0", 816 | "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.10.0.tgz", 817 | "integrity": "sha512-AbegBVI4sh6El+1gNwvD5YIck7nSA36weD7xvIxG4in80j/UoK8AEGaWnnz8v1GxonMCltmlNs5ZKbGvl9b1XQ==", 818 | "requires": { 819 | "basic-auth": "~2.0.1", 820 | "debug": "2.6.9", 821 | "depd": "~2.0.0", 822 | "on-finished": "~2.3.0", 823 | "on-headers": "~1.0.2" 824 | }, 825 | "dependencies": { 826 | "depd": { 827 | "version": "2.0.0", 828 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 829 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==" 830 | } 831 | } 832 | }, 833 | "ms": { 834 | "version": "2.0.0", 835 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 836 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 837 | }, 838 | "nanobuffer": { 839 | "version": "2.0.0", 840 | "resolved": "https://registry.npmjs.org/nanobuffer/-/nanobuffer-2.0.0.tgz", 841 | "integrity": "sha512-5f7M0eixdz8SFuqseZXpK8Xkb7X70GBDrDWO+VXKiQtysiYRKtZmLhyr/I/25KCnmrsCzhAa8zJbGfGdqr7YIQ==" 842 | }, 843 | "negotiator": { 844 | "version": "0.6.2", 845 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 846 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 847 | }, 848 | "nodemon": { 849 | "version": "2.0.12", 850 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.12.tgz", 851 | "integrity": "sha512-egCTmNZdObdBxUBw6ZNwvZ/xzk24CKRs5K6d+5zbmrMr7rOpPmfPeF6OxM3DDpaRx331CQRFEktn+wrFFfBSOA==", 852 | "dev": true, 853 | "requires": { 854 | "chokidar": "^3.2.2", 855 | "debug": "^3.2.6", 856 | "ignore-by-default": "^1.0.1", 857 | "minimatch": "^3.0.4", 858 | "pstree.remy": "^1.1.7", 859 | "semver": "^5.7.1", 860 | "supports-color": "^5.5.0", 861 | "touch": "^3.1.0", 862 | "undefsafe": "^2.0.3", 863 | "update-notifier": "^4.1.0" 864 | }, 865 | "dependencies": { 866 | "debug": { 867 | "version": "3.2.7", 868 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", 869 | "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", 870 | "dev": true, 871 | "requires": { 872 | "ms": "^2.1.1" 873 | } 874 | }, 875 | "ms": { 876 | "version": "2.1.3", 877 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 878 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 879 | "dev": true 880 | } 881 | } 882 | }, 883 | "nopt": { 884 | "version": "1.0.10", 885 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", 886 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", 887 | "dev": true, 888 | "requires": { 889 | "abbrev": "1" 890 | } 891 | }, 892 | "normalize-path": { 893 | "version": "3.0.0", 894 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 895 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 896 | "dev": true 897 | }, 898 | "normalize-url": { 899 | "version": "4.5.1", 900 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.1.tgz", 901 | "integrity": "sha512-9UZCFRHQdNrfTpGg8+1INIg93B6zE0aXMVFkw1WFwvO4SlZywU6aLg5Of0Ap/PgcbSw4LNxvMWXMeugwMCX0AA==", 902 | "dev": true 903 | }, 904 | "on-finished": { 905 | "version": "2.3.0", 906 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 907 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 908 | "requires": { 909 | "ee-first": "1.1.1" 910 | } 911 | }, 912 | "on-headers": { 913 | "version": "1.0.2", 914 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", 915 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==" 916 | }, 917 | "once": { 918 | "version": "1.4.0", 919 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 920 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 921 | "dev": true, 922 | "requires": { 923 | "wrappy": "1" 924 | } 925 | }, 926 | "p-cancelable": { 927 | "version": "1.1.0", 928 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz", 929 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==", 930 | "dev": true 931 | }, 932 | "package-json": { 933 | "version": "6.5.0", 934 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz", 935 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==", 936 | "dev": true, 937 | "requires": { 938 | "got": "^9.6.0", 939 | "registry-auth-token": "^4.0.0", 940 | "registry-url": "^5.0.0", 941 | "semver": "^6.2.0" 942 | }, 943 | "dependencies": { 944 | "semver": { 945 | "version": "6.3.0", 946 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 947 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 948 | "dev": true 949 | } 950 | } 951 | }, 952 | "parseurl": { 953 | "version": "1.3.3", 954 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 955 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 956 | }, 957 | "path-to-regexp": { 958 | "version": "0.1.7", 959 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 960 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 961 | }, 962 | "picomatch": { 963 | "version": "2.3.0", 964 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", 965 | "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", 966 | "dev": true 967 | }, 968 | "prepend-http": { 969 | "version": "2.0.0", 970 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", 971 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", 972 | "dev": true 973 | }, 974 | "proxy-addr": { 975 | "version": "2.0.7", 976 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 977 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 978 | "requires": { 979 | "forwarded": "0.2.0", 980 | "ipaddr.js": "1.9.1" 981 | } 982 | }, 983 | "pstree.remy": { 984 | "version": "1.1.8", 985 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 986 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 987 | "dev": true 988 | }, 989 | "pump": { 990 | "version": "3.0.0", 991 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", 992 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", 993 | "dev": true, 994 | "requires": { 995 | "end-of-stream": "^1.1.0", 996 | "once": "^1.3.1" 997 | } 998 | }, 999 | "pupa": { 1000 | "version": "2.1.1", 1001 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz", 1002 | "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==", 1003 | "dev": true, 1004 | "requires": { 1005 | "escape-goat": "^2.0.0" 1006 | } 1007 | }, 1008 | "qs": { 1009 | "version": "6.7.0", 1010 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", 1011 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" 1012 | }, 1013 | "range-parser": { 1014 | "version": "1.2.1", 1015 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 1016 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 1017 | }, 1018 | "raw-body": { 1019 | "version": "2.4.0", 1020 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", 1021 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", 1022 | "requires": { 1023 | "bytes": "3.1.0", 1024 | "http-errors": "1.7.2", 1025 | "iconv-lite": "0.4.24", 1026 | "unpipe": "1.0.0" 1027 | } 1028 | }, 1029 | "rc": { 1030 | "version": "1.2.8", 1031 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", 1032 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", 1033 | "dev": true, 1034 | "requires": { 1035 | "deep-extend": "^0.6.0", 1036 | "ini": "~1.3.0", 1037 | "minimist": "^1.2.0", 1038 | "strip-json-comments": "~2.0.1" 1039 | } 1040 | }, 1041 | "readdirp": { 1042 | "version": "3.6.0", 1043 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 1044 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 1045 | "dev": true, 1046 | "requires": { 1047 | "picomatch": "^2.2.1" 1048 | } 1049 | }, 1050 | "registry-auth-token": { 1051 | "version": "4.2.1", 1052 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz", 1053 | "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==", 1054 | "dev": true, 1055 | "requires": { 1056 | "rc": "^1.2.8" 1057 | } 1058 | }, 1059 | "registry-url": { 1060 | "version": "5.1.0", 1061 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz", 1062 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==", 1063 | "dev": true, 1064 | "requires": { 1065 | "rc": "^1.2.8" 1066 | } 1067 | }, 1068 | "responselike": { 1069 | "version": "1.0.2", 1070 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", 1071 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", 1072 | "dev": true, 1073 | "requires": { 1074 | "lowercase-keys": "^1.0.0" 1075 | } 1076 | }, 1077 | "safe-buffer": { 1078 | "version": "5.1.2", 1079 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1080 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 1081 | }, 1082 | "safer-buffer": { 1083 | "version": "2.1.2", 1084 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1085 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 1086 | }, 1087 | "semver": { 1088 | "version": "5.7.1", 1089 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1090 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1091 | "dev": true 1092 | }, 1093 | "semver-diff": { 1094 | "version": "3.1.1", 1095 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz", 1096 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==", 1097 | "dev": true, 1098 | "requires": { 1099 | "semver": "^6.3.0" 1100 | }, 1101 | "dependencies": { 1102 | "semver": { 1103 | "version": "6.3.0", 1104 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1105 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1106 | "dev": true 1107 | } 1108 | } 1109 | }, 1110 | "send": { 1111 | "version": "0.17.1", 1112 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", 1113 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", 1114 | "requires": { 1115 | "debug": "2.6.9", 1116 | "depd": "~1.1.2", 1117 | "destroy": "~1.0.4", 1118 | "encodeurl": "~1.0.2", 1119 | "escape-html": "~1.0.3", 1120 | "etag": "~1.8.1", 1121 | "fresh": "0.5.2", 1122 | "http-errors": "~1.7.2", 1123 | "mime": "1.6.0", 1124 | "ms": "2.1.1", 1125 | "on-finished": "~2.3.0", 1126 | "range-parser": "~1.2.1", 1127 | "statuses": "~1.5.0" 1128 | }, 1129 | "dependencies": { 1130 | "ms": { 1131 | "version": "2.1.1", 1132 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 1133 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 1134 | } 1135 | } 1136 | }, 1137 | "serve-static": { 1138 | "version": "1.14.1", 1139 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", 1140 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", 1141 | "requires": { 1142 | "encodeurl": "~1.0.2", 1143 | "escape-html": "~1.0.3", 1144 | "parseurl": "~1.3.3", 1145 | "send": "0.17.1" 1146 | } 1147 | }, 1148 | "setprototypeof": { 1149 | "version": "1.1.1", 1150 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", 1151 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" 1152 | }, 1153 | "signal-exit": { 1154 | "version": "3.0.3", 1155 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", 1156 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", 1157 | "dev": true 1158 | }, 1159 | "statuses": { 1160 | "version": "1.5.0", 1161 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", 1162 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" 1163 | }, 1164 | "string-width": { 1165 | "version": "4.2.2", 1166 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", 1167 | "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", 1168 | "dev": true, 1169 | "requires": { 1170 | "emoji-regex": "^8.0.0", 1171 | "is-fullwidth-code-point": "^3.0.0", 1172 | "strip-ansi": "^6.0.0" 1173 | }, 1174 | "dependencies": { 1175 | "ansi-regex": { 1176 | "version": "5.0.0", 1177 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 1178 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 1179 | "dev": true 1180 | }, 1181 | "emoji-regex": { 1182 | "version": "8.0.0", 1183 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 1184 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 1185 | "dev": true 1186 | }, 1187 | "is-fullwidth-code-point": { 1188 | "version": "3.0.0", 1189 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 1190 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 1191 | "dev": true 1192 | }, 1193 | "strip-ansi": { 1194 | "version": "6.0.0", 1195 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 1196 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 1197 | "dev": true, 1198 | "requires": { 1199 | "ansi-regex": "^5.0.0" 1200 | } 1201 | } 1202 | } 1203 | }, 1204 | "strip-ansi": { 1205 | "version": "5.2.0", 1206 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 1207 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 1208 | "dev": true, 1209 | "requires": { 1210 | "ansi-regex": "^4.1.0" 1211 | } 1212 | }, 1213 | "strip-json-comments": { 1214 | "version": "2.0.1", 1215 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 1216 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 1217 | "dev": true 1218 | }, 1219 | "supports-color": { 1220 | "version": "5.5.0", 1221 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1222 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1223 | "dev": true, 1224 | "requires": { 1225 | "has-flag": "^3.0.0" 1226 | } 1227 | }, 1228 | "term-size": { 1229 | "version": "2.2.1", 1230 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz", 1231 | "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==", 1232 | "dev": true 1233 | }, 1234 | "to-readable-stream": { 1235 | "version": "1.0.0", 1236 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz", 1237 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==", 1238 | "dev": true 1239 | }, 1240 | "to-regex-range": { 1241 | "version": "5.0.1", 1242 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1243 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1244 | "dev": true, 1245 | "requires": { 1246 | "is-number": "^7.0.0" 1247 | } 1248 | }, 1249 | "toidentifier": { 1250 | "version": "1.0.0", 1251 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", 1252 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" 1253 | }, 1254 | "touch": { 1255 | "version": "3.1.0", 1256 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", 1257 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", 1258 | "dev": true, 1259 | "requires": { 1260 | "nopt": "~1.0.10" 1261 | } 1262 | }, 1263 | "type-fest": { 1264 | "version": "0.8.1", 1265 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 1266 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", 1267 | "dev": true 1268 | }, 1269 | "type-is": { 1270 | "version": "1.6.18", 1271 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 1272 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 1273 | "requires": { 1274 | "media-typer": "0.3.0", 1275 | "mime-types": "~2.1.24" 1276 | } 1277 | }, 1278 | "typedarray-to-buffer": { 1279 | "version": "3.1.5", 1280 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", 1281 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", 1282 | "dev": true, 1283 | "requires": { 1284 | "is-typedarray": "^1.0.0" 1285 | } 1286 | }, 1287 | "undefsafe": { 1288 | "version": "2.0.3", 1289 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz", 1290 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==", 1291 | "dev": true, 1292 | "requires": { 1293 | "debug": "^2.2.0" 1294 | } 1295 | }, 1296 | "unique-string": { 1297 | "version": "2.0.0", 1298 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz", 1299 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==", 1300 | "dev": true, 1301 | "requires": { 1302 | "crypto-random-string": "^2.0.0" 1303 | } 1304 | }, 1305 | "unpipe": { 1306 | "version": "1.0.0", 1307 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1308 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 1309 | }, 1310 | "update-notifier": { 1311 | "version": "4.1.3", 1312 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz", 1313 | "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==", 1314 | "dev": true, 1315 | "requires": { 1316 | "boxen": "^4.2.0", 1317 | "chalk": "^3.0.0", 1318 | "configstore": "^5.0.1", 1319 | "has-yarn": "^2.1.0", 1320 | "import-lazy": "^2.1.0", 1321 | "is-ci": "^2.0.0", 1322 | "is-installed-globally": "^0.3.1", 1323 | "is-npm": "^4.0.0", 1324 | "is-yarn-global": "^0.3.0", 1325 | "latest-version": "^5.0.0", 1326 | "pupa": "^2.0.1", 1327 | "semver-diff": "^3.1.1", 1328 | "xdg-basedir": "^4.0.0" 1329 | } 1330 | }, 1331 | "url-parse-lax": { 1332 | "version": "3.0.0", 1333 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", 1334 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", 1335 | "dev": true, 1336 | "requires": { 1337 | "prepend-http": "^2.0.0" 1338 | } 1339 | }, 1340 | "utils-merge": { 1341 | "version": "1.0.1", 1342 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 1343 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 1344 | }, 1345 | "vary": { 1346 | "version": "1.1.2", 1347 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 1348 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 1349 | }, 1350 | "widest-line": { 1351 | "version": "3.1.0", 1352 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz", 1353 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==", 1354 | "dev": true, 1355 | "requires": { 1356 | "string-width": "^4.0.0" 1357 | } 1358 | }, 1359 | "wrappy": { 1360 | "version": "1.0.2", 1361 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1362 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1363 | "dev": true 1364 | }, 1365 | "write-file-atomic": { 1366 | "version": "3.0.3", 1367 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", 1368 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", 1369 | "dev": true, 1370 | "requires": { 1371 | "imurmurhash": "^0.1.4", 1372 | "is-typedarray": "^1.0.0", 1373 | "signal-exit": "^3.0.2", 1374 | "typedarray-to-buffer": "^3.1.5" 1375 | } 1376 | }, 1377 | "xdg-basedir": { 1378 | "version": "4.0.0", 1379 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz", 1380 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==", 1381 | "dev": true 1382 | } 1383 | } 1384 | } 1385 | -------------------------------------------------------------------------------- /polling/exercise/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "polling", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node backend/server.js", 8 | "dev": "nodemon backend/server.js" 9 | }, 10 | "keywords": [], 11 | "type": "module", 12 | "author": "", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "express": "^4.17.1", 17 | "morgan": "^1.10.0", 18 | "nanobuffer": "^2.0.0" 19 | }, 20 | "devDependencies": { 21 | "nodemon": "^2.0.12" 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /polling/no-pause/backend/server.js: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | import bodyParser from "body-parser"; 3 | import nanobuffer from "nanobuffer"; 4 | import morgan from "morgan"; 5 | 6 | const app = express(); 7 | const msg = new nanobuffer(50); 8 | const getMsgs = () => Array.from(msg).reverse(); 9 | 10 | msg.push({ 11 | user: "brian", 12 | text: "hi", 13 | time: Date.now(), 14 | }); 15 | 16 | app.use(morgan("dev")); 17 | app.use(bodyParser.json()); 18 | app.use(express.static("frontend")); 19 | 20 | app.get("/poll", function (req, res) { 21 | res.json({ 22 | msg: getMsgs(), 23 | }); 24 | }); 25 | 26 | app.post("/poll", function (req, res) { 27 | const { user, text } = req.body; 28 | 29 | msg.push({ 30 | user, 31 | text, 32 | time: Date.now(), 33 | }); 34 | 35 | res.json({ 36 | status: "ok", 37 | }); 38 | }); 39 | 40 | const port = process.env.PORT || 3000; 41 | app.listen(port); 42 | console.log(`listening on http://localhost:${port}`); 43 | -------------------------------------------------------------------------------- /polling/no-pause/frontend/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Polling Chat 8 | 9 | 10 | 11 | 12 | 13 |

    Chat With Me

    14 |
    15 |
    16 |
    17 |
    18 | account_circle 19 | 20 | 21 |
    22 |
    23 | message 24 | 25 | 26 |
    27 |
    28 |
    29 | 32 |
    33 |
    34 |
    35 | 36 | 37 | 38 | 39 | 40 | 41 | -------------------------------------------------------------------------------- /polling/no-pause/frontend/polling-chat.js: -------------------------------------------------------------------------------- 1 | const chat = document.getElementById("chat"); 2 | const msgs = document.getElementById("msgs"); 3 | 4 | // this will hold all the most recent messages 5 | let allChat = []; 6 | 7 | const INTERVAL = 3000; 8 | 9 | chat.addEventListener("submit", function (e) { 10 | e.preventDefault(); 11 | postNewMsg(chat.elements.user.value, chat.elements.text.value); 12 | chat.elements.text.value = ""; 13 | }); 14 | async function postNewMsg(user, text) { 15 | const data = { 16 | user, 17 | text, 18 | }; 19 | 20 | // request options 21 | const options = { 22 | method: "POST", 23 | body: JSON.stringify(data), 24 | headers: { 25 | "Content-Type": "application/json", 26 | }, 27 | }; 28 | 29 | // send POST request 30 | const res = await fetch("/poll", options); 31 | const json = await res.json(); 32 | } 33 | 34 | async function getNewMsgs() { 35 | let json; 36 | try { 37 | const res = await fetch("/poll"); 38 | json = await res.json(); 39 | } catch (e) { 40 | // back off could would go here 41 | console.error("polling error", e); 42 | } 43 | allChat = json.msg; 44 | render(); 45 | setTimeout(getNewMsgs, INTERVAL); 46 | } 47 | 48 | function render() { 49 | // todo 50 | const html = allChat.map(({ user, text, time, id }) => 51 | template(user, text, time, id) 52 | ); 53 | msgs.innerHTML = html.join("\n"); 54 | } 55 | 56 | const template = (user, msg) => 57 | `
  • ${user}${msg}
  • `; 58 | 59 | getNewMsgs(); 60 | -------------------------------------------------------------------------------- /polling/no-pause/frontend/style.css: -------------------------------------------------------------------------------- 1 | .page-column { 2 | max-width: 600px; 3 | margin: 30px auto; 4 | } 5 | 6 | h1 { 7 | text-align: center; 8 | color: #26a69a; 9 | } 10 | 11 | .btn-box { 12 | text-align: center; 13 | } 14 | -------------------------------------------------------------------------------- /polling/no-pause/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "polling", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node backend/server.js", 8 | "dev": "nodemon backend/server.js" 9 | }, 10 | "keywords": [], 11 | "type": "module", 12 | "author": "", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "express": "^4.17.1", 17 | "morgan": "^1.10.0", 18 | "nanobuffer": "^2.0.0" 19 | }, 20 | "devDependencies": { 21 | "nodemon": "^2.0.12" 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /polling/pause-on-unfocus/backend/server.js: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | import bodyParser from "body-parser"; 3 | import nanobuffer from "nanobuffer"; 4 | import morgan from "morgan"; 5 | 6 | const app = express(); 7 | const msg = new nanobuffer(50); 8 | const getMsgs = () => Array.from(msg).reverse(); 9 | 10 | msg.push({ 11 | user: "brian", 12 | text: "hi", 13 | time: Date.now(), 14 | }); 15 | 16 | app.use(morgan("dev")); 17 | app.use(bodyParser.json()); 18 | app.use(express.static("frontend")); 19 | 20 | app.get("/poll", function (req, res) { 21 | res.json({ 22 | msg: getMsgs(), 23 | }); 24 | }); 25 | 26 | app.post("/poll", function (req, res) { 27 | const { user, text } = req.body; 28 | 29 | msg.push({ 30 | user, 31 | text, 32 | time: Date.now(), 33 | }); 34 | 35 | res.json({ 36 | status: "ok", 37 | }); 38 | }); 39 | 40 | const port = process.env.PORT || 3000; 41 | app.listen(port); 42 | console.log(`listening on http://localhost:${port}`); 43 | -------------------------------------------------------------------------------- /polling/pause-on-unfocus/frontend/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Polling Chat 8 | 9 | 10 | 11 | 12 | 13 |

    Chat With Me

    14 |
    15 |
    16 |
    17 |
    18 | account_circle 19 | 20 | 21 |
    22 |
    23 | message 24 | 25 | 26 |
    27 |
    28 |
    29 | 32 |
    33 |
    34 |
    35 | 36 | 37 | 38 | 39 | 40 | 41 | -------------------------------------------------------------------------------- /polling/pause-on-unfocus/frontend/polling-chat.js: -------------------------------------------------------------------------------- 1 | const chat = document.getElementById("chat"); 2 | const msgs = document.getElementById("msgs"); 3 | 4 | let allChat = []; 5 | 6 | const INTERVAL = 3000; 7 | 8 | let timeToMakeNextRequest = 0; 9 | 10 | chat.addEventListener("submit", function (e) { 11 | e.preventDefault(); 12 | postNewMsg(chat.elements.user.value, chat.elements.text.value); 13 | chat.elements.text.value = ""; 14 | }); 15 | 16 | async function postNewMsg(user, text) { 17 | const data = { 18 | user, 19 | text, 20 | }; 21 | 22 | // request options 23 | const options = { 24 | method: "POST", 25 | body: JSON.stringify(data), 26 | headers: { 27 | "Content-Type": "application/json", 28 | }, 29 | }; 30 | 31 | // send POST request 32 | const res = await fetch("/poll", options); 33 | const json = await res.json(); 34 | 35 | console.log("success", json); 36 | } 37 | 38 | async function getNewMsgs() { 39 | let json; 40 | try { 41 | const res = await fetch("/poll"); 42 | json = await res.json(); 43 | } catch (e) { 44 | // back off 45 | console.error("polling error", e); 46 | } 47 | allChat = json.msg; 48 | render(); 49 | } 50 | 51 | function render() { 52 | const html = allChat.map(({ user, text }) => template(user, text)); 53 | msgs.innerHTML = html.join("\n"); 54 | } 55 | 56 | const template = (user, msg) => 57 | `
  • ${user}${msg}
  • `; 58 | 59 | function rafTimer(time) { 60 | if (timeToMakeNextRequest <= time) { 61 | getNewMsgs(); 62 | timeToMakeNextRequest = time + INTERVAL; 63 | } 64 | requestAnimationFrame(rafTimer); 65 | } 66 | 67 | requestAnimationFrame(rafTimer); 68 | -------------------------------------------------------------------------------- /polling/pause-on-unfocus/frontend/style.css: -------------------------------------------------------------------------------- 1 | .page-column { 2 | max-width: 600px; 3 | margin: 30px auto; 4 | } 5 | 6 | h1 { 7 | text-align: center; 8 | color: #26a69a; 9 | } 10 | 11 | .btn-box { 12 | text-align: center; 13 | } 14 | -------------------------------------------------------------------------------- /polling/pause-on-unfocus/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "polling", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node backend/server.js", 8 | "dev": "nodemon backend/server.js" 9 | }, 10 | "keywords": [], 11 | "type": "module", 12 | "author": "", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "express": "^4.17.1", 17 | "morgan": "^1.10.0", 18 | "nanobuffer": "^2.0.0" 19 | }, 20 | "devDependencies": { 21 | "nodemon": "^2.0.12" 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /websockets/exercise-raw/backend/generate-accept-value.js: -------------------------------------------------------------------------------- 1 | import crypto from "crypto"; 2 | 3 | function generateAcceptValue(acceptKey) { 4 | return ( 5 | crypto 6 | .createHash("sha1") 7 | // this magic string key is actually in the spec 8 | .update(acceptKey + "258EAFA5-E914-47DA-95CA-C5AB0DC85B11", "binary") 9 | .digest("base64") 10 | ); 11 | } 12 | 13 | export default generateAcceptValue; 14 | -------------------------------------------------------------------------------- /websockets/exercise-raw/backend/obj-to-response.js: -------------------------------------------------------------------------------- 1 | export default function objToResponse(obj) { 2 | const string = JSON.stringify(obj); 3 | const stringBytes = Buffer.byteLength(string); 4 | // we're only doing two frames 5 | const lengthByteCount = stringBytes < 126 ? 0 : 2; 6 | const payloadLength = lengthByteCount === 0 ? stringBytes : 126; 7 | const buffer = Buffer.alloc(2 + lengthByteCount + stringBytes); 8 | 9 | buffer.writeUInt8(0b10000001, 0); 10 | buffer.writeUInt8(payloadLength, 1); 11 | 12 | let payloadOffset = 2; 13 | if (lengthByteCount > 0) { 14 | buffer.writeUInt16BE(stringBytes, 2); 15 | payloadOffset += lengthByteCount; 16 | } 17 | 18 | buffer.write(string, payloadOffset); 19 | return buffer; 20 | } 21 | -------------------------------------------------------------------------------- /websockets/exercise-raw/backend/parse-message.js: -------------------------------------------------------------------------------- 1 | function parseMessage(buffer) { 2 | const firstByte = buffer.readUInt8(0); 3 | const opCode = firstByte & 0xf; 4 | 5 | if (opCode === 8) { 6 | // connection closed 7 | return null; 8 | } 9 | if (opCode !== 1) { 10 | // we only care about text frames 11 | return; 12 | } 13 | 14 | const secondByte = buffer.readUInt8(1); 15 | const isMasked = secondByte >>> 7 === 1; 16 | // we should only be seeing masked frames from the browser 17 | if (!isMasked) { 18 | throw new Error("we only care about masked frames from the browser"); 19 | } 20 | 21 | const maskingKey = buffer.readUInt32BE(2); 22 | let currentOffset = 6; 23 | 24 | const messageLength = secondByte & 0x7f; 25 | if (messageLength > 125) { 26 | throw new Error("lol we're not doing big frames"); 27 | } 28 | 29 | // getting all of the bytes together for the string 30 | // then we'll convert it to a utf8 string 31 | const response = Buffer.alloc(messageLength); 32 | for (let i = 0; i < messageLength; i++) { 33 | // applying the mask to get the correct byte out 34 | const maskPosition = i % 4; 35 | 36 | let shift; 37 | if (maskPosition === 3) { 38 | shift = 0; 39 | } else { 40 | shift = (3 - maskPosition) << 3; 41 | } 42 | 43 | let mask; 44 | if (shift === 0) { 45 | mask = maskingKey & 0xff; 46 | } else { 47 | mask = (maskingKey >>> shift) & 0xff; 48 | } 49 | 50 | const source = buffer.readUInt8(currentOffset); 51 | currentOffset++; 52 | response.writeUInt8(mask ^ source, i); 53 | } 54 | 55 | return JSON.parse(response.toString("utf8")); 56 | } 57 | 58 | export default parseMessage; 59 | -------------------------------------------------------------------------------- /websockets/exercise-raw/backend/server.js: -------------------------------------------------------------------------------- 1 | import http from "http"; 2 | import handler from "serve-handler"; 3 | import nanobuffer from "nanobuffer"; 4 | 5 | // these are helpers to help you deal with the binary data that websockets use 6 | import objToResponse from "./obj-to-response.js"; 7 | import generateAcceptValue from "./generate-accept-value.js"; 8 | import parseMessage from "./parse-message.js"; 9 | 10 | let connections = []; 11 | const msg = new nanobuffer(50); 12 | const getMsgs = () => Array.from(msg).reverse(); 13 | 14 | msg.push({ 15 | user: "brian", 16 | text: "hi", 17 | time: Date.now(), 18 | }); 19 | 20 | // serve static assets 21 | const server = http.createServer((request, response) => { 22 | return handler(request, response, { 23 | public: "./frontend", 24 | }); 25 | }); 26 | 27 | /* 28 | * 29 | * your code goes here 30 | * 31 | */ 32 | 33 | const port = process.env.PORT || 8080; 34 | server.listen(port, () => 35 | console.log(`Server running at http://localhost:${port}`) 36 | ); 37 | -------------------------------------------------------------------------------- /websockets/exercise-raw/frontend/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Polling Chat 8 | 9 | 10 | 11 | 12 | 13 |
    🔴
    14 |

    Chat With Me

    15 |
    16 |
    17 |
    18 |
    19 | account_circle 20 | 21 | 22 |
    23 |
    24 | message 25 | 26 | 27 |
    28 |
    29 |
    30 | 33 |
    34 |
    35 |
    36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | -------------------------------------------------------------------------------- /websockets/exercise-raw/frontend/raw-chat.js: -------------------------------------------------------------------------------- 1 | const chat = document.getElementById("chat"); 2 | const msgs = document.getElementById("msgs"); 3 | const presence = document.getElementById("presence-indicator"); 4 | let allChat = []; 5 | 6 | // listen for events on the form 7 | chat.addEventListener("submit", function (e) { 8 | e.preventDefault(); 9 | postNewMsg(chat.elements.user.value, chat.elements.text.value); 10 | chat.elements.text.value = ""; 11 | }); 12 | 13 | async function postNewMsg(user, text) { 14 | // code goes here 15 | } 16 | 17 | /* 18 | * 19 | * your code goes here 20 | * 21 | */ 22 | 23 | function render() { 24 | const html = allChat.map(({ user, text }) => template(user, text)); 25 | msgs.innerHTML = html.join("\n"); 26 | } 27 | 28 | const template = (user, msg) => 29 | `
  • ${user}${msg}
  • `; 30 | -------------------------------------------------------------------------------- /websockets/exercise-raw/frontend/style.css: -------------------------------------------------------------------------------- 1 | .page-column { 2 | max-width: 600px; 3 | margin: 30px auto; 4 | } 5 | 6 | h1 { 7 | text-align: center; 8 | color: #26a69a; 9 | } 10 | 11 | .btn-box { 12 | text-align: center; 13 | } 14 | 15 | #presence-indicator { 16 | position: absolute; 17 | top: 5px; 18 | right: 15px; 19 | } 20 | -------------------------------------------------------------------------------- /websockets/exercise-raw/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "raw-websocket", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node backend/server.js", 8 | "dev": "nodemon backend/server.js" 9 | }, 10 | "keywords": [], 11 | "type": "module", 12 | "author": "", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "express": "^4.17.1", 17 | "morgan": "^1.10.0", 18 | "nanobuffer": "^2.0.0", 19 | "serve-handler": "^6.1.3" 20 | }, 21 | "devDependencies": { 22 | "nodemon": "^2.0.12" 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /websockets/exercise-socketio/backend/server.js: -------------------------------------------------------------------------------- 1 | import http from "http"; 2 | import handler from "serve-handler"; 3 | import nanobuffer from "nanobuffer"; 4 | import { Server } from "socket.io"; 5 | 6 | const msg = new nanobuffer(50); 7 | const getMsgs = () => Array.from(msg).reverse(); 8 | 9 | msg.push({ 10 | user: "brian", 11 | text: "hi", 12 | time: Date.now(), 13 | }); 14 | 15 | // serve static assets 16 | const server = http.createServer((request, response) => { 17 | return handler(request, response, { 18 | public: "./frontend", 19 | }); 20 | }); 21 | 22 | /* 23 | * 24 | * Code goes here 25 | * 26 | */ 27 | 28 | const port = process.env.PORT || 8080; 29 | server.listen(port, () => 30 | console.log(`Server running at http://localhost:${port}`) 31 | ); 32 | -------------------------------------------------------------------------------- /websockets/exercise-socketio/frontend/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Polling Chat 8 | 9 | 10 | 11 | 12 | 13 |
    🔴
    14 |

    Chat With Me

    15 |
    16 |
    17 |
    18 |
    19 | account_circle 20 | 21 | 22 |
    23 |
    24 | message 25 | 26 | 27 |
    28 |
    29 |
    30 | 33 |
    34 |
    35 |
    36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | -------------------------------------------------------------------------------- /websockets/exercise-socketio/frontend/socketio-chat.js: -------------------------------------------------------------------------------- 1 | // a global called "io" is being loaded separately 2 | 3 | const chat = document.getElementById("chat"); 4 | const msgs = document.getElementById("msgs"); 5 | const presence = document.getElementById("presence-indicator"); 6 | let allChat = []; 7 | 8 | /* 9 | * 10 | * Code goes here 11 | * 12 | */ 13 | 14 | chat.addEventListener("submit", function (e) { 15 | e.preventDefault(); 16 | postNewMsg(chat.elements.user.value, chat.elements.text.value); 17 | chat.elements.text.value = ""; 18 | }); 19 | 20 | async function postNewMsg(user, text) { 21 | /* 22 | * 23 | * Code goes here 24 | * 25 | */ 26 | } 27 | 28 | function render() { 29 | const html = allChat.map(({ user, text }) => template(user, text)); 30 | msgs.innerHTML = html.join("\n"); 31 | } 32 | 33 | const template = (user, msg) => 34 | `
  • ${user}${msg}
  • `; 35 | -------------------------------------------------------------------------------- /websockets/exercise-socketio/frontend/style.css: -------------------------------------------------------------------------------- 1 | .page-column { 2 | max-width: 600px; 3 | margin: 30px auto; 4 | } 5 | 6 | h1 { 7 | text-align: center; 8 | color: #26a69a; 9 | } 10 | 11 | .btn-box { 12 | text-align: center; 13 | } 14 | 15 | #presence-indicator { 16 | position: absolute; 17 | top: 5px; 18 | right: 15px; 19 | } 20 | -------------------------------------------------------------------------------- /websockets/exercise-socketio/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "raw-websocket", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node backend/server.js", 8 | "dev": "nodemon backend/server.js" 9 | }, 10 | "keywords": [], 11 | "type": "module", 12 | "author": "", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "express": "^4.17.1", 17 | "morgan": "^1.10.0", 18 | "nanobuffer": "^2.0.0", 19 | "serve-handler": "^6.1.3", 20 | "socket.io": "^4.1.3" 21 | }, 22 | "devDependencies": { 23 | "nodemon": "^2.0.12" 24 | } 25 | } 26 | -------------------------------------------------------------------------------- /websockets/raw/backend/generate-accept-value.js: -------------------------------------------------------------------------------- 1 | import crypto from "crypto"; 2 | 3 | function generateAcceptValue(acceptKey) { 4 | return ( 5 | crypto 6 | .createHash("sha1") 7 | // this magic string key is actually in the spec 8 | .update(acceptKey + "258EAFA5-E914-47DA-95CA-C5AB0DC85B11", "binary") 9 | .digest("base64") 10 | ); 11 | } 12 | 13 | export default generateAcceptValue; 14 | -------------------------------------------------------------------------------- /websockets/raw/backend/obj-to-response.js: -------------------------------------------------------------------------------- 1 | export default function objToResponse(obj) { 2 | const string = JSON.stringify(obj); 3 | const stringBytes = Buffer.byteLength(string); 4 | // we're only doing two frames 5 | const lengthByteCount = stringBytes < 126 ? 0 : 2; 6 | const payloadLength = lengthByteCount === 0 ? stringBytes : 126; 7 | const buffer = Buffer.alloc(2 + lengthByteCount + stringBytes); 8 | 9 | buffer.writeUInt8(0b10000001, 0); 10 | buffer.writeUInt8(payloadLength, 1); 11 | 12 | let payloadOffset = 2; 13 | if (lengthByteCount > 0) { 14 | buffer.writeUInt16BE(stringBytes, 2); 15 | payloadOffset += lengthByteCount; 16 | } 17 | 18 | buffer.write(string, payloadOffset); 19 | return buffer; 20 | } 21 | -------------------------------------------------------------------------------- /websockets/raw/backend/parse-message.js: -------------------------------------------------------------------------------- 1 | function parseMessage(buffer) { 2 | const firstByte = buffer.readUInt8(0); 3 | const opCode = firstByte & 0xf; 4 | 5 | if (opCode === 8) { 6 | // connection closed 7 | return null; 8 | } 9 | if (opCode !== 1) { 10 | // we only care about text frames 11 | return; 12 | } 13 | 14 | const secondByte = buffer.readUInt8(1); 15 | const isMasked = secondByte >>> 7 === 1; 16 | // we should only be seeing masked frames from the browser 17 | if (!isMasked) { 18 | throw new Error("we only care about masked frames from the browser"); 19 | } 20 | 21 | const maskingKey = buffer.readUInt32BE(2); 22 | let currentOffset = 6; 23 | 24 | const messageLength = secondByte & 0x7f; 25 | if (messageLength > 125) { 26 | throw new Error("lol we're not doing big frames"); 27 | } 28 | 29 | // getting all of the bytes together for the string 30 | // then we'll convert it to a utf8 string 31 | const response = Buffer.alloc(messageLength); 32 | for (let i = 0; i < messageLength; i++) { 33 | // applying the mask to get the correct byte out 34 | const maskPosition = i % 4; 35 | 36 | let shift; 37 | if (maskPosition === 3) { 38 | shift = 0; 39 | } else { 40 | shift = (3 - maskPosition) << 3; 41 | } 42 | 43 | let mask; 44 | if (shift === 0) { 45 | mask = maskingKey & 0xff; 46 | } else { 47 | mask = (maskingKey >>> shift) & 0xff; 48 | } 49 | 50 | const source = buffer.readUInt8(currentOffset); 51 | currentOffset++; 52 | response.writeUInt8(mask ^ source, i); 53 | } 54 | 55 | return JSON.parse(response.toString("utf8")); 56 | } 57 | 58 | export default parseMessage; 59 | -------------------------------------------------------------------------------- /websockets/raw/backend/server.js: -------------------------------------------------------------------------------- 1 | import http from "http"; 2 | import handler from "serve-handler"; 3 | import nanobuffer from "nanobuffer"; 4 | import objToResponse from "./obj-to-response.js"; 5 | import generateAcceptValue from "./generate-accept-value.js"; 6 | import parseMessage from "./parse-message.js"; 7 | 8 | let connections = []; 9 | const msg = new nanobuffer(50); 10 | const getMsgs = () => Array.from(msg).reverse(); 11 | 12 | msg.push({ 13 | user: "brian", 14 | text: "hi", 15 | time: Date.now(), 16 | }); 17 | 18 | // serve static assets 19 | const server = http.createServer((request, response) => { 20 | return handler(request, response, { 21 | public: "./frontend", 22 | }); 23 | }); 24 | 25 | server.on("upgrade", function (req, socket) { 26 | if (req.headers["upgrade"] !== "websocket") { 27 | // we only care about websockets 28 | socket.end("HTTP/1.1 400 Bad Request"); 29 | return; 30 | } 31 | const acceptKey = req.headers["sec-websocket-key"]; 32 | const acceptValue = generateAcceptValue(acceptKey); 33 | const headers = [ 34 | "HTTP/1.1 101 Web Socket Protocol Handshake", 35 | "Upgrade: WebSocket", 36 | "Connection: Upgrade", 37 | `Sec-WebSocket-Accept: ${acceptValue}`, 38 | "Sec-WebSocket-Protocol: json", 39 | "\r\n", 40 | ]; 41 | 42 | socket.write(headers.join("\r\n")); 43 | 44 | socket.write(objToResponse({ msg: getMsgs() })); 45 | 46 | connections.push(socket); 47 | 48 | socket.on("data", (buffer) => { 49 | const message = parseMessage(buffer); 50 | if (message) { 51 | console.log(message); 52 | msg.push({ 53 | user: message.user, 54 | text: message.text, 55 | time: Date.now(), 56 | }); 57 | 58 | connections.forEach((s) => s.write(objToResponse({ msg: getMsgs() }))); 59 | } else if (message === null) { 60 | // remove from my active connections 61 | socket.end(); 62 | } 63 | }); 64 | 65 | socket.on("end", () => { 66 | connections = connections.filter((s) => s !== socket); 67 | }); 68 | }); 69 | 70 | const port = process.env.PORT || 8080; 71 | server.listen(port, () => 72 | console.log(`Server running at http://localhost:${port}`) 73 | ); 74 | -------------------------------------------------------------------------------- /websockets/raw/frontend/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Polling Chat 8 | 9 | 10 | 11 | 12 | 13 |
    🔴
    14 |

    Chat With Me

    15 |
    16 |
    17 |
    18 |
    19 | account_circle 20 | 21 | 22 |
    23 |
    24 | message 25 | 26 | 27 |
    28 |
    29 |
    30 | 33 |
    34 |
    35 |
    36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | -------------------------------------------------------------------------------- /websockets/raw/frontend/raw-chat.js: -------------------------------------------------------------------------------- 1 | const chat = document.getElementById("chat"); 2 | const msgs = document.getElementById("msgs"); 3 | const presence = document.getElementById("presence-indicator"); 4 | let allChat = []; 5 | 6 | const ws = new WebSocket("ws://localhost:8080", ["json"]); 7 | 8 | ws.addEventListener("open", () => { 9 | console.log("connected"); 10 | presence.innerText = "🟢"; 11 | }); 12 | 13 | ws.addEventListener("close", () => { 14 | presence.innerText = "🔴"; 15 | }); 16 | 17 | ws.addEventListener("message", (event) => { 18 | const data = JSON.parse(event.data); 19 | allChat = data.msg; 20 | render(); 21 | }); 22 | 23 | chat.addEventListener("submit", function (e) { 24 | e.preventDefault(); 25 | postNewMsg(chat.elements.user.value, chat.elements.text.value); 26 | chat.elements.text.value = ""; 27 | }); 28 | 29 | async function postNewMsg(user, text) { 30 | const data = { 31 | user, 32 | text, 33 | }; 34 | 35 | ws.send(JSON.stringify(data)); 36 | } 37 | 38 | function render() { 39 | const html = allChat.map(({ user, text }) => template(user, text)); 40 | msgs.innerHTML = html.join("\n"); 41 | } 42 | 43 | const template = (user, msg) => 44 | `
  • ${user}${msg}
  • `; 45 | -------------------------------------------------------------------------------- /websockets/raw/frontend/style.css: -------------------------------------------------------------------------------- 1 | .page-column { 2 | max-width: 600px; 3 | margin: 30px auto; 4 | } 5 | 6 | h1 { 7 | text-align: center; 8 | color: #26a69a; 9 | } 10 | 11 | .btn-box { 12 | text-align: center; 13 | } 14 | 15 | #presence-indicator { 16 | position: absolute; 17 | top: 5px; 18 | right: 15px; 19 | } 20 | -------------------------------------------------------------------------------- /websockets/raw/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "raw-websocket", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node backend/server.js", 8 | "dev": "nodemon backend/server.js" 9 | }, 10 | "keywords": [], 11 | "type": "module", 12 | "author": "", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "express": "^4.17.1", 17 | "morgan": "^1.10.0", 18 | "nanobuffer": "^2.0.0", 19 | "serve-handler": "^6.1.3" 20 | }, 21 | "devDependencies": { 22 | "nodemon": "^2.0.12" 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /websockets/socketio/backend/server.js: -------------------------------------------------------------------------------- 1 | import http from "http"; 2 | import handler from "serve-handler"; 3 | import nanobuffer from "nanobuffer"; 4 | import { Server } from "socket.io"; 5 | 6 | const msg = new nanobuffer(50); 7 | const getMsgs = () => Array.from(msg).reverse(); 8 | 9 | msg.push({ 10 | user: "brian", 11 | text: "hi", 12 | time: Date.now(), 13 | }); 14 | 15 | // serve static assets 16 | const server = http.createServer((request, response) => { 17 | return handler(request, response, { 18 | public: "./frontend", 19 | }); 20 | }); 21 | 22 | const io = new Server(server, {}); 23 | 24 | io.on("connection", (socket) => { 25 | console.log(`connected: ${socket.id}`); 26 | 27 | socket.emit("msg:get", { msg: getMsgs() }); 28 | 29 | socket.on("msg:post", (data) => { 30 | msg.push({ 31 | user: data.user, 32 | text: data.text, 33 | time: Date.now(), 34 | }); 35 | io.emit("msg:get", { msg: getMsgs() }); 36 | }); 37 | 38 | socket.on("disconnect", () => { 39 | console.log(`disconnect: ${socket.id}`); 40 | }); 41 | }); 42 | 43 | const port = process.env.PORT || 8080; 44 | server.listen(port, () => 45 | console.log(`Server running at http://localhost:${port}`) 46 | ); 47 | -------------------------------------------------------------------------------- /websockets/socketio/frontend/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Polling Chat 8 | 9 | 10 | 11 | 12 | 13 |
    🔴
    14 |

    Chat With Me

    15 |
    16 |
    17 |
    18 |
    19 | account_circle 20 | 21 | 22 |
    23 |
    24 | message 25 | 26 | 27 |
    28 |
    29 |
    30 | 33 |
    34 |
    35 |
    36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | -------------------------------------------------------------------------------- /websockets/socketio/frontend/socketio-chat.js: -------------------------------------------------------------------------------- 1 | const chat = document.getElementById("chat"); 2 | const msgs = document.getElementById("msgs"); 3 | const presence = document.getElementById("presence-indicator"); 4 | let allChat = []; 5 | 6 | const socket = io("http://localhost:8080"); 7 | 8 | socket.on("connect", () => { 9 | console.log("connected"); 10 | presence.innerText = "🟢"; 11 | }); 12 | 13 | socket.on("disconnect", () => { 14 | presence.innerText = "🔴"; 15 | }); 16 | 17 | socket.on("msg:get", (data) => { 18 | allChat = data.msg; 19 | render(); 20 | }); 21 | 22 | chat.addEventListener("submit", function (e) { 23 | e.preventDefault(); 24 | postNewMsg(chat.elements.user.value, chat.elements.text.value); 25 | chat.elements.text.value = ""; 26 | }); 27 | 28 | async function postNewMsg(user, text) { 29 | const data = { 30 | user, 31 | text, 32 | }; 33 | 34 | socket.emit("msg:post", data); 35 | } 36 | 37 | function render() { 38 | const html = allChat.map(({ user, text }) => template(user, text)); 39 | msgs.innerHTML = html.join("\n"); 40 | } 41 | 42 | const template = (user, msg) => 43 | `
  • ${user}${msg}
  • `; 44 | -------------------------------------------------------------------------------- /websockets/socketio/frontend/style.css: -------------------------------------------------------------------------------- 1 | .page-column { 2 | max-width: 600px; 3 | margin: 30px auto; 4 | } 5 | 6 | h1 { 7 | text-align: center; 8 | color: #26a69a; 9 | } 10 | 11 | .btn-box { 12 | text-align: center; 13 | } 14 | 15 | #presence-indicator { 16 | position: absolute; 17 | top: 5px; 18 | right: 15px; 19 | } 20 | -------------------------------------------------------------------------------- /websockets/socketio/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "raw-websocket", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node backend/server.js", 8 | "dev": "nodemon backend/server.js" 9 | }, 10 | "keywords": [], 11 | "type": "module", 12 | "author": "", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "express": "^4.17.1", 17 | "morgan": "^1.10.0", 18 | "nanobuffer": "^2.0.0", 19 | "serve-handler": "^6.1.3", 20 | "socket.io": "^4.1.3" 21 | }, 22 | "devDependencies": { 23 | "nodemon": "^2.0.12" 24 | } 25 | } 26 | --------------------------------------------------------------------------------