@omen.foundation/node-microservice-runtime 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.env +13 -0
- package/dist/auth.cjs +97 -0
- package/dist/auth.d.ts +14 -0
- package/dist/auth.d.ts.map +1 -0
- package/dist/auth.js +93 -0
- package/dist/auth.js.map +1 -0
- package/dist/cli/index.d.ts +3 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +588 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/decorators.cjs +181 -0
- package/dist/decorators.d.ts +23 -0
- package/dist/decorators.d.ts.map +1 -0
- package/dist/decorators.js +155 -0
- package/dist/decorators.js.map +1 -0
- package/dist/dependency.cjs +165 -0
- package/dist/dependency.d.ts +56 -0
- package/dist/dependency.d.ts.map +1 -0
- package/dist/dependency.js +162 -0
- package/dist/dependency.js.map +1 -0
- package/dist/dev.cjs +34 -0
- package/dist/dev.d.ts +9 -0
- package/dist/dev.d.ts.map +1 -0
- package/dist/dev.js +32 -0
- package/dist/dev.js.map +1 -0
- package/dist/discovery.cjs +79 -0
- package/dist/discovery.d.ts +20 -0
- package/dist/discovery.d.ts.map +1 -0
- package/dist/discovery.js +75 -0
- package/dist/discovery.js.map +1 -0
- package/dist/docs.cjs +206 -0
- package/dist/docs.d.ts +30 -0
- package/dist/docs.d.ts.map +1 -0
- package/dist/docs.js +209 -0
- package/dist/docs.js.map +1 -0
- package/dist/env.cjs +106 -0
- package/dist/env.d.ts +4 -0
- package/dist/env.d.ts.map +1 -0
- package/dist/env.js +108 -0
- package/dist/env.js.map +1 -0
- package/dist/errors.cjs +58 -0
- package/dist/errors.d.ts +26 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +48 -0
- package/dist/errors.js.map +1 -0
- package/dist/federation.cjs +356 -0
- package/dist/federation.d.ts +108 -0
- package/dist/federation.d.ts.map +1 -0
- package/dist/federation.js +341 -0
- package/dist/federation.js.map +1 -0
- package/dist/index.cjs +42 -0
- package/dist/index.d.ts +13 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +10 -0
- package/dist/index.js.map +1 -0
- package/dist/inventory.cjs +361 -0
- package/dist/inventory.d.ts +116 -0
- package/dist/inventory.d.ts.map +1 -0
- package/dist/inventory.js +351 -0
- package/dist/inventory.js.map +1 -0
- package/dist/logger.cjs +62 -0
- package/dist/logger.d.ts +9 -0
- package/dist/logger.d.ts.map +1 -0
- package/dist/logger.js +29 -0
- package/dist/logger.js.map +1 -0
- package/dist/message.cjs +19 -0
- package/dist/message.d.ts +5 -0
- package/dist/message.d.ts.map +1 -0
- package/dist/message.js +15 -0
- package/dist/message.js.map +1 -0
- package/dist/requester.cjs +100 -0
- package/dist/requester.d.ts +20 -0
- package/dist/requester.d.ts.map +1 -0
- package/dist/requester.js +99 -0
- package/dist/requester.js.map +1 -0
- package/dist/routing.cjs +39 -0
- package/dist/routing.d.ts +2 -0
- package/dist/routing.d.ts.map +1 -0
- package/dist/routing.js +36 -0
- package/dist/routing.js.map +1 -0
- package/dist/runtime.cjs +735 -0
- package/dist/runtime.d.ts +40 -0
- package/dist/runtime.d.ts.map +1 -0
- package/dist/runtime.js +825 -0
- package/dist/runtime.js.map +1 -0
- package/dist/services.cjs +346 -0
- package/dist/services.d.ts +46 -0
- package/dist/services.d.ts.map +1 -0
- package/dist/services.js +343 -0
- package/dist/services.js.map +1 -0
- package/dist/storage.cjs +147 -0
- package/dist/storage.d.ts +46 -0
- package/dist/storage.d.ts.map +1 -0
- package/dist/storage.js +144 -0
- package/dist/storage.js.map +1 -0
- package/dist/types.cjs +2 -0
- package/dist/types.d.ts +108 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/dist/utils/urls.cjs +55 -0
- package/dist/utils/urls.d.ts +5 -0
- package/dist/utils/urls.d.ts.map +1 -0
- package/dist/utils/urls.js +50 -0
- package/dist/utils/urls.js.map +1 -0
- package/dist/websocket.cjs +142 -0
- package/dist/websocket.d.ts +33 -0
- package/dist/websocket.d.ts.map +1 -0
- package/dist/websocket.js +139 -0
- package/dist/websocket.js.map +1 -0
- package/env.sample +13 -0
- package/package.json +49 -0
- package/scripts/generate-openapi.mjs +114 -0
- package/scripts/lib/cli-utils.mjs +58 -0
- package/scripts/prepare-cjs.mjs +44 -0
- package/scripts/publish-service.mjs +1126 -0
- package/scripts/validate-service.mjs +103 -0
- package/scripts/ws-test.mjs +25 -0
- package/src/auth.ts +117 -0
- package/src/cli/index.ts +699 -0
- package/src/decorators.ts +207 -0
- package/src/dependency.ts +211 -0
- package/src/dev.ts +17 -0
- package/src/discovery.ts +88 -0
- package/src/docs.ts +262 -0
- package/src/env.ts +125 -0
- package/src/errors.ts +55 -0
- package/src/federation.ts +559 -0
- package/src/index.ts +51 -0
- package/src/inventory.ts +491 -0
- package/src/logger.ts +38 -0
- package/src/message.ts +19 -0
- package/src/requester.ts +126 -0
- package/src/routing.ts +42 -0
- package/src/runtime.ts +967 -0
- package/src/services.ts +459 -0
- package/src/storage.ts +206 -0
- package/src/types/beamable-sdk-api.d.ts +5 -0
- package/src/types.ts +117 -0
- package/src/utils/urls.ts +53 -0
- package/src/websocket.ts +170 -0
- package/tsconfig.base.json +31 -0
- package/tsconfig.build.json +10 -0
- package/tsconfig.cjs.json +16 -0
- package/tsconfig.dev.json +14 -0
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.GatewayRequester = void 0;
|
|
4
|
+
const node_events_1 = require("node:events");
|
|
5
|
+
const errors_js_1 = require("./errors.js");
|
|
6
|
+
const message_js_1 = require("./message.js");
|
|
7
|
+
class GatewayRequester {
|
|
8
|
+
constructor(socket, logger) {
|
|
9
|
+
this.pending = new Map();
|
|
10
|
+
this.emitter = new node_events_1.EventEmitter();
|
|
11
|
+
this.requestTimeoutMs = 15000;
|
|
12
|
+
this.socket = socket;
|
|
13
|
+
this.logger = logger.child({ component: 'GatewayRequester' });
|
|
14
|
+
this.socket.on('message', (...args) => { var _a; return this.onRawMessage(String((_a = args[0]) !== null && _a !== void 0 ? _a : '')); });
|
|
15
|
+
}
|
|
16
|
+
on(event, listener) {
|
|
17
|
+
this.emitter.on(event, listener);
|
|
18
|
+
}
|
|
19
|
+
off(event, listener) {
|
|
20
|
+
this.emitter.off(event, listener);
|
|
21
|
+
}
|
|
22
|
+
setRequestTimeout(timeoutMs) {
|
|
23
|
+
this.requestTimeoutMs = timeoutMs;
|
|
24
|
+
}
|
|
25
|
+
async request(method, path, body) {
|
|
26
|
+
const id = (0, message_js_1.allocateRequestId)();
|
|
27
|
+
const effectiveBody = body !== null && body !== void 0 ? body : {};
|
|
28
|
+
const request = {
|
|
29
|
+
id,
|
|
30
|
+
method: method.toLowerCase(),
|
|
31
|
+
path,
|
|
32
|
+
body: effectiveBody,
|
|
33
|
+
};
|
|
34
|
+
this.logger.debug({ id, method, path, body: effectiveBody }, 'Sending gateway request.');
|
|
35
|
+
const payload = (0, message_js_1.serializeGatewayRequest)(request);
|
|
36
|
+
return new Promise((resolve, reject) => {
|
|
37
|
+
const timeoutHandle = setTimeout(() => {
|
|
38
|
+
this.pending.delete(id);
|
|
39
|
+
reject(new errors_js_1.TimeoutError(`Request ${method} ${path} timed out after ${this.requestTimeoutMs}ms`));
|
|
40
|
+
}, this.requestTimeoutMs).unref();
|
|
41
|
+
this.pending.set(id, {
|
|
42
|
+
resolve: resolve,
|
|
43
|
+
reject,
|
|
44
|
+
timeoutHandle,
|
|
45
|
+
path,
|
|
46
|
+
method,
|
|
47
|
+
});
|
|
48
|
+
this.socket
|
|
49
|
+
.send(payload)
|
|
50
|
+
.catch((error) => {
|
|
51
|
+
clearTimeout(timeoutHandle);
|
|
52
|
+
this.pending.delete(id);
|
|
53
|
+
reject(error instanceof Error ? error : new Error(String(error)));
|
|
54
|
+
});
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
async sendResponse(response) {
|
|
58
|
+
const payload = JSON.stringify(response);
|
|
59
|
+
await this.socket.send(payload);
|
|
60
|
+
}
|
|
61
|
+
async acknowledge(id, status = 200) {
|
|
62
|
+
await this.sendResponse({ id, status, body: null });
|
|
63
|
+
}
|
|
64
|
+
onRawMessage(raw) {
|
|
65
|
+
this.logger.debug({ raw }, 'Received websocket frame.');
|
|
66
|
+
let envelope;
|
|
67
|
+
try {
|
|
68
|
+
envelope = (0, message_js_1.deserializeGatewayResponse)(raw);
|
|
69
|
+
}
|
|
70
|
+
catch (error) {
|
|
71
|
+
this.logger.error({ err: error, raw }, 'Failed to parse gateway message.');
|
|
72
|
+
return;
|
|
73
|
+
}
|
|
74
|
+
if (typeof envelope.id !== 'number') {
|
|
75
|
+
this.logger.warn({ envelope }, 'Invalid gateway message without id.');
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
const pending = this.pending.get(envelope.id);
|
|
79
|
+
if (pending) {
|
|
80
|
+
clearTimeout(pending.timeoutHandle);
|
|
81
|
+
this.pending.delete(envelope.id);
|
|
82
|
+
if ('status' in envelope && envelope.status === 403) {
|
|
83
|
+
pending.reject(new errors_js_1.AuthenticationError('Gateway rejected the request with 403.'));
|
|
84
|
+
return;
|
|
85
|
+
}
|
|
86
|
+
pending.resolve(envelope.body);
|
|
87
|
+
return;
|
|
88
|
+
}
|
|
89
|
+
this.emitter.emit('event', envelope);
|
|
90
|
+
}
|
|
91
|
+
dispose() {
|
|
92
|
+
for (const [id, pending] of this.pending.entries()) {
|
|
93
|
+
clearTimeout(pending.timeoutHandle);
|
|
94
|
+
pending.reject(new Error(`Request ${pending.method} ${pending.path} cancelled during shutdown.`));
|
|
95
|
+
this.pending.delete(id);
|
|
96
|
+
}
|
|
97
|
+
this.emitter.removeAllListeners();
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
exports.GatewayRequester = GatewayRequester;
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import type { Logger } from 'pino';
|
|
2
|
+
import { BeamableWebSocket } from './websocket.js';
|
|
3
|
+
import type { GatewayResponse, WebsocketEventEnvelope } from './types.js';
|
|
4
|
+
export declare class GatewayRequester {
|
|
5
|
+
private readonly socket;
|
|
6
|
+
private readonly logger;
|
|
7
|
+
private readonly pending;
|
|
8
|
+
private readonly emitter;
|
|
9
|
+
private requestTimeoutMs;
|
|
10
|
+
constructor(socket: BeamableWebSocket, logger: Logger);
|
|
11
|
+
on(event: 'event', listener: (envelope: WebsocketEventEnvelope) => void): void;
|
|
12
|
+
off(event: 'event', listener: (envelope: WebsocketEventEnvelope) => void): void;
|
|
13
|
+
setRequestTimeout(timeoutMs: number): void;
|
|
14
|
+
request<T>(method: string, path: string, body?: unknown): Promise<T>;
|
|
15
|
+
sendResponse(response: GatewayResponse): Promise<void>;
|
|
16
|
+
acknowledge(id: number, status?: number): Promise<void>;
|
|
17
|
+
private onRawMessage;
|
|
18
|
+
dispose(): void;
|
|
19
|
+
}
|
|
20
|
+
//# sourceMappingURL=requester.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"requester.d.ts","sourceRoot":"","sources":["../src/requester.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AAGnC,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,KAAK,EAAkB,eAAe,EAAE,sBAAsB,EAAE,MAAM,YAAY,CAAC;AAU1F,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAoB;IAC3C,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAS;IAChC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAqC;IAC7D,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAsB;IAC9C,OAAO,CAAC,gBAAgB,CAAU;gBAEtB,MAAM,EAAE,iBAAiB,EAAE,MAAM,EAAE,MAAM;IAMrD,EAAE,CAAC,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,CAAC,QAAQ,EAAE,sBAAsB,KAAK,IAAI,GAAG,IAAI;IAI9E,GAAG,CAAC,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,CAAC,QAAQ,EAAE,sBAAsB,KAAK,IAAI,GAAG,IAAI;IAI/E,iBAAiB,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI;IAIpC,OAAO,CAAC,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,OAAO,GAAG,OAAO,CAAC,CAAC,CAAC;IAsCpE,YAAY,CAAC,QAAQ,EAAE,eAAe,GAAG,OAAO,CAAC,IAAI,CAAC;IAKtD,WAAW,CAAC,EAAE,EAAE,MAAM,EAAE,MAAM,SAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAI1D,OAAO,CAAC,YAAY;IA8BpB,OAAO,IAAI,IAAI;CAQhB"}
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
import { EventEmitter } from 'node:events';
|
|
2
|
+
import { AuthenticationError, TimeoutError } from './errors.js';
|
|
3
|
+
import { allocateRequestId, deserializeGatewayResponse, serializeGatewayRequest } from './message.js';
|
|
4
|
+
export class GatewayRequester {
|
|
5
|
+
socket;
|
|
6
|
+
logger;
|
|
7
|
+
pending = new Map();
|
|
8
|
+
emitter = new EventEmitter();
|
|
9
|
+
requestTimeoutMs = 15_000;
|
|
10
|
+
constructor(socket, logger) {
|
|
11
|
+
this.socket = socket;
|
|
12
|
+
this.logger = logger.child({ component: 'GatewayRequester' });
|
|
13
|
+
this.socket.on('message', (...args) => this.onRawMessage(String(args[0] ?? '')));
|
|
14
|
+
}
|
|
15
|
+
on(event, listener) {
|
|
16
|
+
this.emitter.on(event, listener);
|
|
17
|
+
}
|
|
18
|
+
off(event, listener) {
|
|
19
|
+
this.emitter.off(event, listener);
|
|
20
|
+
}
|
|
21
|
+
setRequestTimeout(timeoutMs) {
|
|
22
|
+
this.requestTimeoutMs = timeoutMs;
|
|
23
|
+
}
|
|
24
|
+
async request(method, path, body) {
|
|
25
|
+
const id = allocateRequestId();
|
|
26
|
+
const effectiveBody = body ?? {};
|
|
27
|
+
const request = {
|
|
28
|
+
id,
|
|
29
|
+
method: method.toLowerCase(),
|
|
30
|
+
path,
|
|
31
|
+
body: effectiveBody,
|
|
32
|
+
};
|
|
33
|
+
this.logger.debug({ id, method, path, body: effectiveBody }, 'Sending gateway request.');
|
|
34
|
+
const payload = serializeGatewayRequest(request);
|
|
35
|
+
return new Promise((resolve, reject) => {
|
|
36
|
+
const timeoutHandle = setTimeout(() => {
|
|
37
|
+
this.pending.delete(id);
|
|
38
|
+
reject(new TimeoutError(`Request ${method} ${path} timed out after ${this.requestTimeoutMs}ms`));
|
|
39
|
+
}, this.requestTimeoutMs).unref();
|
|
40
|
+
this.pending.set(id, {
|
|
41
|
+
resolve: resolve,
|
|
42
|
+
reject,
|
|
43
|
+
timeoutHandle,
|
|
44
|
+
path,
|
|
45
|
+
method,
|
|
46
|
+
});
|
|
47
|
+
this.socket
|
|
48
|
+
.send(payload)
|
|
49
|
+
.catch((error) => {
|
|
50
|
+
clearTimeout(timeoutHandle);
|
|
51
|
+
this.pending.delete(id);
|
|
52
|
+
reject(error instanceof Error ? error : new Error(String(error)));
|
|
53
|
+
});
|
|
54
|
+
});
|
|
55
|
+
}
|
|
56
|
+
async sendResponse(response) {
|
|
57
|
+
const payload = JSON.stringify(response);
|
|
58
|
+
await this.socket.send(payload);
|
|
59
|
+
}
|
|
60
|
+
async acknowledge(id, status = 200) {
|
|
61
|
+
await this.sendResponse({ id, status, body: null });
|
|
62
|
+
}
|
|
63
|
+
onRawMessage(raw) {
|
|
64
|
+
this.logger.debug({ raw }, 'Received websocket frame.');
|
|
65
|
+
let envelope;
|
|
66
|
+
try {
|
|
67
|
+
envelope = deserializeGatewayResponse(raw);
|
|
68
|
+
}
|
|
69
|
+
catch (error) {
|
|
70
|
+
this.logger.error({ err: error, raw }, 'Failed to parse gateway message.');
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
if (typeof envelope.id !== 'number') {
|
|
74
|
+
this.logger.warn({ envelope }, 'Invalid gateway message without id.');
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
const pending = this.pending.get(envelope.id);
|
|
78
|
+
if (pending) {
|
|
79
|
+
clearTimeout(pending.timeoutHandle);
|
|
80
|
+
this.pending.delete(envelope.id);
|
|
81
|
+
if ('status' in envelope && envelope.status === 403) {
|
|
82
|
+
pending.reject(new AuthenticationError('Gateway rejected the request with 403.'));
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
pending.resolve(envelope.body);
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
this.emitter.emit('event', envelope);
|
|
89
|
+
}
|
|
90
|
+
dispose() {
|
|
91
|
+
for (const [id, pending] of this.pending.entries()) {
|
|
92
|
+
clearTimeout(pending.timeoutHandle);
|
|
93
|
+
pending.reject(new Error(`Request ${pending.method} ${pending.path} cancelled during shutdown.`));
|
|
94
|
+
this.pending.delete(id);
|
|
95
|
+
}
|
|
96
|
+
this.emitter.removeAllListeners();
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
//# sourceMappingURL=requester.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"requester.js","sourceRoot":"","sources":["../src/requester.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter } from 'node:events';\r\nimport type { Logger } from 'pino';\r\nimport { AuthenticationError, TimeoutError } from './errors.js';\r\nimport { allocateRequestId, deserializeGatewayResponse, serializeGatewayRequest } from './message.js';\r\nimport { BeamableWebSocket } from './websocket.js';\r\nimport type { GatewayRequest, GatewayResponse, WebsocketEventEnvelope } from './types.js';\r\n\r\ninterface PendingRequest<T = unknown> {\r\n resolve: (value: T) => void;\r\n reject: (error: Error) => void;\r\n timeoutHandle: NodeJS.Timeout;\r\n path: string;\r\n method: string;\r\n}\r\n\r\nexport class GatewayRequester {\r\n private readonly socket: BeamableWebSocket;\r\n private readonly logger: Logger;\r\n private readonly pending = new Map<number, PendingRequest>();\r\n private readonly emitter = new EventEmitter();\r\n private requestTimeoutMs = 15_000;\r\n\r\n constructor(socket: BeamableWebSocket, logger: Logger) {\r\n this.socket = socket;\r\n this.logger = logger.child({ component: 'GatewayRequester' });\r\n this.socket.on('message', (...args) => this.onRawMessage(String(args[0] ?? '')));\r\n }\r\n\r\n on(event: 'event', listener: (envelope: WebsocketEventEnvelope) => void): void {\r\n this.emitter.on(event, listener);\r\n }\r\n\r\n off(event: 'event', listener: (envelope: WebsocketEventEnvelope) => void): void {\r\n this.emitter.off(event, listener);\r\n }\r\n\r\n setRequestTimeout(timeoutMs: number): void {\r\n this.requestTimeoutMs = timeoutMs;\r\n }\r\n\r\n async request<T>(method: string, path: string, body?: unknown): Promise<T> {\r\n const id = allocateRequestId();\r\n const effectiveBody = body ?? {};\r\n const request: GatewayRequest = {\r\n id,\r\n method: method.toLowerCase(),\r\n path,\r\n body: effectiveBody,\r\n };\r\n\r\n this.logger.debug({ id, method, path, body: effectiveBody }, 'Sending gateway request.');\r\n\r\n const payload = serializeGatewayRequest(request);\r\n\r\n return new Promise<T>((resolve, reject) => {\r\n const timeoutHandle = setTimeout(() => {\r\n this.pending.delete(id);\r\n reject(new TimeoutError(`Request ${method} ${path} timed out after ${this.requestTimeoutMs}ms`));\r\n }, this.requestTimeoutMs).unref();\r\n\r\n this.pending.set(id, {\r\n resolve: resolve as (value: unknown) => void,\r\n reject,\r\n timeoutHandle,\r\n path,\r\n method,\r\n });\r\n\r\n this.socket\r\n .send(payload)\r\n .catch((error) => {\r\n clearTimeout(timeoutHandle);\r\n this.pending.delete(id);\r\n reject(error instanceof Error ? error : new Error(String(error)));\r\n });\r\n });\r\n }\r\n\r\n async sendResponse(response: GatewayResponse): Promise<void> {\r\n const payload = JSON.stringify(response);\r\n await this.socket.send(payload);\r\n }\r\n\r\n async acknowledge(id: number, status = 200): Promise<void> {\r\n await this.sendResponse({ id, status, body: null });\r\n }\r\n\r\n private onRawMessage(raw: string): void {\r\n this.logger.debug({ raw }, 'Received websocket frame.');\r\n let envelope: GatewayResponse | WebsocketEventEnvelope;\r\n try {\r\n envelope = deserializeGatewayResponse(raw);\r\n } catch (error) {\r\n this.logger.error({ err: error, raw }, 'Failed to parse gateway message.');\r\n return;\r\n }\r\n\r\n if (typeof envelope.id !== 'number') {\r\n this.logger.warn({ envelope }, 'Invalid gateway message without id.');\r\n return;\r\n }\r\n\r\n const pending = this.pending.get(envelope.id);\r\n if (pending) {\r\n clearTimeout(pending.timeoutHandle);\r\n this.pending.delete(envelope.id);\r\n if ('status' in envelope && envelope.status === 403) {\r\n pending.reject(new AuthenticationError('Gateway rejected the request with 403.'));\r\n return;\r\n }\r\n pending.resolve(envelope.body as unknown);\r\n return;\r\n }\r\n\r\n this.emitter.emit('event', envelope as WebsocketEventEnvelope);\r\n }\r\n\r\n dispose(): void {\r\n for (const [id, pending] of this.pending.entries()) {\r\n clearTimeout(pending.timeoutHandle);\r\n pending.reject(new Error(`Request ${pending.method} ${pending.path} cancelled during shutdown.`));\r\n this.pending.delete(id);\r\n }\r\n this.emitter.removeAllListeners();\r\n }\r\n}\r\n"]}
|
package/dist/routing.cjs
ADDED
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.getDefaultRoutingKeyForMachine = getDefaultRoutingKeyForMachine;
|
|
4
|
+
const node_crypto_1 = require("node:crypto");
|
|
5
|
+
const node_os_1 = require("node:os");
|
|
6
|
+
function getMacAddressBytes() {
|
|
7
|
+
var _a, _b;
|
|
8
|
+
const interfaces = (0, node_os_1.networkInterfaces)();
|
|
9
|
+
const names = Object.keys(interfaces).sort((a, b) => a.localeCompare(b));
|
|
10
|
+
for (const name of names) {
|
|
11
|
+
const entries = interfaces[name];
|
|
12
|
+
if (!entries) {
|
|
13
|
+
continue;
|
|
14
|
+
}
|
|
15
|
+
for (const entry of entries) {
|
|
16
|
+
const mac = (_b = (_a = entry.mac) === null || _a === void 0 ? void 0 : _a.toLowerCase()) !== null && _b !== void 0 ? _b : '';
|
|
17
|
+
if (!mac || mac === '00:00:00:00:00:00') {
|
|
18
|
+
continue;
|
|
19
|
+
}
|
|
20
|
+
const sanitized = mac.replace(/:/g, '');
|
|
21
|
+
if (sanitized.length % 2 !== 0) {
|
|
22
|
+
continue;
|
|
23
|
+
}
|
|
24
|
+
const macBytes = Buffer.from(sanitized, 'hex');
|
|
25
|
+
if (macBytes.length > 0) {
|
|
26
|
+
return macBytes;
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
throw new Error('No network interface with a valid MAC address was found.');
|
|
31
|
+
}
|
|
32
|
+
function getDefaultRoutingKeyForMachine() {
|
|
33
|
+
const macBytes = getMacAddressBytes();
|
|
34
|
+
const hash = (0, node_crypto_1.createHash)('md5').update(macBytes).digest('hex');
|
|
35
|
+
const key = `${(0, node_os_1.hostname)()}_${hash}`
|
|
36
|
+
.replace(/[: ,]/g, '_')
|
|
37
|
+
.toLowerCase();
|
|
38
|
+
return key;
|
|
39
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"routing.d.ts","sourceRoot":"","sources":["../src/routing.ts"],"names":[],"mappings":"AAkCA,wBAAgB,8BAA8B,IAAI,MAAM,CAOvD"}
|
package/dist/routing.js
ADDED
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { createHash } from 'node:crypto';
|
|
2
|
+
import { hostname, networkInterfaces } from 'node:os';
|
|
3
|
+
function getMacAddressBytes() {
|
|
4
|
+
const interfaces = networkInterfaces();
|
|
5
|
+
const names = Object.keys(interfaces).sort((a, b) => a.localeCompare(b));
|
|
6
|
+
for (const name of names) {
|
|
7
|
+
const entries = interfaces[name];
|
|
8
|
+
if (!entries) {
|
|
9
|
+
continue;
|
|
10
|
+
}
|
|
11
|
+
for (const entry of entries) {
|
|
12
|
+
const mac = entry.mac?.toLowerCase() ?? '';
|
|
13
|
+
if (!mac || mac === '00:00:00:00:00:00') {
|
|
14
|
+
continue;
|
|
15
|
+
}
|
|
16
|
+
const sanitized = mac.replace(/:/g, '');
|
|
17
|
+
if (sanitized.length % 2 !== 0) {
|
|
18
|
+
continue;
|
|
19
|
+
}
|
|
20
|
+
const macBytes = Buffer.from(sanitized, 'hex');
|
|
21
|
+
if (macBytes.length > 0) {
|
|
22
|
+
return macBytes;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
throw new Error('No network interface with a valid MAC address was found.');
|
|
27
|
+
}
|
|
28
|
+
export function getDefaultRoutingKeyForMachine() {
|
|
29
|
+
const macBytes = getMacAddressBytes();
|
|
30
|
+
const hash = createHash('md5').update(macBytes).digest('hex');
|
|
31
|
+
const key = `${hostname()}_${hash}`
|
|
32
|
+
.replace(/[: ,]/g, '_')
|
|
33
|
+
.toLowerCase();
|
|
34
|
+
return key;
|
|
35
|
+
}
|
|
36
|
+
//# sourceMappingURL=routing.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"routing.js","sourceRoot":"","sources":["../src/routing.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AACzC,OAAO,EAAE,QAAQ,EAAE,iBAAiB,EAAE,MAAM,SAAS,CAAC;AAEtD,SAAS,kBAAkB;IACzB,MAAM,UAAU,GAAG,iBAAiB,EAAE,CAAC;IACvC,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;IAEzE,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,MAAM,OAAO,GAAG,UAAU,CAAC,IAAI,CAAC,CAAC;QACjC,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,SAAS;QACX,CAAC;QAED,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;YAC5B,MAAM,GAAG,GAAG,KAAK,CAAC,GAAG,EAAE,WAAW,EAAE,IAAI,EAAE,CAAC;YAC3C,IAAI,CAAC,GAAG,IAAI,GAAG,KAAK,mBAAmB,EAAE,CAAC;gBACxC,SAAS;YACX,CAAC;YAED,MAAM,SAAS,GAAG,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;YACxC,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC/B,SAAS;YACX,CAAC;YAED,MAAM,QAAQ,GAAG,MAAM,CAAC,IAAI,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;YAC/C,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACxB,OAAO,QAAQ,CAAC;YAClB,CAAC;QACH,CAAC;IACH,CAAC;IAED,MAAM,IAAI,KAAK,CAAC,0DAA0D,CAAC,CAAC;AAC9E,CAAC;AAED,MAAM,UAAU,8BAA8B;IAC5C,MAAM,QAAQ,GAAG,kBAAkB,EAAE,CAAC;IACtC,MAAM,IAAI,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IAC9D,MAAM,GAAG,GAAG,GAAG,QAAQ,EAAE,IAAI,IAAI,EAAE;SAChC,OAAO,CAAC,QAAQ,EAAE,GAAG,CAAC;SACtB,WAAW,EAAE,CAAC;IACjB,OAAO,GAAG,CAAC;AACb,CAAC","sourcesContent":["import { createHash } from 'node:crypto';\r\nimport { hostname, networkInterfaces } from 'node:os';\r\n\r\nfunction getMacAddressBytes(): Buffer {\r\n const interfaces = networkInterfaces();\r\n const names = Object.keys(interfaces).sort((a, b) => a.localeCompare(b));\r\n\r\n for (const name of names) {\r\n const entries = interfaces[name];\r\n if (!entries) {\r\n continue;\r\n }\r\n\r\n for (const entry of entries) {\r\n const mac = entry.mac?.toLowerCase() ?? '';\r\n if (!mac || mac === '00:00:00:00:00:00') {\r\n continue;\r\n }\r\n\r\n const sanitized = mac.replace(/:/g, '');\r\n if (sanitized.length % 2 !== 0) {\r\n continue;\r\n }\r\n\r\n const macBytes = Buffer.from(sanitized, 'hex');\r\n if (macBytes.length > 0) {\r\n return macBytes;\r\n }\r\n }\r\n }\r\n\r\n throw new Error('No network interface with a valid MAC address was found.');\r\n}\r\n\r\nexport function getDefaultRoutingKeyForMachine(): string {\r\n const macBytes = getMacAddressBytes();\r\n const hash = createHash('md5').update(macBytes).digest('hex');\r\n const key = `${hostname()}_${hash}`\r\n .replace(/[: ,]/g, '_')\r\n .toLowerCase();\r\n return key;\r\n}\r\n"]}
|