Skip to content

Commit 6e4aaee

Browse files
committed
Implements client-side http transport
1 parent b49996d commit 6e4aaee

File tree

2 files changed

+91
-43
lines changed

2 files changed

+91
-43
lines changed

JavaScript/9-logger/http.js

+33-16
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,11 @@
33
const http = require('node:http');
44
const console = require('./logger.js');
55

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

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

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

JavaScript/9-logger/static/client.js

+58-27
Original file line numberDiff line numberDiff line change
@@ -1,44 +1,75 @@
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+
country: {
62+
read: ['id'],
63+
delete: ['id'],
64+
find: ['mask'],
65+
},
3366
},
34-
country: {
35-
read: ['id'],
36-
delete: ['id'],
37-
find: ['mask'],
38-
},
39-
});
67+
transports.http,
68+
);
69+
70+
// socket.addEventListener('open', async () => {
71+
// const data = await api.user.read(3);
72+
// console.dir({ data });
73+
// });
4074

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

0 commit comments

Comments
 (0)