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.
Files changed (89) hide show
  1. package/README.md +125 -0
  2. package/build/src/background-jobs/client.d.ts +23 -0
  3. package/build/src/background-jobs/client.d.ts.map +1 -0
  4. package/build/src/background-jobs/client.js +58 -0
  5. package/build/src/background-jobs/job-record.d.ts +4 -0
  6. package/build/src/background-jobs/job-record.d.ts.map +1 -0
  7. package/build/src/background-jobs/job-record.js +11 -0
  8. package/build/src/background-jobs/job-registry.d.ts +23 -0
  9. package/build/src/background-jobs/job-registry.d.ts.map +1 -0
  10. package/build/src/background-jobs/job-registry.js +55 -0
  11. package/build/src/background-jobs/job-runner.d.ts +6 -0
  12. package/build/src/background-jobs/job-runner.d.ts.map +1 -0
  13. package/build/src/background-jobs/job-runner.js +44 -0
  14. package/build/src/background-jobs/job.d.ts +35 -0
  15. package/build/src/background-jobs/job.d.ts.map +1 -0
  16. package/build/src/background-jobs/job.js +61 -0
  17. package/build/src/background-jobs/json-socket.d.ts +27 -0
  18. package/build/src/background-jobs/json-socket.d.ts.map +1 -0
  19. package/build/src/background-jobs/json-socket.js +55 -0
  20. package/build/src/background-jobs/main.d.ts +62 -0
  21. package/build/src/background-jobs/main.d.ts.map +1 -0
  22. package/build/src/background-jobs/main.js +216 -0
  23. package/build/src/background-jobs/status-reporter.d.ts +54 -0
  24. package/build/src/background-jobs/status-reporter.d.ts.map +1 -0
  25. package/build/src/background-jobs/status-reporter.js +113 -0
  26. package/build/src/background-jobs/store.d.ts +237 -0
  27. package/build/src/background-jobs/store.d.ts.map +1 -0
  28. package/build/src/background-jobs/store.js +488 -0
  29. package/build/src/background-jobs/types.d.ts +17 -0
  30. package/build/src/background-jobs/types.d.ts.map +1 -0
  31. package/build/src/background-jobs/types.js +8 -0
  32. package/build/src/background-jobs/worker.d.ts +64 -0
  33. package/build/src/background-jobs/worker.d.ts.map +1 -0
  34. package/build/src/background-jobs/worker.js +155 -0
  35. package/build/src/cli/commands/background-jobs-main.d.ts +5 -0
  36. package/build/src/cli/commands/background-jobs-main.d.ts.map +1 -0
  37. package/build/src/cli/commands/background-jobs-main.js +7 -0
  38. package/build/src/cli/commands/background-jobs-runner.d.ts +5 -0
  39. package/build/src/cli/commands/background-jobs-runner.d.ts.map +1 -0
  40. package/build/src/cli/commands/background-jobs-runner.js +7 -0
  41. package/build/src/cli/commands/background-jobs-worker.d.ts +5 -0
  42. package/build/src/cli/commands/background-jobs-worker.d.ts.map +1 -0
  43. package/build/src/cli/commands/background-jobs-worker.js +7 -0
  44. package/build/src/configuration-types.d.ts +25 -0
  45. package/build/src/configuration-types.d.ts.map +1 -1
  46. package/build/src/configuration-types.js +8 -1
  47. package/build/src/configuration.d.ts +11 -1
  48. package/build/src/configuration.d.ts.map +1 -1
  49. package/build/src/configuration.js +26 -2
  50. package/build/src/database/drivers/mssql/sql/update.js +3 -3
  51. package/build/src/database/drivers/mysql/sql/update.js +3 -3
  52. package/build/src/database/drivers/pgsql/sql/update.js +3 -3
  53. package/build/src/database/drivers/sqlite/sql/update.js +3 -3
  54. package/build/src/database/query/update-base.d.ts +5 -0
  55. package/build/src/database/query/update-base.d.ts.map +1 -1
  56. package/build/src/database/query/update-base.js +10 -1
  57. package/build/src/database/query-parser/limit-parser.d.ts.map +1 -1
  58. package/build/src/database/query-parser/limit-parser.js +8 -7
  59. package/build/src/environment-handlers/base.d.ts +15 -0
  60. package/build/src/environment-handlers/base.d.ts.map +1 -1
  61. package/build/src/environment-handlers/base.js +22 -1
  62. package/build/src/environment-handlers/browser.d.ts.map +1 -1
  63. package/build/src/environment-handlers/browser.js +10 -1
  64. package/build/src/environment-handlers/node/cli/commands/background-jobs-main.d.ts +5 -0
  65. package/build/src/environment-handlers/node/cli/commands/background-jobs-main.d.ts.map +1 -0
  66. package/build/src/environment-handlers/node/cli/commands/background-jobs-main.js +18 -0
  67. package/build/src/environment-handlers/node/cli/commands/background-jobs-runner.d.ts +5 -0
  68. package/build/src/environment-handlers/node/cli/commands/background-jobs-runner.d.ts.map +1 -0
  69. package/build/src/environment-handlers/node/cli/commands/background-jobs-runner.js +13 -0
  70. package/build/src/environment-handlers/node/cli/commands/background-jobs-worker.d.ts +5 -0
  71. package/build/src/environment-handlers/node/cli/commands/background-jobs-worker.d.ts.map +1 -0
  72. package/build/src/environment-handlers/node/cli/commands/background-jobs-worker.js +18 -0
  73. package/build/src/environment-handlers/node/cli/commands/generate/base-models.d.ts.map +1 -1
  74. package/build/src/environment-handlers/node/cli/commands/generate/base-models.js +6 -4
  75. package/build/src/environment-handlers/node/cli/commands/test.d.ts.map +1 -1
  76. package/build/src/environment-handlers/node/cli/commands/test.js +2 -116
  77. package/build/src/environment-handlers/node.d.ts.map +1 -1
  78. package/build/src/environment-handlers/node.js +25 -1
  79. package/build/src/testing/browser-test-app.d.ts +2 -0
  80. package/build/src/testing/browser-test-app.d.ts.map +1 -0
  81. package/build/src/testing/browser-test-app.js +24 -0
  82. package/build/src/testing/test-filter-parser.d.ts +16 -0
  83. package/build/src/testing/test-filter-parser.d.ts.map +1 -0
  84. package/build/src/testing/test-filter-parser.js +117 -0
  85. package/build/src/testing/test-runner.d.ts +35 -0
  86. package/build/src/testing/test-runner.d.ts.map +1 -1
  87. package/build/src/testing/test-runner.js +100 -17
  88. package/build/tsconfig.tsbuildinfo +1 -1
  89. 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"]}