velocious 1.0.177 → 1.0.179
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/README.md +125 -0
- package/build/src/background-jobs/client.d.ts +23 -0
- package/build/src/background-jobs/client.d.ts.map +1 -0
- package/build/src/background-jobs/client.js +58 -0
- package/build/src/background-jobs/job-record.d.ts +4 -0
- package/build/src/background-jobs/job-record.d.ts.map +1 -0
- package/build/src/background-jobs/job-record.js +11 -0
- package/build/src/background-jobs/job-registry.d.ts +23 -0
- package/build/src/background-jobs/job-registry.d.ts.map +1 -0
- package/build/src/background-jobs/job-registry.js +55 -0
- package/build/src/background-jobs/job-runner.d.ts +6 -0
- package/build/src/background-jobs/job-runner.d.ts.map +1 -0
- package/build/src/background-jobs/job-runner.js +44 -0
- package/build/src/background-jobs/job.d.ts +35 -0
- package/build/src/background-jobs/job.d.ts.map +1 -0
- package/build/src/background-jobs/job.js +61 -0
- package/build/src/background-jobs/json-socket.d.ts +27 -0
- package/build/src/background-jobs/json-socket.d.ts.map +1 -0
- package/build/src/background-jobs/json-socket.js +55 -0
- package/build/src/background-jobs/main.d.ts +62 -0
- package/build/src/background-jobs/main.d.ts.map +1 -0
- package/build/src/background-jobs/main.js +216 -0
- package/build/src/background-jobs/status-reporter.d.ts +54 -0
- package/build/src/background-jobs/status-reporter.d.ts.map +1 -0
- package/build/src/background-jobs/status-reporter.js +113 -0
- package/build/src/background-jobs/store.d.ts +237 -0
- package/build/src/background-jobs/store.d.ts.map +1 -0
- package/build/src/background-jobs/store.js +488 -0
- package/build/src/background-jobs/types.d.ts +17 -0
- package/build/src/background-jobs/types.d.ts.map +1 -0
- package/build/src/background-jobs/types.js +8 -0
- package/build/src/background-jobs/worker.d.ts +64 -0
- package/build/src/background-jobs/worker.d.ts.map +1 -0
- package/build/src/background-jobs/worker.js +155 -0
- package/build/src/cli/commands/background-jobs-main.d.ts +5 -0
- package/build/src/cli/commands/background-jobs-main.d.ts.map +1 -0
- package/build/src/cli/commands/background-jobs-main.js +7 -0
- package/build/src/cli/commands/background-jobs-runner.d.ts +5 -0
- package/build/src/cli/commands/background-jobs-runner.d.ts.map +1 -0
- package/build/src/cli/commands/background-jobs-runner.js +7 -0
- package/build/src/cli/commands/background-jobs-worker.d.ts +5 -0
- package/build/src/cli/commands/background-jobs-worker.d.ts.map +1 -0
- package/build/src/cli/commands/background-jobs-worker.js +7 -0
- package/build/src/configuration-types.d.ts +25 -0
- package/build/src/configuration-types.d.ts.map +1 -1
- package/build/src/configuration-types.js +8 -1
- package/build/src/configuration.d.ts +11 -1
- package/build/src/configuration.d.ts.map +1 -1
- package/build/src/configuration.js +26 -2
- package/build/src/database/drivers/mssql/sql/update.js +3 -3
- package/build/src/database/drivers/mysql/sql/update.js +3 -3
- package/build/src/database/drivers/pgsql/sql/update.js +3 -3
- package/build/src/database/drivers/sqlite/sql/update.js +3 -3
- package/build/src/database/query/update-base.d.ts +5 -0
- package/build/src/database/query/update-base.d.ts.map +1 -1
- package/build/src/database/query/update-base.js +10 -1
- package/build/src/database/query-parser/limit-parser.d.ts.map +1 -1
- package/build/src/database/query-parser/limit-parser.js +8 -7
- package/build/src/environment-handlers/base.d.ts +15 -0
- package/build/src/environment-handlers/base.d.ts.map +1 -1
- package/build/src/environment-handlers/base.js +22 -1
- package/build/src/environment-handlers/browser.d.ts.map +1 -1
- package/build/src/environment-handlers/browser.js +10 -1
- package/build/src/environment-handlers/node/cli/commands/background-jobs-main.d.ts +5 -0
- package/build/src/environment-handlers/node/cli/commands/background-jobs-main.d.ts.map +1 -0
- package/build/src/environment-handlers/node/cli/commands/background-jobs-main.js +18 -0
- package/build/src/environment-handlers/node/cli/commands/background-jobs-runner.d.ts +5 -0
- package/build/src/environment-handlers/node/cli/commands/background-jobs-runner.d.ts.map +1 -0
- package/build/src/environment-handlers/node/cli/commands/background-jobs-runner.js +13 -0
- package/build/src/environment-handlers/node/cli/commands/background-jobs-worker.d.ts +5 -0
- package/build/src/environment-handlers/node/cli/commands/background-jobs-worker.d.ts.map +1 -0
- package/build/src/environment-handlers/node/cli/commands/background-jobs-worker.js +18 -0
- package/build/src/environment-handlers/node/cli/commands/generate/base-models.d.ts.map +1 -1
- package/build/src/environment-handlers/node/cli/commands/generate/base-models.js +6 -4
- package/build/src/environment-handlers/node/cli/commands/test.d.ts.map +1 -1
- package/build/src/environment-handlers/node/cli/commands/test.js +2 -116
- package/build/src/environment-handlers/node.d.ts.map +1 -1
- package/build/src/environment-handlers/node.js +25 -1
- package/build/src/testing/browser-test-app.d.ts +2 -0
- package/build/src/testing/browser-test-app.d.ts.map +1 -0
- package/build/src/testing/browser-test-app.js +24 -0
- package/build/src/testing/test-filter-parser.d.ts +16 -0
- package/build/src/testing/test-filter-parser.d.ts.map +1 -0
- package/build/src/testing/test-filter-parser.js +117 -0
- package/build/src/testing/test-runner.d.ts +35 -0
- package/build/src/testing/test-runner.d.ts.map +1 -1
- package/build/src/testing/test-runner.js +100 -17
- package/build/tsconfig.tsbuildinfo +1 -1
- package/package.json +4 -1
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
export default class BackgroundJobsMain {
|
|
2
|
+
/**
|
|
3
|
+
* @param {object} args - Options.
|
|
4
|
+
* @param {import("../configuration.js").default} args.configuration - Configuration.
|
|
5
|
+
* @param {string} [args.host] - Hostname.
|
|
6
|
+
* @param {number} [args.port] - Port.
|
|
7
|
+
*/
|
|
8
|
+
constructor({ configuration, host, port }: {
|
|
9
|
+
configuration: import("../configuration.js").default;
|
|
10
|
+
host?: string;
|
|
11
|
+
port?: number;
|
|
12
|
+
});
|
|
13
|
+
configuration: import("../configuration.js").default;
|
|
14
|
+
host: string;
|
|
15
|
+
port: number;
|
|
16
|
+
store: BackgroundJobsStore;
|
|
17
|
+
logger: Logger;
|
|
18
|
+
/** @type {Set<JsonSocket>} */
|
|
19
|
+
workers: Set<JsonSocket>;
|
|
20
|
+
/** @type {Set<JsonSocket>} */
|
|
21
|
+
readyWorkers: Set<JsonSocket>;
|
|
22
|
+
_dispatching: boolean;
|
|
23
|
+
/**
|
|
24
|
+
* @returns {Promise<void>} - Resolves when listening.
|
|
25
|
+
*/
|
|
26
|
+
start(): Promise<void>;
|
|
27
|
+
server: net.Server;
|
|
28
|
+
_dispatchTimer: NodeJS.Timeout;
|
|
29
|
+
_orphanTimer: NodeJS.Timeout;
|
|
30
|
+
/**
|
|
31
|
+
* @returns {Promise<void>} - Resolves when closed.
|
|
32
|
+
*/
|
|
33
|
+
stop(): Promise<void>;
|
|
34
|
+
/**
|
|
35
|
+
* @returns {number} - Bound port.
|
|
36
|
+
*/
|
|
37
|
+
getPort(): number;
|
|
38
|
+
/**
|
|
39
|
+
* @param {import("net").Socket} socket - Socket.
|
|
40
|
+
* @returns {void}
|
|
41
|
+
*/
|
|
42
|
+
_handleConnection(socket: import("net").Socket): void;
|
|
43
|
+
_handleEnqueue({ jsonSocket, message }: {
|
|
44
|
+
jsonSocket: any;
|
|
45
|
+
message: any;
|
|
46
|
+
}): Promise<void>;
|
|
47
|
+
_handleJobComplete({ jsonSocket, message }: {
|
|
48
|
+
jsonSocket: any;
|
|
49
|
+
message: any;
|
|
50
|
+
}): Promise<void>;
|
|
51
|
+
_handleJobFailed({ jsonSocket, message }: {
|
|
52
|
+
jsonSocket: any;
|
|
53
|
+
message: any;
|
|
54
|
+
}): Promise<void>;
|
|
55
|
+
_dispatch(): Promise<void>;
|
|
56
|
+
_sweepOrphans(): Promise<void>;
|
|
57
|
+
}
|
|
58
|
+
import BackgroundJobsStore from "./store.js";
|
|
59
|
+
import { Logger } from "../logger.js";
|
|
60
|
+
import JsonSocket from "./json-socket.js";
|
|
61
|
+
import net from "net";
|
|
62
|
+
//# sourceMappingURL=main.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"main.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/main.js"],"names":[],"mappings":"AAOA;IACE;;;;;OAKG;IACH,2CAJG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QACvB,IAAI,GAAlB,MAAM;QACQ,IAAI,GAAlB,MAAM;KAChB,EAaA;IAXC,qDAAkC;IAElC,aAA+B;IAC/B,aAAyD;IACzD,2BAAoG;IACpG,eAA8B;IAC9B,8BAA8B;IAC9B,SADW,GAAG,CAAC,UAAU,CAAC,CACF;IACxB,8BAA8B;IAC9B,cADW,GAAG,CAAC,UAAU,CAAC,CACG;IAC7B,sBAAyB;IAG3B;;OAEG;IACH,SAFa,OAAO,CAAC,IAAI,CAAC,CAyBzB;IAnBC,mBAA0E;IAY1E,+BAEQ;IAER,6BAES;IAGX;;OAEG;IACH,QAFa,OAAO,CAAC,IAAI,CAAC,CAazB;IAED;;OAEG;IACH,WAFa,MAAM,CAIlB;IAED;;;OAGG;IACH,0BAHW,OAAO,KAAK,EAAE,MAAM,GAClB,IAAI,CAqDhB;IAED;;;sBAcC;IAED;;;sBAYC;IAED;;;sBAcC;IAED,2BAyCC;IAED,+BAUC;CACF;gCAxO+B,YAAY;uBACvB,cAAc;uBAFZ,kBAAkB;gBADzB,KAAK"}
|
|
@@ -0,0 +1,216 @@
|
|
|
1
|
+
// @ts-check
|
|
2
|
+
import net from "net";
|
|
3
|
+
import JsonSocket from "./json-socket.js";
|
|
4
|
+
import BackgroundJobsStore from "./store.js";
|
|
5
|
+
import { Logger } from "../logger.js";
|
|
6
|
+
export default class BackgroundJobsMain {
|
|
7
|
+
/**
|
|
8
|
+
* @param {object} args - Options.
|
|
9
|
+
* @param {import("../configuration.js").default} args.configuration - Configuration.
|
|
10
|
+
* @param {string} [args.host] - Hostname.
|
|
11
|
+
* @param {number} [args.port] - Port.
|
|
12
|
+
*/
|
|
13
|
+
constructor({ configuration, host, port }) {
|
|
14
|
+
this.configuration = configuration;
|
|
15
|
+
const config = configuration.getBackgroundJobsConfig();
|
|
16
|
+
this.host = host || config.host;
|
|
17
|
+
this.port = typeof port === "number" ? port : config.port;
|
|
18
|
+
this.store = new BackgroundJobsStore({ configuration, databaseIdentifier: config.databaseIdentifier });
|
|
19
|
+
this.logger = new Logger(this);
|
|
20
|
+
/** @type {Set<JsonSocket>} */
|
|
21
|
+
this.workers = new Set();
|
|
22
|
+
/** @type {Set<JsonSocket>} */
|
|
23
|
+
this.readyWorkers = new Set();
|
|
24
|
+
this._dispatching = false;
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* @returns {Promise<void>} - Resolves when listening.
|
|
28
|
+
*/
|
|
29
|
+
async start() {
|
|
30
|
+
this.configuration.setCurrent();
|
|
31
|
+
await this.configuration.initialize({ type: "background-jobs-main" });
|
|
32
|
+
await this.store.ensureReady();
|
|
33
|
+
this.server = net.createServer((socket) => this._handleConnection(socket));
|
|
34
|
+
await new Promise((resolve, reject) => {
|
|
35
|
+
this.server.once("error", reject);
|
|
36
|
+
this.server.listen(this.port, this.host, () => resolve(undefined));
|
|
37
|
+
});
|
|
38
|
+
const address = this.server.address();
|
|
39
|
+
if (address && typeof address === "object") {
|
|
40
|
+
this.port = address.port;
|
|
41
|
+
}
|
|
42
|
+
this._dispatchTimer = setInterval(() => {
|
|
43
|
+
void this._dispatch();
|
|
44
|
+
}, 1000);
|
|
45
|
+
this._orphanTimer = setInterval(() => {
|
|
46
|
+
void this._sweepOrphans();
|
|
47
|
+
}, 60000);
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* @returns {Promise<void>} - Resolves when closed.
|
|
51
|
+
*/
|
|
52
|
+
async stop() {
|
|
53
|
+
for (const worker of this.workers) {
|
|
54
|
+
worker.close();
|
|
55
|
+
}
|
|
56
|
+
if (this._dispatchTimer)
|
|
57
|
+
clearInterval(this._dispatchTimer);
|
|
58
|
+
if (this._orphanTimer)
|
|
59
|
+
clearInterval(this._orphanTimer);
|
|
60
|
+
if (!this.server)
|
|
61
|
+
return;
|
|
62
|
+
await new Promise((resolve) => this.server.close(() => resolve(undefined)));
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* @returns {number} - Bound port.
|
|
66
|
+
*/
|
|
67
|
+
getPort() {
|
|
68
|
+
return this.port;
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* @param {import("net").Socket} socket - Socket.
|
|
72
|
+
* @returns {void}
|
|
73
|
+
*/
|
|
74
|
+
_handleConnection(socket) {
|
|
75
|
+
const jsonSocket = new JsonSocket(socket);
|
|
76
|
+
let role = null;
|
|
77
|
+
const cleanup = () => {
|
|
78
|
+
if (role === "worker") {
|
|
79
|
+
this.workers.delete(jsonSocket);
|
|
80
|
+
this.readyWorkers.delete(jsonSocket);
|
|
81
|
+
}
|
|
82
|
+
};
|
|
83
|
+
jsonSocket.on("close", cleanup);
|
|
84
|
+
jsonSocket.on("error", (error) => {
|
|
85
|
+
this.logger.warn(() => ["Background jobs connection error:", error]);
|
|
86
|
+
cleanup();
|
|
87
|
+
});
|
|
88
|
+
jsonSocket.on("message", (message) => {
|
|
89
|
+
if (!role && message?.type === "hello") {
|
|
90
|
+
role = message.role;
|
|
91
|
+
if (role === "worker") {
|
|
92
|
+
jsonSocket.workerId = message.workerId;
|
|
93
|
+
this.workers.add(jsonSocket);
|
|
94
|
+
this.readyWorkers.add(jsonSocket);
|
|
95
|
+
this._dispatch();
|
|
96
|
+
}
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
if (role === "client" && message?.type === "enqueue") {
|
|
100
|
+
this._handleEnqueue({ jsonSocket, message });
|
|
101
|
+
return;
|
|
102
|
+
}
|
|
103
|
+
if (role === "worker" && message?.type === "ready") {
|
|
104
|
+
this.readyWorkers.add(jsonSocket);
|
|
105
|
+
this._dispatch();
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
if ((role === "worker" || role === "reporter") && message?.type === "job-complete") {
|
|
109
|
+
this._handleJobComplete({ jsonSocket, message });
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
if ((role === "worker" || role === "reporter") && message?.type === "job-failed") {
|
|
113
|
+
this._handleJobFailed({ jsonSocket, message });
|
|
114
|
+
}
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
async _handleEnqueue({ jsonSocket, message }) {
|
|
118
|
+
try {
|
|
119
|
+
const jobId = await this.store.enqueue({
|
|
120
|
+
jobName: message.jobName,
|
|
121
|
+
args: message.args || [],
|
|
122
|
+
options: message.options || {}
|
|
123
|
+
});
|
|
124
|
+
jsonSocket.send({ type: "enqueued", jobId });
|
|
125
|
+
await this._dispatch();
|
|
126
|
+
}
|
|
127
|
+
catch (error) {
|
|
128
|
+
this.logger.error(() => ["Failed to enqueue background job:", error]);
|
|
129
|
+
jsonSocket.send({ type: "enqueue-error", error: "Failed to enqueue job" });
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
async _handleJobComplete({ jsonSocket, message }) {
|
|
133
|
+
try {
|
|
134
|
+
await this.store.markCompleted({
|
|
135
|
+
jobId: message.jobId,
|
|
136
|
+
workerId: message.workerId,
|
|
137
|
+
handedOffAtMs: message.handedOffAtMs
|
|
138
|
+
});
|
|
139
|
+
jsonSocket.send({ type: "job-updated", jobId: message.jobId });
|
|
140
|
+
}
|
|
141
|
+
catch (error) {
|
|
142
|
+
this.logger.error(() => ["Failed to update job completion:", error]);
|
|
143
|
+
jsonSocket.send({ type: "job-update-error", jobId: message.jobId, error: "Failed to update job" });
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
async _handleJobFailed({ jsonSocket, message }) {
|
|
147
|
+
try {
|
|
148
|
+
await this.store.markFailed({
|
|
149
|
+
jobId: message.jobId,
|
|
150
|
+
error: message.error,
|
|
151
|
+
workerId: message.workerId,
|
|
152
|
+
handedOffAtMs: message.handedOffAtMs
|
|
153
|
+
});
|
|
154
|
+
jsonSocket.send({ type: "job-updated", jobId: message.jobId });
|
|
155
|
+
await this._dispatch();
|
|
156
|
+
}
|
|
157
|
+
catch (error) {
|
|
158
|
+
this.logger.error(() => ["Failed to update job failure:", error]);
|
|
159
|
+
jsonSocket.send({ type: "job-update-error", jobId: message.jobId, error: "Failed to update job" });
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
async _dispatch() {
|
|
163
|
+
if (this._dispatching)
|
|
164
|
+
return;
|
|
165
|
+
if (this.readyWorkers.size === 0)
|
|
166
|
+
return;
|
|
167
|
+
this._dispatching = true;
|
|
168
|
+
try {
|
|
169
|
+
while (this.readyWorkers.size > 0) {
|
|
170
|
+
const job = await this.store.nextAvailableJob();
|
|
171
|
+
if (!job)
|
|
172
|
+
return;
|
|
173
|
+
const [worker] = this.readyWorkers;
|
|
174
|
+
if (!worker)
|
|
175
|
+
return;
|
|
176
|
+
this.readyWorkers.delete(worker);
|
|
177
|
+
const handedOffAtMs = await this.store.markHandedOff({ jobId: job.id, workerId: worker.workerId });
|
|
178
|
+
try {
|
|
179
|
+
worker.send({
|
|
180
|
+
type: "job",
|
|
181
|
+
payload: {
|
|
182
|
+
id: job.id,
|
|
183
|
+
jobName: job.jobName,
|
|
184
|
+
args: job.args,
|
|
185
|
+
workerId: worker.workerId,
|
|
186
|
+
handedOffAtMs,
|
|
187
|
+
options: {
|
|
188
|
+
forked: job.forked
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
});
|
|
192
|
+
}
|
|
193
|
+
catch (error) {
|
|
194
|
+
this.logger.warn(() => ["Failed to send job to worker, re-queueing:", error]);
|
|
195
|
+
await this.store.markReturnedToQueue({ jobId: job.id });
|
|
196
|
+
this.readyWorkers.add(worker);
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
finally {
|
|
201
|
+
this._dispatching = false;
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
async _sweepOrphans() {
|
|
205
|
+
try {
|
|
206
|
+
const count = await this.store.markOrphanedJobs();
|
|
207
|
+
if (count > 0) {
|
|
208
|
+
this.logger.warn(() => ["Marked orphaned background jobs", count]);
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
catch (error) {
|
|
212
|
+
this.logger.error(() => ["Failed to mark orphaned jobs:", error]);
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"main.js","sourceRoot":"","sources":["../../../src/background-jobs/main.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport net from \"net\"\nimport JsonSocket from \"./json-socket.js\"\nimport BackgroundJobsStore from \"./store.js\"\nimport {Logger} from \"../logger.js\"\n\nexport default class BackgroundJobsMain {\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration.\n   * @param {string} [args.host] - Hostname.\n   * @param {number} [args.port] - Port.\n   */\n  constructor({configuration, host, port}) {\n    this.configuration = configuration\n    const config = configuration.getBackgroundJobsConfig()\n    this.host = host || config.host\n    this.port = typeof port === \"number\" ? port : config.port\n    this.store = new BackgroundJobsStore({configuration, databaseIdentifier: config.databaseIdentifier})\n    this.logger = new Logger(this)\n    /** @type {Set<JsonSocket>} */\n    this.workers = new Set()\n    /** @type {Set<JsonSocket>} */\n    this.readyWorkers = new Set()\n    this._dispatching = false\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when listening.\n   */\n  async start() {\n    this.configuration.setCurrent()\n    await this.configuration.initialize({type: \"background-jobs-main\"})\n    await this.store.ensureReady()\n    this.server = net.createServer((socket) => this._handleConnection(socket))\n\n    await new Promise((resolve, reject) => {\n      this.server.once(\"error\", reject)\n      this.server.listen(this.port, this.host, () => resolve(undefined))\n    })\n\n    const address = this.server.address()\n    if (address && typeof address === \"object\") {\n      this.port = address.port\n    }\n\n    this._dispatchTimer = setInterval(() => {\n      void this._dispatch()\n    }, 1000)\n\n    this._orphanTimer = setInterval(() => {\n      void this._sweepOrphans()\n    }, 60000)\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when closed.\n   */\n  async stop() {\n    for (const worker of this.workers) {\n      worker.close()\n    }\n\n    if (this._dispatchTimer) clearInterval(this._dispatchTimer)\n    if (this._orphanTimer) clearInterval(this._orphanTimer)\n\n    if (!this.server) return\n\n    await new Promise((resolve) => this.server.close(() => resolve(undefined)))\n  }\n\n  /**\n   * @returns {number} - Bound port.\n   */\n  getPort() {\n    return this.port\n  }\n\n  /**\n   * @param {import(\"net\").Socket} socket - Socket.\n   * @returns {void}\n   */\n  _handleConnection(socket) {\n    const jsonSocket = new JsonSocket(socket)\n    let role = null\n\n    const cleanup = () => {\n      if (role === \"worker\") {\n        this.workers.delete(jsonSocket)\n        this.readyWorkers.delete(jsonSocket)\n      }\n    }\n\n    jsonSocket.on(\"close\", cleanup)\n    jsonSocket.on(\"error\", (error) => {\n      this.logger.warn(() => [\"Background jobs connection error:\", error])\n      cleanup()\n    })\n\n    jsonSocket.on(\"message\", (message) => {\n      if (!role && message?.type === \"hello\") {\n        role = message.role\n\n        if (role === \"worker\") {\n          jsonSocket.workerId = message.workerId\n          this.workers.add(jsonSocket)\n          this.readyWorkers.add(jsonSocket)\n          this._dispatch()\n        }\n\n        return\n      }\n\n      if (role === \"client\" && message?.type === \"enqueue\") {\n        this._handleEnqueue({jsonSocket, message})\n        return\n      }\n\n      if (role === \"worker\" && message?.type === \"ready\") {\n        this.readyWorkers.add(jsonSocket)\n        this._dispatch()\n        return\n      }\n\n      if ((role === \"worker\" || role === \"reporter\") && message?.type === \"job-complete\") {\n        this._handleJobComplete({jsonSocket, message})\n        return\n      }\n\n      if ((role === \"worker\" || role === \"reporter\") && message?.type === \"job-failed\") {\n        this._handleJobFailed({jsonSocket, message})\n      }\n    })\n  }\n\n  async _handleEnqueue({jsonSocket, message}) {\n    try {\n      const jobId = await this.store.enqueue({\n        jobName: message.jobName,\n        args: message.args || [],\n        options: message.options || {}\n      })\n\n      jsonSocket.send({type: \"enqueued\", jobId})\n      await this._dispatch()\n    } catch (error) {\n      this.logger.error(() => [\"Failed to enqueue background job:\", error])\n      jsonSocket.send({type: \"enqueue-error\", error: \"Failed to enqueue job\"})\n    }\n  }\n\n  async _handleJobComplete({jsonSocket, message}) {\n    try {\n      await this.store.markCompleted({\n        jobId: message.jobId,\n        workerId: message.workerId,\n        handedOffAtMs: message.handedOffAtMs\n      })\n      jsonSocket.send({type: \"job-updated\", jobId: message.jobId})\n    } catch (error) {\n      this.logger.error(() => [\"Failed to update job completion:\", error])\n      jsonSocket.send({type: \"job-update-error\", jobId: message.jobId, error: \"Failed to update job\"})\n    }\n  }\n\n  async _handleJobFailed({jsonSocket, message}) {\n    try {\n      await this.store.markFailed({\n        jobId: message.jobId,\n        error: message.error,\n        workerId: message.workerId,\n        handedOffAtMs: message.handedOffAtMs\n      })\n      jsonSocket.send({type: \"job-updated\", jobId: message.jobId})\n      await this._dispatch()\n    } catch (error) {\n      this.logger.error(() => [\"Failed to update job failure:\", error])\n      jsonSocket.send({type: \"job-update-error\", jobId: message.jobId, error: \"Failed to update job\"})\n    }\n  }\n\n  async _dispatch() {\n    if (this._dispatching) return\n    if (this.readyWorkers.size === 0) return\n\n    this._dispatching = true\n\n    try {\n      while (this.readyWorkers.size > 0) {\n        const job = await this.store.nextAvailableJob()\n        if (!job) return\n\n        const [worker] = this.readyWorkers\n        if (!worker) return\n\n        this.readyWorkers.delete(worker)\n\n        const handedOffAtMs = await this.store.markHandedOff({jobId: job.id, workerId: worker.workerId})\n\n        try {\n          worker.send({\n            type: \"job\",\n            payload: {\n              id: job.id,\n              jobName: job.jobName,\n              args: job.args,\n              workerId: worker.workerId,\n              handedOffAtMs,\n              options: {\n                forked: job.forked\n              }\n            }\n          })\n        } catch (error) {\n          this.logger.warn(() => [\"Failed to send job to worker, re-queueing:\", error])\n          await this.store.markReturnedToQueue({jobId: job.id})\n          this.readyWorkers.add(worker)\n        }\n      }\n    } finally {\n      this._dispatching = false\n    }\n  }\n\n  async _sweepOrphans() {\n    try {\n      const count = await this.store.markOrphanedJobs()\n\n      if (count > 0) {\n        this.logger.warn(() => [\"Marked orphaned background jobs\", count])\n      }\n    } catch (error) {\n      this.logger.error(() => [\"Failed to mark orphaned jobs:\", error])\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
export default class BackgroundJobsStatusReporter {
|
|
2
|
+
/**
|
|
3
|
+
* @param {object} args - Options.
|
|
4
|
+
* @param {import("../configuration.js").default} args.configuration - Configuration.
|
|
5
|
+
* @param {string} [args.host] - Host.
|
|
6
|
+
* @param {number} [args.port] - Port.
|
|
7
|
+
*/
|
|
8
|
+
constructor({ configuration, host, port }: {
|
|
9
|
+
configuration: import("../configuration.js").default;
|
|
10
|
+
host?: string;
|
|
11
|
+
port?: number;
|
|
12
|
+
});
|
|
13
|
+
configuration: import("../configuration.js").default;
|
|
14
|
+
host: string;
|
|
15
|
+
port: number;
|
|
16
|
+
logger: Logger;
|
|
17
|
+
/**
|
|
18
|
+
* @param {object} args - Options.
|
|
19
|
+
* @param {string} args.jobId - Job id.
|
|
20
|
+
* @param {"completed" | "failed"} args.status - Status.
|
|
21
|
+
* @param {unknown} [args.error] - Error.
|
|
22
|
+
* @param {number} [args.handedOffAtMs] - Handed off timestamp.
|
|
23
|
+
* @param {string} [args.workerId] - Worker id.
|
|
24
|
+
* @returns {Promise<void>} - Resolves when reported.
|
|
25
|
+
*/
|
|
26
|
+
report({ jobId, status, error, handedOffAtMs, workerId }: {
|
|
27
|
+
jobId: string;
|
|
28
|
+
status: "completed" | "failed";
|
|
29
|
+
error?: unknown;
|
|
30
|
+
handedOffAtMs?: number;
|
|
31
|
+
workerId?: string;
|
|
32
|
+
}): Promise<void>;
|
|
33
|
+
/**
|
|
34
|
+
* @param {object} args - Options.
|
|
35
|
+
* @param {string} args.jobId - Job id.
|
|
36
|
+
* @param {"completed" | "failed"} args.status - Status.
|
|
37
|
+
* @param {unknown} [args.error] - Error.
|
|
38
|
+
* @param {number} [args.handedOffAtMs] - Handed off timestamp.
|
|
39
|
+
* @param {string} [args.workerId] - Worker id.
|
|
40
|
+
* @param {number} [args.maxDurationMs] - Max duration for retries.
|
|
41
|
+
* @returns {Promise<void>} - Resolves when reported.
|
|
42
|
+
*/
|
|
43
|
+
reportWithRetry({ jobId, status, error, handedOffAtMs, workerId, maxDurationMs }: {
|
|
44
|
+
jobId: string;
|
|
45
|
+
status: "completed" | "failed";
|
|
46
|
+
error?: unknown;
|
|
47
|
+
handedOffAtMs?: number;
|
|
48
|
+
workerId?: string;
|
|
49
|
+
maxDurationMs?: number;
|
|
50
|
+
}): Promise<void>;
|
|
51
|
+
_normalizeError(error: any): string;
|
|
52
|
+
}
|
|
53
|
+
import { Logger } from "../logger.js";
|
|
54
|
+
//# sourceMappingURL=status-reporter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"status-reporter.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/status-reporter.js"],"names":[],"mappings":"AAQA;IACE;;;;;OAKG;IACH,2CAJG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QACvB,IAAI,GAAlB,MAAM;QACQ,IAAI,GAAlB,MAAM;KAChB,EAMA;IAJC,qDAAkC;IAClC,aAAgB;IAChB,aAAgB;IAChB,eAA8B;IAGhC;;;;;;;;OAQG;IACH,0DAPG;QAAqB,KAAK,EAAlB,MAAM;QACuB,MAAM,EAAnC,WAAW,GAAG,QAAQ;QACP,KAAK,GAApB,OAAO;QACO,aAAa,GAA3B,MAAM;QACQ,QAAQ,GAAtB,MAAM;KACd,GAAU,OAAO,CAAC,IAAI,CAAC,CAgDzB;IAED;;;;;;;;;OASG;IACH,kFARG;QAAqB,KAAK,EAAlB,MAAM;QACuB,MAAM,EAAnC,WAAW,GAAG,QAAQ;QACP,KAAK,GAApB,OAAO;QACO,aAAa,GAA3B,MAAM;QACQ,QAAQ,GAAtB,MAAM;QACQ,aAAa,GAA3B,MAAM;KACd,GAAU,OAAO,CAAC,IAAI,CAAC,CAwBzB;IAED,oCASC;CACF;uBArHoB,cAAc"}
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
// @ts-check
|
|
2
|
+
import net from "net";
|
|
3
|
+
import timeout from "awaitery/build/timeout.js";
|
|
4
|
+
import wait from "awaitery/build/wait.js";
|
|
5
|
+
import JsonSocket from "./json-socket.js";
|
|
6
|
+
import { Logger } from "../logger.js";
|
|
7
|
+
export default class BackgroundJobsStatusReporter {
|
|
8
|
+
/**
|
|
9
|
+
* @param {object} args - Options.
|
|
10
|
+
* @param {import("../configuration.js").default} args.configuration - Configuration.
|
|
11
|
+
* @param {string} [args.host] - Host.
|
|
12
|
+
* @param {number} [args.port] - Port.
|
|
13
|
+
*/
|
|
14
|
+
constructor({ configuration, host, port }) {
|
|
15
|
+
this.configuration = configuration;
|
|
16
|
+
this.host = host;
|
|
17
|
+
this.port = port;
|
|
18
|
+
this.logger = new Logger(this);
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* @param {object} args - Options.
|
|
22
|
+
* @param {string} args.jobId - Job id.
|
|
23
|
+
* @param {"completed" | "failed"} args.status - Status.
|
|
24
|
+
* @param {unknown} [args.error] - Error.
|
|
25
|
+
* @param {number} [args.handedOffAtMs] - Handed off timestamp.
|
|
26
|
+
* @param {string} [args.workerId] - Worker id.
|
|
27
|
+
* @returns {Promise<void>} - Resolves when reported.
|
|
28
|
+
*/
|
|
29
|
+
async report({ jobId, status, error, handedOffAtMs, workerId }) {
|
|
30
|
+
const config = this.configuration.getBackgroundJobsConfig();
|
|
31
|
+
const host = this.host || config.host;
|
|
32
|
+
const port = typeof this.port === "number" ? this.port : config.port;
|
|
33
|
+
await timeout({ timeout: 5000 }, async () => {
|
|
34
|
+
await new Promise((resolve, reject) => {
|
|
35
|
+
const socket = net.createConnection({ host, port });
|
|
36
|
+
const jsonSocket = new JsonSocket(socket);
|
|
37
|
+
const cleanup = () => {
|
|
38
|
+
jsonSocket.removeAllListeners();
|
|
39
|
+
};
|
|
40
|
+
jsonSocket.on("error", (err) => {
|
|
41
|
+
cleanup();
|
|
42
|
+
reject(err);
|
|
43
|
+
});
|
|
44
|
+
jsonSocket.on("message", (message) => {
|
|
45
|
+
if (message?.type === "job-updated" && message.jobId === jobId) {
|
|
46
|
+
cleanup();
|
|
47
|
+
jsonSocket.close();
|
|
48
|
+
resolve(undefined);
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
if (message?.type === "job-update-error" && message.jobId === jobId) {
|
|
52
|
+
cleanup();
|
|
53
|
+
jsonSocket.close();
|
|
54
|
+
reject(new Error(message.error || "Job update failed"));
|
|
55
|
+
}
|
|
56
|
+
});
|
|
57
|
+
socket.on("connect", () => {
|
|
58
|
+
jsonSocket.send({ type: "hello", role: "reporter" });
|
|
59
|
+
jsonSocket.send({
|
|
60
|
+
type: status === "completed" ? "job-complete" : "job-failed",
|
|
61
|
+
jobId,
|
|
62
|
+
workerId,
|
|
63
|
+
handedOffAtMs,
|
|
64
|
+
error: error ? this._normalizeError(error) : undefined
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
});
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* @param {object} args - Options.
|
|
72
|
+
* @param {string} args.jobId - Job id.
|
|
73
|
+
* @param {"completed" | "failed"} args.status - Status.
|
|
74
|
+
* @param {unknown} [args.error] - Error.
|
|
75
|
+
* @param {number} [args.handedOffAtMs] - Handed off timestamp.
|
|
76
|
+
* @param {string} [args.workerId] - Worker id.
|
|
77
|
+
* @param {number} [args.maxDurationMs] - Max duration for retries.
|
|
78
|
+
* @returns {Promise<void>} - Resolves when reported.
|
|
79
|
+
*/
|
|
80
|
+
async reportWithRetry({ jobId, status, error, handedOffAtMs, workerId, maxDurationMs }) {
|
|
81
|
+
let attempt = 0;
|
|
82
|
+
const startTime = Date.now();
|
|
83
|
+
while (true) {
|
|
84
|
+
try {
|
|
85
|
+
await this.report({ jobId, status, error, handedOffAtMs, workerId });
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
catch (err) {
|
|
89
|
+
attempt += 1;
|
|
90
|
+
const delaySeconds = Math.min(30, 0.5 * attempt);
|
|
91
|
+
this.logger.debug(() => ["Background job status report failed, retrying", err]);
|
|
92
|
+
if (maxDurationMs && Date.now() - startTime >= maxDurationMs) {
|
|
93
|
+
this.logger.warn(() => ["Background job status report timed out, giving up", err]);
|
|
94
|
+
return;
|
|
95
|
+
}
|
|
96
|
+
await wait(delaySeconds);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
_normalizeError(error) {
|
|
101
|
+
if (error instanceof Error)
|
|
102
|
+
return error.stack || error.message;
|
|
103
|
+
if (typeof error === "string")
|
|
104
|
+
return error;
|
|
105
|
+
try {
|
|
106
|
+
return JSON.stringify(error);
|
|
107
|
+
}
|
|
108
|
+
catch {
|
|
109
|
+
return String(error);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"status-reporter.js","sourceRoot":"","sources":["../../../src/background-jobs/status-reporter.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport net from \"net\"\nimport timeout from \"awaitery/build/timeout.js\"\nimport wait from \"awaitery/build/wait.js\"\nimport JsonSocket from \"./json-socket.js\"\nimport {Logger} from \"../logger.js\"\n\nexport default class BackgroundJobsStatusReporter {\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration.\n   * @param {string} [args.host] - Host.\n   * @param {number} [args.port] - Port.\n   */\n  constructor({configuration, host, port}) {\n    this.configuration = configuration\n    this.host = host\n    this.port = port\n    this.logger = new Logger(this)\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.jobId - Job id.\n   * @param {\"completed\" | \"failed\"} args.status - Status.\n   * @param {unknown} [args.error] - Error.\n   * @param {number} [args.handedOffAtMs] - Handed off timestamp.\n   * @param {string} [args.workerId] - Worker id.\n   * @returns {Promise<void>} - Resolves when reported.\n   */\n  async report({jobId, status, error, handedOffAtMs, workerId}) {\n    const config = this.configuration.getBackgroundJobsConfig()\n    const host = this.host || config.host\n    const port = typeof this.port === \"number\" ? this.port : config.port\n\n    await timeout({timeout: 5000}, async () => {\n      await new Promise((resolve, reject) => {\n        const socket = net.createConnection({host, port})\n        const jsonSocket = new JsonSocket(socket)\n\n        const cleanup = () => {\n          jsonSocket.removeAllListeners()\n        }\n\n        jsonSocket.on(\"error\", (err) => {\n          cleanup()\n          reject(err)\n        })\n\n        jsonSocket.on(\"message\", (message) => {\n          if (message?.type === \"job-updated\" && message.jobId === jobId) {\n            cleanup()\n            jsonSocket.close()\n            resolve(undefined)\n            return\n          }\n\n          if (message?.type === \"job-update-error\" && message.jobId === jobId) {\n            cleanup()\n            jsonSocket.close()\n            reject(new Error(message.error || \"Job update failed\"))\n          }\n        })\n\n        socket.on(\"connect\", () => {\n          jsonSocket.send({type: \"hello\", role: \"reporter\"})\n          jsonSocket.send({\n            type: status === \"completed\" ? \"job-complete\" : \"job-failed\",\n            jobId,\n            workerId,\n            handedOffAtMs,\n            error: error ? this._normalizeError(error) : undefined\n          })\n        })\n      })\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.jobId - Job id.\n   * @param {\"completed\" | \"failed\"} args.status - Status.\n   * @param {unknown} [args.error] - Error.\n   * @param {number} [args.handedOffAtMs] - Handed off timestamp.\n   * @param {string} [args.workerId] - Worker id.\n   * @param {number} [args.maxDurationMs] - Max duration for retries.\n   * @returns {Promise<void>} - Resolves when reported.\n   */\n  async reportWithRetry({jobId, status, error, handedOffAtMs, workerId, maxDurationMs}) {\n    let attempt = 0\n    const startTime = Date.now()\n\n    while (true) {\n      try {\n        await this.report({jobId, status, error, handedOffAtMs, workerId})\n        return\n      } catch (err) {\n        attempt += 1\n        const delaySeconds = Math.min(30, 0.5 * attempt)\n\n        this.logger.debug(() => [\"Background job status report failed, retrying\", err])\n\n        if (maxDurationMs && Date.now() - startTime >= maxDurationMs) {\n          this.logger.warn(() => [\"Background job status report timed out, giving up\", err])\n          return\n        }\n\n        await wait(delaySeconds)\n      }\n    }\n  }\n\n  _normalizeError(error) {\n    if (error instanceof Error) return error.stack || error.message\n    if (typeof error === \"string\") return error\n\n    try {\n      return JSON.stringify(error)\n    } catch {\n      return String(error)\n    }\n  }\n}\n"]}
|