├── .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 |
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 |
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 |
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 |
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 |
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 |
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 |
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 |
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 |
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 |
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 |
--------------------------------------------------------------------------------