Skip to content

Commit 3a02b6b

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

File tree

4 files changed

+99
-40
lines changed

4 files changed

+99
-40
lines changed

JavaScript/9-logger/config.js

+7-7
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,14 @@
11
module.exports = {
2-
transport: "http", // ws | http
2+
transport: 'ws', // ws | http
33
apiPort: 8001,
44
staticPort: 8000,
5-
staticPath: "./static",
6-
logPath: "./log",
5+
staticPath: './static',
6+
logPath: './log',
77
db: {
8-
host: "127.0.0.1",
8+
host: '127.0.0.1',
99
port: 5432,
10-
database: "example",
11-
user: "marcus",
12-
password: "marcus",
10+
database: 'example',
11+
user: 'marcus',
12+
password: 'marcus',
1313
},
1414
};

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

+58-16
Original file line numberDiff line numberDiff line change
@@ -1,44 +1,86 @@
11
'use strict';
22

3-
const socket = new WebSocket('ws://127.0.0.1:8001/');
3+
const API_URL = 'ws://localhost:8001';
44

5-
const scaffold = (structure) => {
5+
const getTransport = (url) => {
6+
let socket;
7+
const protocol = url.split(':')[0];
8+
9+
const transports = {
10+
ws: (name, method) => {
11+
if (!socket) socket = new WebSocket(url);
12+
return (...args) => {
13+
return new Promise((resolve) => {
14+
const packet = { name, method, args };
15+
socket.send(JSON.stringify(packet));
16+
socket.onmessage = (event) => {
17+
const data = JSON.parse(event.data);
18+
resolve(data);
19+
};
20+
});
21+
};
22+
},
23+
24+
http:
25+
(name, method, methodArgs) =>
26+
(...args) => {
27+
const urlParts = [url, name, method];
28+
if (methodArgs[0] === 'id') urlParts.push(args.shift());
29+
30+
return fetch(urlParts.join('/'), {
31+
method: 'POST',
32+
body: JSON.stringify(args),
33+
}).then((response) => response.json());
34+
},
35+
};
36+
37+
return transports[protocol];
38+
};
39+
40+
const scaffold = (url, structure) => {
41+
const transport = getTransport(url);
642
const api = {};
743
const services = Object.keys(structure);
844
for (const serviceName of services) {
945
api[serviceName] = {};
1046
const service = structure[serviceName];
1147
const methods = Object.keys(service);
1248
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-
});
49+
api[serviceName][methodName] = transport(
50+
serviceName,
51+
methodName,
52+
service[methodName],
53+
);
2154
}
2255
}
2356
return api;
2457
};
2558

26-
const api = scaffold({
59+
const api = scaffold(API_URL, {
2760
user: {
2861
create: ['record'],
2962
read: ['id'],
3063
update: ['id', 'record'],
3164
delete: ['id'],
3265
find: ['mask'],
3366
},
67+
city: {
68+
read: ['id'],
69+
create: ['record'],
70+
update: ['id', 'record'],
71+
delete: ['id'],
72+
},
3473
country: {
3574
read: ['id'],
75+
create: ['record'],
76+
update: ['id', 'record'],
3677
delete: ['id'],
37-
find: ['mask'],
3878
},
3979
});
4080

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

0 commit comments

Comments
 (0)