velocious 1.0.249 → 1.0.251

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 (46) hide show
  1. package/README.md +46 -0
  2. package/build/src/background-jobs/main.d.ts +2 -0
  3. package/build/src/background-jobs/main.d.ts.map +1 -1
  4. package/build/src/background-jobs/main.js +34 -14
  5. package/build/src/background-jobs/scheduler.d.ts +69 -0
  6. package/build/src/background-jobs/scheduler.d.ts.map +1 -0
  7. package/build/src/background-jobs/scheduler.js +148 -0
  8. package/build/src/configuration-types.d.ts +90 -10
  9. package/build/src/configuration-types.d.ts.map +1 -1
  10. package/build/src/configuration-types.js +31 -6
  11. package/build/src/configuration.d.ts +11 -1
  12. package/build/src/configuration.d.ts.map +1 -1
  13. package/build/src/configuration.js +22 -2
  14. package/build/src/database/query/model-class-query.d.ts +5 -0
  15. package/build/src/database/query/model-class-query.d.ts.map +1 -1
  16. package/build/src/database/query/model-class-query.js +97 -1
  17. package/build/src/database/query/where-model-class-hash.d.ts.map +1 -1
  18. package/build/src/database/query/where-model-class-hash.js +15 -3
  19. package/build/src/database/record/index.d.ts +7 -0
  20. package/build/src/database/record/index.d.ts.map +1 -1
  21. package/build/src/database/record/index.js +10 -1
  22. package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.d.ts +3 -1
  23. package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.d.ts.map +1 -1
  24. package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.js +81 -14
  25. package/build/src/frontend-model-controller.d.ts +7 -3
  26. package/build/src/frontend-model-controller.d.ts.map +1 -1
  27. package/build/src/frontend-model-controller.js +54 -4
  28. package/build/src/frontend-model-resource/base-resource.d.ts +4 -0
  29. package/build/src/frontend-model-resource/base-resource.d.ts.map +1 -1
  30. package/build/src/frontend-model-resource/base-resource.js +9 -1
  31. package/build/src/frontend-models/base.d.ts +12 -3
  32. package/build/src/frontend-models/base.d.ts.map +1 -1
  33. package/build/src/frontend-models/base.js +24 -9
  34. package/build/src/frontend-models/query.d.ts +7 -2
  35. package/build/src/frontend-models/query.d.ts.map +1 -1
  36. package/build/src/frontend-models/query.js +52 -2
  37. package/build/src/frontend-models/resource-definition.d.ts +17 -0
  38. package/build/src/frontend-models/resource-definition.d.ts.map +1 -1
  39. package/build/src/frontend-models/resource-definition.js +126 -15
  40. package/build/src/routes/hooks/frontend-model-command-route-hook.d.ts.map +1 -1
  41. package/build/src/routes/hooks/frontend-model-command-route-hook.js +23 -2
  42. package/build/src/utils/ransack.d.ts +27 -0
  43. package/build/src/utils/ransack.d.ts.map +1 -0
  44. package/build/src/utils/ransack.js +346 -0
  45. package/build/tsconfig.tsbuildinfo +1 -1
  46. package/package.json +1 -1
package/README.md CHANGED
@@ -930,6 +930,23 @@ const specificTask = await Task.where({
930
930
  }).toArray()
931
931
  ```
932
932
 
933
+ ### Ransack-style filtering
934
+
935
+ Use `.ransack(...)` on record queries, record classes, frontend-model queries, and frontend-model classes when you want Rails/Ransack-style predicate keys without hand-writing nested `where(...)` or `search(...)` calls.
936
+
937
+ Supported predicates include `_eq`, `_not_eq`, `_gt`, `_gteq`, `_lt`, `_lteq`, `_cont`, `_start`, `_end`, `_in`, `_not_in`, and `_null`.
938
+
939
+ ```js
940
+ const tasks = await Task.ransack({
941
+ name_cont: "deploy",
942
+ project_project_detail_is_active_eq: true
943
+ }).toArray()
944
+
945
+ const frontendTasks = await FrontendTask
946
+ .ransack({name_cont: "deploy", id_in: ["1", "2"]})
947
+ .toArray()
948
+ ```
949
+
933
950
  ### Raw where clauses
934
951
 
935
952
  ```js
@@ -1493,6 +1510,35 @@ await MyJob.performLaterWithOptions({
1493
1510
  })
1494
1511
  ```
1495
1512
 
1513
+ ## Scheduled jobs
1514
+
1515
+ Velocious can enqueue recurring jobs from the `background-jobs-main` process. Configure them with `scheduledBackgroundJobs` using Sidekiq Scheduler-style `every` arrays:
1516
+
1517
+ ```js
1518
+ import BuildCleanupJob from "./src/jobs/build-cleanup-job.js"
1519
+
1520
+ export default new Configuration({
1521
+ // ...
1522
+ scheduledBackgroundJobs: {
1523
+ jobs: {
1524
+ buildCleanup: {
1525
+ class: BuildCleanupJob,
1526
+ every: ["1h", {first_in: "10s"}],
1527
+ options: {forked: false}
1528
+ }
1529
+ }
1530
+ }
1531
+ })
1532
+ ```
1533
+
1534
+ Supported schedule syntax:
1535
+
1536
+ - `every: "5m"`
1537
+ - `every: ["1h", {first_in: "30s"}]`
1538
+ - `every: ["1 day", {firstIn: "5 minutes"}]`
1539
+
1540
+ `background-jobs-main` owns the schedule and enqueues the configured jobs into the normal Velocious background-jobs queue. The HTTP server does not run scheduled jobs itself.
1541
+
1496
1542
  ## Persistence and retries
1497
1543
 
1498
1544
  Jobs are persisted in the configured database (`backgroundJobs.databaseIdentifier`) in an internal `background_jobs` table. When a worker picks a job, the job is marked as handed off and the worker reports completion or failure back to the main process.
@@ -27,6 +27,7 @@ export default class BackgroundJobsMain {
27
27
  server: net.Server;
28
28
  _dispatchTimer: NodeJS.Timeout;
29
29
  _orphanTimer: NodeJS.Timeout;
30
+ scheduler: BackgroundJobsScheduler;
30
31
  /**
31
32
  * @returns {Promise<void>} - Resolves when closed.
32
33
  */
@@ -59,4 +60,5 @@ import BackgroundJobsStore from "./store.js";
59
60
  import Logger from "../logger.js";
60
61
  import JsonSocket from "./json-socket.js";
61
62
  import net from "net";
63
+ import BackgroundJobsScheduler from "./scheduler.js";
62
64
  //# sourceMappingURL=main.d.ts.map
@@ -1 +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;mBACzB,cAAc;uBAFV,kBAAkB;gBADzB,KAAK"}
1
+ {"version":3,"file":"main.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/main.js"],"names":[],"mappings":"AAQA;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,CA2CzB;IArCC,mBAA0E;IAaxE,+BAEQ;IAER,6BAES;IAET,mCAUE;IAQN;;OAEG;IACH,QAFa,OAAO,CAAC,IAAI,CAAC,CAczB;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;gCA3P+B,YAAY;mBACzB,cAAc;uBAHV,kBAAkB;gBADzB,KAAK;oCAEe,gBAAgB"}
@@ -1,6 +1,7 @@
1
1
  // @ts-check
2
2
  import net from "net";
3
3
  import JsonSocket from "./json-socket.js";
4
+ import BackgroundJobsScheduler from "./scheduler.js";
4
5
  import BackgroundJobsStore from "./store.js";
5
6
  import Logger from "../logger.js";
6
7
  export default class BackgroundJobsMain {
@@ -31,20 +32,38 @@ export default class BackgroundJobsMain {
31
32
  await this.configuration.initialize({ type: "background-jobs-main" });
32
33
  await this.store.ensureReady();
33
34
  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;
35
+ try {
36
+ await new Promise((resolve, reject) => {
37
+ this.server.once("error", reject);
38
+ this.server.listen(this.port, this.host, () => resolve(undefined));
39
+ });
40
+ const address = this.server.address();
41
+ if (address && typeof address === "object") {
42
+ this.port = address.port;
43
+ }
44
+ this._dispatchTimer = setInterval(() => {
45
+ void this._dispatch();
46
+ }, 1000);
47
+ this._orphanTimer = setInterval(() => {
48
+ void this._sweepOrphans();
49
+ }, 60000);
50
+ this.scheduler = new BackgroundJobsScheduler({
51
+ configuration: this.configuration,
52
+ enqueueJob: async ({ args, jobClass, options }) => {
53
+ await this.store.enqueue({
54
+ jobName: jobClass.jobName(),
55
+ args,
56
+ options
57
+ });
58
+ await this._dispatch();
59
+ }
60
+ });
61
+ await this.scheduler.start();
62
+ }
63
+ catch (error) {
64
+ await this.stop();
65
+ throw error;
41
66
  }
42
- this._dispatchTimer = setInterval(() => {
43
- void this._dispatch();
44
- }, 1000);
45
- this._orphanTimer = setInterval(() => {
46
- void this._sweepOrphans();
47
- }, 60000);
48
67
  }
49
68
  /**
50
69
  * @returns {Promise<void>} - Resolves when closed.
@@ -57,6 +76,7 @@ export default class BackgroundJobsMain {
57
76
  clearInterval(this._dispatchTimer);
58
77
  if (this._orphanTimer)
59
78
  clearInterval(this._orphanTimer);
79
+ this.scheduler?.stop();
60
80
  if (!this.server)
61
81
  return;
62
82
  await new Promise((resolve) => this.server.close(() => resolve(undefined)));
@@ -213,4 +233,4 @@ export default class BackgroundJobsMain {
213
233
  }
214
234
  }
215
235
  }
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"]}
236
+ //# 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 BackgroundJobsScheduler from \"./scheduler.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    try {\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      this.scheduler = new BackgroundJobsScheduler({\n        configuration: this.configuration,\n        enqueueJob: async ({args, jobClass, options}) => {\n          await this.store.enqueue({\n            jobName: jobClass.jobName(),\n            args,\n            options\n          })\n          await this._dispatch()\n        }\n      })\n      await this.scheduler.start()\n    } catch (error) {\n      await this.stop()\n      throw error\n    }\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    this.scheduler?.stop()\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,69 @@
1
+ /**
2
+ * @param {number | string} value - Duration value.
3
+ * @param {string} fieldName - Field name for errors.
4
+ * @returns {number} - Duration in milliseconds.
5
+ */
6
+ export function parseScheduledDuration(value: number | string, fieldName: string): number;
7
+ /** Runs configured recurring background job schedules. */
8
+ export default class BackgroundJobsScheduler {
9
+ /**
10
+ * @param {object} args - Options.
11
+ * @param {import("../configuration.js").default} args.configuration - Configuration.
12
+ * @param {function({args: any[], jobClass: typeof import("./job.js").default, jobKey: string, options: import("./types.js").BackgroundJobOptions}) : Promise<void>} args.enqueueJob - Enqueue callback.
13
+ */
14
+ constructor({ configuration, enqueueJob }: {
15
+ configuration: import("../configuration.js").default;
16
+ enqueueJob: (arg0: {
17
+ args: any[];
18
+ jobClass: typeof import("./job.js").default;
19
+ jobKey: string;
20
+ options: import("./types.js").BackgroundJobOptions;
21
+ }) => Promise<void>;
22
+ });
23
+ configuration: import("../configuration.js").default;
24
+ enqueueJob: (arg0: {
25
+ args: any[];
26
+ jobClass: typeof import("./job.js").default;
27
+ jobKey: string;
28
+ options: import("./types.js").BackgroundJobOptions;
29
+ }) => Promise<void>;
30
+ logger: Logger;
31
+ /** @type {Array<ReturnType<typeof setInterval>>} */
32
+ intervalIds: Array<ReturnType<typeof setInterval>>;
33
+ /** @type {Array<ReturnType<typeof setTimeout>>} */
34
+ timeoutIds: Array<ReturnType<typeof setTimeout>>;
35
+ /** @returns {Promise<void>} */
36
+ start(): Promise<void>;
37
+ /** @returns {void} */
38
+ stop(): void;
39
+ /**
40
+ * @param {object} args - Options.
41
+ * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.
42
+ * @param {string} args.jobKey - Job key.
43
+ * @returns {void}
44
+ */
45
+ scheduleJob({ jobConfiguration, jobKey }: {
46
+ jobConfiguration: import("../configuration-types.js").ScheduledBackgroundJobConfiguration;
47
+ jobKey: string;
48
+ }): void;
49
+ /**
50
+ * @param {object} args - Options.
51
+ * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.
52
+ * @param {string} args.jobKey - Job key.
53
+ * @returns {Promise<void>}
54
+ */
55
+ enqueueScheduledJob({ jobConfiguration, jobKey }: {
56
+ jobConfiguration: import("../configuration-types.js").ScheduledBackgroundJobConfiguration;
57
+ jobKey: string;
58
+ }): Promise<void>;
59
+ /**
60
+ * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration["every"]} every - Every config.
61
+ * @returns {{everyValue: number | string, firstInValue?: number | string}} - Normalized interval and first-run delay values.
62
+ */
63
+ normalizeEvery(every: import("../configuration-types.js").ScheduledBackgroundJobConfiguration["every"]): {
64
+ everyValue: number | string;
65
+ firstInValue?: number | string;
66
+ };
67
+ }
68
+ import Logger from "../logger.js";
69
+ //# sourceMappingURL=scheduler.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scheduler.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/scheduler.js"],"names":[],"mappings":"AAuBA;;;;GAIG;AACH,8CAJW,MAAM,GAAG,MAAM,aACf,MAAM,GACJ,MAAM,CA8BlB;AAED,0DAA0D;AAC1D;IACE;;;;OAIG;IACH,2CAHG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QAC0H,UAAU,EAAzK,CAAS,IAA8H,EAA9H;YAAC,IAAI,EAAE,GAAG,EAAE,CAAC;YAAC,QAAQ,EAAE,cAAc,UAAU,EAAE,OAAO,CAAC;YAAC,MAAM,EAAE,MAAM,CAAC;YAAC,OAAO,EAAE,OAAO,YAAY,EAAE,oBAAoB,CAAA;SAAC,KAAI,OAAO,CAAC,IAAI,CAAC;KAClK,EASA;IAPC,qDAAkC;IAClC,mBAJkB;QAAC,IAAI,EAAE,GAAG,EAAE,CAAC;QAAC,QAAQ,EAAE,cAAc,UAAU,EAAE,OAAO,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,OAAO,YAAY,EAAE,oBAAoB,CAAA;KAAC,KAAI,OAAO,CAAC,IAAI,CAAC,CAIrI;IAC5B,eAA8B;IAC9B,oDAAoD;IACpD,aADW,KAAK,CAAC,UAAU,CAAC,OAAO,WAAW,CAAC,CAAC,CAC3B;IACrB,mDAAmD;IACnD,YADW,KAAK,CAAC,UAAU,CAAC,OAAO,UAAU,CAAC,CAAC,CAC3B;IAGtB,+BAA+B;IAC/B,SADc,OAAO,CAAC,IAAI,CAAC,CAiB1B;IAED,sBAAsB;IACtB,QADc,IAAI,CAYjB;IAED;;;;;OAKG;IACH,0CAJG;QAAsF,gBAAgB,EAA9F,OAAO,2BAA2B,EAAE,mCAAmC;QAC1D,MAAM,EAAnB,MAAM;KACd,GAAU,IAAI,CA0BhB;IAED;;;;;OAKG;IACH,kDAJG;QAAsF,gBAAgB,EAA9F,OAAO,2BAA2B,EAAE,mCAAmC;QAC1D,MAAM,EAAnB,MAAM;KACd,GAAU,OAAO,CAAC,IAAI,CAAC,CAazB;IAED;;;OAGG;IACH,sBAHW,OAAO,2BAA2B,EAAE,mCAAmC,CAAC,OAAO,CAAC,GAC9E;QAAC,UAAU,EAAE,MAAM,GAAG,MAAM,CAAC;QAAC,YAAY,CAAC,EAAE,MAAM,GAAG,MAAM,CAAA;KAAC,CAczE;CACF;mBA9KkB,cAAc"}
@@ -0,0 +1,148 @@
1
+ // @ts-check
2
+ import Logger from "../logger.js";
3
+ const DURATION_MULTIPLIERS = {
4
+ d: 24 * 60 * 60 * 1000,
5
+ day: 24 * 60 * 60 * 1000,
6
+ days: 24 * 60 * 60 * 1000,
7
+ h: 60 * 60 * 1000,
8
+ hour: 60 * 60 * 1000,
9
+ hours: 60 * 60 * 1000,
10
+ m: 60 * 1000,
11
+ minute: 60 * 1000,
12
+ minutes: 60 * 1000,
13
+ ms: 1,
14
+ s: 1000,
15
+ second: 1000,
16
+ seconds: 1000,
17
+ w: 7 * 24 * 60 * 60 * 1000,
18
+ week: 7 * 24 * 60 * 60 * 1000,
19
+ weeks: 7 * 24 * 60 * 60 * 1000
20
+ };
21
+ /**
22
+ * @param {number | string} value - Duration value.
23
+ * @param {string} fieldName - Field name for errors.
24
+ * @returns {number} - Duration in milliseconds.
25
+ */
26
+ export function parseScheduledDuration(value, fieldName) {
27
+ if (typeof value === "number") {
28
+ if (!Number.isFinite(value) || value < 1) {
29
+ throw new Error(`Scheduled background job ${fieldName} must be a positive number of milliseconds.`);
30
+ }
31
+ return value;
32
+ }
33
+ if (typeof value !== "string" || !value.trim()) {
34
+ throw new Error(`Scheduled background job ${fieldName} must be a non-empty string or number.`);
35
+ }
36
+ const normalizedValue = value.trim().toLowerCase();
37
+ const match = normalizedValue.match(/^(\d+(?:\.\d+)?)\s*(ms|s|m|h|d|w|second|seconds|minute|minutes|hour|hours|day|days|week|weeks)$/);
38
+ if (!match) {
39
+ throw new Error(`Invalid scheduled background job ${fieldName}: ${value}`);
40
+ }
41
+ const numericValue = Number(match[1]);
42
+ const multiplier = DURATION_MULTIPLIERS[match[2]];
43
+ if (!multiplier) {
44
+ throw new Error(`Invalid scheduled background job ${fieldName}: ${value}`);
45
+ }
46
+ return Math.round(numericValue * multiplier);
47
+ }
48
+ /** Runs configured recurring background job schedules. */
49
+ export default class BackgroundJobsScheduler {
50
+ /**
51
+ * @param {object} args - Options.
52
+ * @param {import("../configuration.js").default} args.configuration - Configuration.
53
+ * @param {function({args: any[], jobClass: typeof import("./job.js").default, jobKey: string, options: import("./types.js").BackgroundJobOptions}) : Promise<void>} args.enqueueJob - Enqueue callback.
54
+ */
55
+ constructor({ configuration, enqueueJob }) {
56
+ this.configuration = configuration;
57
+ this.enqueueJob = enqueueJob;
58
+ this.logger = new Logger(this);
59
+ /** @type {Array<ReturnType<typeof setInterval>>} */
60
+ this.intervalIds = [];
61
+ /** @type {Array<ReturnType<typeof setTimeout>>} */
62
+ this.timeoutIds = [];
63
+ }
64
+ /** @returns {Promise<void>} */
65
+ async start() {
66
+ const scheduledBackgroundJobsConfig = await this.configuration.getScheduledBackgroundJobsConfig();
67
+ if (!scheduledBackgroundJobsConfig?.jobs) {
68
+ return;
69
+ }
70
+ for (const jobKey of Object.keys(scheduledBackgroundJobsConfig.jobs)) {
71
+ const jobConfiguration = scheduledBackgroundJobsConfig.jobs[jobKey];
72
+ if (!jobConfiguration || jobConfiguration.enabled === false) {
73
+ continue;
74
+ }
75
+ this.scheduleJob({ jobConfiguration, jobKey });
76
+ }
77
+ }
78
+ /** @returns {void} */
79
+ stop() {
80
+ for (const intervalId of this.intervalIds) {
81
+ clearInterval(intervalId);
82
+ }
83
+ for (const timeoutId of this.timeoutIds) {
84
+ clearTimeout(timeoutId);
85
+ }
86
+ this.intervalIds = [];
87
+ this.timeoutIds = [];
88
+ }
89
+ /**
90
+ * @param {object} args - Options.
91
+ * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.
92
+ * @param {string} args.jobKey - Job key.
93
+ * @returns {void}
94
+ */
95
+ scheduleJob({ jobConfiguration, jobKey }) {
96
+ const { everyValue, firstInValue } = this.normalizeEvery(jobConfiguration.every);
97
+ const intervalMs = parseScheduledDuration(everyValue, `${jobKey}.every`);
98
+ const firstInMs = firstInValue !== undefined ? parseScheduledDuration(firstInValue, `${jobKey}.first_in`) : intervalMs;
99
+ if (intervalMs < 1) {
100
+ throw new Error(`Scheduled background job ${jobKey}.every must be at least 1 millisecond.`);
101
+ }
102
+ if (!jobConfiguration.class || typeof jobConfiguration.class.performLaterWithOptions !== "function") {
103
+ throw new Error(`Scheduled background job ${jobKey} must define a job class.`);
104
+ }
105
+ const timeoutId = setTimeout(() => {
106
+ void this.enqueueScheduledJob({ jobConfiguration, jobKey });
107
+ const intervalId = setInterval(() => {
108
+ void this.enqueueScheduledJob({ jobConfiguration, jobKey });
109
+ }, intervalMs);
110
+ this.intervalIds.push(intervalId);
111
+ }, firstInMs);
112
+ this.timeoutIds.push(timeoutId);
113
+ }
114
+ /**
115
+ * @param {object} args - Options.
116
+ * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.
117
+ * @param {string} args.jobKey - Job key.
118
+ * @returns {Promise<void>}
119
+ */
120
+ async enqueueScheduledJob({ jobConfiguration, jobKey }) {
121
+ try {
122
+ await this.enqueueJob({
123
+ args: Array.isArray(jobConfiguration.args) ? jobConfiguration.args : [],
124
+ jobClass: jobConfiguration.class,
125
+ jobKey,
126
+ options: jobConfiguration.options || {}
127
+ });
128
+ }
129
+ catch (error) {
130
+ await this.logger.error(() => ["Failed to enqueue scheduled background job", { jobKey, jobName: jobConfiguration.class.jobName() }, error]);
131
+ }
132
+ }
133
+ /**
134
+ * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration["every"]} every - Every config.
135
+ * @returns {{everyValue: number | string, firstInValue?: number | string}} - Normalized interval and first-run delay values.
136
+ */
137
+ normalizeEvery(every) {
138
+ if (Array.isArray(every)) {
139
+ const [everyValue, everyOptions] = every;
140
+ if (!everyOptions || typeof everyOptions !== "object" || Array.isArray(everyOptions)) {
141
+ return { everyValue };
142
+ }
143
+ return { everyValue, firstInValue: everyOptions.firstIn ?? everyOptions.first_in };
144
+ }
145
+ return { everyValue: every };
146
+ }
147
+ }
148
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scheduler.js","sourceRoot":"","sources":["../../../src/background-jobs/scheduler.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Logger from \"../logger.js\"\n\nconst DURATION_MULTIPLIERS = {\n  d: 24 * 60 * 60 * 1000,\n  day: 24 * 60 * 60 * 1000,\n  days: 24 * 60 * 60 * 1000,\n  h: 60 * 60 * 1000,\n  hour: 60 * 60 * 1000,\n  hours: 60 * 60 * 1000,\n  m: 60 * 1000,\n  minute: 60 * 1000,\n  minutes: 60 * 1000,\n  ms: 1,\n  s: 1000,\n  second: 1000,\n  seconds: 1000,\n  w: 7 * 24 * 60 * 60 * 1000,\n  week: 7 * 24 * 60 * 60 * 1000,\n  weeks: 7 * 24 * 60 * 60 * 1000\n}\n\n/**\n * @param {number | string} value - Duration value.\n * @param {string} fieldName - Field name for errors.\n * @returns {number} - Duration in milliseconds.\n */\nexport function parseScheduledDuration(value, fieldName) {\n  if (typeof value === \"number\") {\n    if (!Number.isFinite(value) || value < 1) {\n      throw new Error(`Scheduled background job ${fieldName} must be a positive number of milliseconds.`)\n    }\n\n    return value\n  }\n\n  if (typeof value !== \"string\" || !value.trim()) {\n    throw new Error(`Scheduled background job ${fieldName} must be a non-empty string or number.`)\n  }\n\n  const normalizedValue = value.trim().toLowerCase()\n  const match = normalizedValue.match(/^(\\d+(?:\\.\\d+)?)\\s*(ms|s|m|h|d|w|second|seconds|minute|minutes|hour|hours|day|days|week|weeks)$/)\n\n  if (!match) {\n    throw new Error(`Invalid scheduled background job ${fieldName}: ${value}`)\n  }\n\n  const numericValue = Number(match[1])\n  const multiplier = DURATION_MULTIPLIERS[match[2]]\n\n  if (!multiplier) {\n    throw new Error(`Invalid scheduled background job ${fieldName}: ${value}`)\n  }\n\n  return Math.round(numericValue * multiplier)\n}\n\n/** Runs configured recurring background job schedules. */\nexport default class BackgroundJobsScheduler {\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration.\n   * @param {function({args: any[], jobClass: typeof import(\"./job.js\").default, jobKey: string, options: import(\"./types.js\").BackgroundJobOptions}) : Promise<void>} args.enqueueJob - Enqueue callback.\n   */\n  constructor({configuration, enqueueJob}) {\n    this.configuration = configuration\n    this.enqueueJob = enqueueJob\n    this.logger = new Logger(this)\n    /** @type {Array<ReturnType<typeof setInterval>>} */\n    this.intervalIds = []\n    /** @type {Array<ReturnType<typeof setTimeout>>} */\n    this.timeoutIds = []\n  }\n\n  /** @returns {Promise<void>} */\n  async start() {\n    const scheduledBackgroundJobsConfig = await this.configuration.getScheduledBackgroundJobsConfig()\n\n    if (!scheduledBackgroundJobsConfig?.jobs) {\n      return\n    }\n\n    for (const jobKey of Object.keys(scheduledBackgroundJobsConfig.jobs)) {\n      const jobConfiguration = scheduledBackgroundJobsConfig.jobs[jobKey]\n\n      if (!jobConfiguration || jobConfiguration.enabled === false) {\n        continue\n      }\n\n      this.scheduleJob({jobConfiguration, jobKey})\n    }\n  }\n\n  /** @returns {void} */\n  stop() {\n    for (const intervalId of this.intervalIds) {\n      clearInterval(intervalId)\n    }\n\n    for (const timeoutId of this.timeoutIds) {\n      clearTimeout(timeoutId)\n    }\n\n    this.intervalIds = []\n    this.timeoutIds = []\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.\n   * @param {string} args.jobKey - Job key.\n   * @returns {void}\n   */\n  scheduleJob({jobConfiguration, jobKey}) {\n    const {everyValue, firstInValue} = this.normalizeEvery(jobConfiguration.every)\n    const intervalMs = parseScheduledDuration(everyValue, `${jobKey}.every`)\n    const firstInMs = firstInValue !== undefined ? parseScheduledDuration(firstInValue, `${jobKey}.first_in`) : intervalMs\n\n    if (intervalMs < 1) {\n      throw new Error(`Scheduled background job ${jobKey}.every must be at least 1 millisecond.`)\n    }\n\n    if (!jobConfiguration.class || typeof jobConfiguration.class.performLaterWithOptions !== \"function\") {\n      throw new Error(`Scheduled background job ${jobKey} must define a job class.`)\n    }\n\n    const timeoutId = setTimeout(() => {\n      void this.enqueueScheduledJob({jobConfiguration, jobKey})\n\n      const intervalId = setInterval(() => {\n        void this.enqueueScheduledJob({jobConfiguration, jobKey})\n      }, intervalMs)\n\n      this.intervalIds.push(intervalId)\n    }, firstInMs)\n\n    this.timeoutIds.push(timeoutId)\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.\n   * @param {string} args.jobKey - Job key.\n   * @returns {Promise<void>}\n   */\n  async enqueueScheduledJob({jobConfiguration, jobKey}) {\n    try {\n      await this.enqueueJob({\n        args: Array.isArray(jobConfiguration.args) ? jobConfiguration.args : [],\n        jobClass: jobConfiguration.class,\n        jobKey,\n        options: jobConfiguration.options || {}\n      })\n    } catch (error) {\n      await this.logger.error(() => [\"Failed to enqueue scheduled background job\", {jobKey, jobName: jobConfiguration.class.jobName()}, error])\n    }\n  }\n\n  /**\n   * @param {import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration[\"every\"]} every - Every config.\n   * @returns {{everyValue: number | string, firstInValue?: number | string}} - Normalized interval and first-run delay values.\n   */\n  normalizeEvery(every) {\n    if (Array.isArray(every)) {\n      const [everyValue, everyOptions] = every\n\n      if (!everyOptions || typeof everyOptions !== \"object\" || Array.isArray(everyOptions)) {\n        return {everyValue}\n      }\n\n      return {everyValue, firstInValue: everyOptions.firstIn ?? everyOptions.first_in}\n    }\n\n    return {everyValue: every}\n  }\n}\n"]}