Skip to content

Commit e2ee3c2

Browse files
committed
Implements client-side http transport
* Adds support of node 20 to package.json
1 parent 9e894c4 commit e2ee3c2

File tree

3 files changed

+99
-44
lines changed

3 files changed

+99
-44
lines changed

JavaScript/9-logger/http.js

+33-16
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,11 @@
22

33
const http = require('node:http');
44

5+
const answerNotFound = (res, headers) => {
6+
res.writeHead(404, headers);
7+
res.end('Not found');
8+
};
9+
510
const receiveArgs = async (req) => {
611
const buffers = [];
712
for await (const chunk of req) buffers.push(chunk);
@@ -10,22 +15,34 @@ const receiveArgs = async (req) => {
1015
};
1116

1217
module.exports = (routing, port, console) => {
13-
http.createServer(async (req, res) => {
14-
const { url, socket } = req;
15-
const [name, method, id] = url.substring(1).split('/');
16-
const entity = routing[name];
17-
if (!entity) return void res.end('Not found');
18-
const handler = entity[method];
19-
if (!handler) return void res.end('Not found');
20-
const src = handler.toString();
21-
const signature = src.substring(0, src.indexOf(')'));
22-
const args = [];
23-
if (signature.includes('(id')) args.push(id);
24-
if (signature.includes('{')) args.push(await receiveArgs(req));
25-
console.log(`${socket.remoteAddress} ${method} ${url}`);
26-
const result = await handler(...args);
27-
res.end(JSON.stringify(result.rows));
28-
}).listen(port);
18+
http
19+
.createServer(async (req, res) => {
20+
const headers = {
21+
'Access-Control-Allow-Origin':
22+
'*' /* @dev First, read about security */,
23+
'Access-Control-Allow-Methods': 'POST',
24+
'Access-Control-Max-Age': 2592000, // 30 days
25+
/** add other headers as per requirement */
26+
};
27+
28+
const { url, socket } = req;
29+
const [name, method, id] = url.substring(1).split('/');
30+
const entity = routing[name];
31+
if (!entity) return void answerNotFound(res, headers);
32+
const handler = entity[method];
33+
if (!handler) return void answerNotFound(res, headers);
34+
35+
res.writeHead(200, headers);
36+
const src = handler.toString();
37+
const signature = src.substring(0, src.indexOf(')'));
38+
const args = [];
39+
if (signature.includes('(id')) args.push(id);
40+
args.push(...(await receiveArgs(req)));
41+
console.log(`${socket.remoteAddress} ${req.method} ${url}`);
42+
const result = await handler(...args);
43+
res.end(JSON.stringify(result.rows));
44+
})
45+
.listen(port);
2946

3047
console.log(`API on port ${port}`);
3148
};

JavaScript/9-logger/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66
"private": true,
77
"main": "main.js",
88
"engines": {
9-
"node": "14 || 16 || 18"
9+
"node": "14 || 16 || 18 || 20"
1010
},
1111
"scripts": {
1212
"start": "node --experimental-vm-modules main.js"

JavaScript/9-logger/static/client.js

+65-27
Original file line numberDiff line numberDiff line change
@@ -1,44 +1,82 @@
11
'use strict';
22

3-
const socket = new WebSocket('ws://127.0.0.1:8001/');
3+
const HTTP_API_URL = 'http://localhost:8001';
4+
let socket;
45

5-
const scaffold = (structure) => {
6+
const transports = {
7+
ws: (name, method) => {
8+
if (!socket) socket = new WebSocket('ws://127.0.0.1:8001/');
9+
return (...args) => {
10+
return new Promise((resolve) => {
11+
const packet = { name, method, args };
12+
socket.send(JSON.stringify(packet));
13+
socket.onmessage = (event) => {
14+
const data = JSON.parse(event.data);
15+
resolve(data);
16+
};
17+
});
18+
};
19+
},
20+
21+
http:
22+
(name, method, methodArgs) =>
23+
(...args) => {
24+
const urlParts = [HTTP_API_URL, name, method];
25+
if (methodArgs[0] === 'id') urlParts.push(args.shift());
26+
27+
return fetch(urlParts.join('/'), {
28+
method: 'POST',
29+
body: JSON.stringify(args),
30+
}).then((response) => response.json());
31+
},
32+
};
33+
34+
const scaffold = (structure, transport) => {
635
const api = {};
736
const services = Object.keys(structure);
837
for (const serviceName of services) {
938
api[serviceName] = {};
1039
const service = structure[serviceName];
1140
const methods = Object.keys(service);
1241
for (const methodName of methods) {
13-
api[serviceName][methodName] = (...args) => new Promise((resolve) => {
14-
const packet = { name: serviceName, method: methodName, args };
15-
socket.send(JSON.stringify(packet));
16-
socket.onmessage = (event) => {
17-
const data = JSON.parse(event.data);
18-
resolve(data);
19-
};
20-
});
42+
api[serviceName][methodName] = transport(
43+
serviceName,
44+
methodName,
45+
service[methodName],
46+
);
2147
}
2248
}
2349
return api;
2450
};
2551

26-
const api = scaffold({
27-
user: {
28-
create: ['record'],
29-
read: ['id'],
30-
update: ['id', 'record'],
31-
delete: ['id'],
32-
find: ['mask'],
52+
const api = scaffold(
53+
{
54+
user: {
55+
create: ['record'],
56+
read: ['id'],
57+
update: ['id', 'record'],
58+
delete: ['id'],
59+
find: ['mask'],
60+
},
61+
city: {
62+
read: ['id'],
63+
create: ['record'],
64+
update: ['id', 'record'],
65+
delete: ['id'],
66+
},
67+
country: {
68+
read: ['id'],
69+
create: ['record'],
70+
update: ['id', 'record'],
71+
delete: ['id'],
72+
},
3373
},
34-
country: {
35-
read: ['id'],
36-
delete: ['id'],
37-
find: ['mask'],
38-
},
39-
});
74+
transports.http,
75+
);
76+
77+
// socket.addEventListener('open', async () => {
78+
// const data = await api.user.read(3);
79+
// console.dir({ data });
80+
// });
4081

41-
socket.addEventListener('open', async () => {
42-
const data = await api.user.read(3);
43-
console.dir({ data });
44-
});
82+
(async () => console.log(await api.user.read(3)))();

0 commit comments

Comments
 (0)