@suzuke/agend 0.0.1 → 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (202) hide show
  1. package/README.md +557 -1
  2. package/README.zh-TW.md +504 -0
  3. package/dist/access-path.d.ts +7 -0
  4. package/dist/access-path.js +12 -0
  5. package/dist/access-path.js.map +1 -0
  6. package/dist/approval/approval-server.d.ts +30 -0
  7. package/dist/approval/approval-server.js +156 -0
  8. package/dist/approval/approval-server.js.map +1 -0
  9. package/dist/approval/tmux-prompt-detector.d.ts +34 -0
  10. package/dist/approval/tmux-prompt-detector.js +264 -0
  11. package/dist/approval/tmux-prompt-detector.js.map +1 -0
  12. package/dist/backend/approval-strategy.d.ts +14 -0
  13. package/dist/backend/approval-strategy.js +2 -0
  14. package/dist/backend/approval-strategy.js.map +1 -0
  15. package/dist/backend/claude-code.d.ts +13 -0
  16. package/dist/backend/claude-code.js +114 -0
  17. package/dist/backend/claude-code.js.map +1 -0
  18. package/dist/backend/codex.d.ts +10 -0
  19. package/dist/backend/codex.js +58 -0
  20. package/dist/backend/codex.js.map +1 -0
  21. package/dist/backend/factory.d.ts +2 -0
  22. package/dist/backend/factory.js +19 -0
  23. package/dist/backend/factory.js.map +1 -0
  24. package/dist/backend/gemini-cli.d.ts +10 -0
  25. package/dist/backend/gemini-cli.js +68 -0
  26. package/dist/backend/gemini-cli.js.map +1 -0
  27. package/dist/backend/hook-based-approval.d.ts +20 -0
  28. package/dist/backend/hook-based-approval.js +41 -0
  29. package/dist/backend/hook-based-approval.js.map +1 -0
  30. package/dist/backend/index.d.ts +6 -0
  31. package/dist/backend/index.js +6 -0
  32. package/dist/backend/index.js.map +1 -0
  33. package/dist/backend/opencode.d.ts +10 -0
  34. package/dist/backend/opencode.js +63 -0
  35. package/dist/backend/opencode.js.map +1 -0
  36. package/dist/backend/types.d.ts +26 -0
  37. package/dist/backend/types.js +2 -0
  38. package/dist/backend/types.js.map +1 -0
  39. package/dist/channel/access-manager.d.ts +18 -0
  40. package/dist/channel/access-manager.js +149 -0
  41. package/dist/channel/access-manager.js.map +1 -0
  42. package/dist/channel/adapters/discord.d.ts +45 -0
  43. package/dist/channel/adapters/discord.js +366 -0
  44. package/dist/channel/adapters/discord.js.map +1 -0
  45. package/dist/channel/adapters/telegram.d.ts +58 -0
  46. package/dist/channel/adapters/telegram.js +569 -0
  47. package/dist/channel/adapters/telegram.js.map +1 -0
  48. package/dist/channel/attachment-handler.d.ts +15 -0
  49. package/dist/channel/attachment-handler.js +55 -0
  50. package/dist/channel/attachment-handler.js.map +1 -0
  51. package/dist/channel/factory.d.ts +12 -0
  52. package/dist/channel/factory.js +38 -0
  53. package/dist/channel/factory.js.map +1 -0
  54. package/dist/channel/ipc-bridge.d.ts +26 -0
  55. package/dist/channel/ipc-bridge.js +170 -0
  56. package/dist/channel/ipc-bridge.js.map +1 -0
  57. package/dist/channel/mcp-server.d.ts +10 -0
  58. package/dist/channel/mcp-server.js +196 -0
  59. package/dist/channel/mcp-server.js.map +1 -0
  60. package/dist/channel/mcp-tools.d.ts +909 -0
  61. package/dist/channel/mcp-tools.js +346 -0
  62. package/dist/channel/mcp-tools.js.map +1 -0
  63. package/dist/channel/message-bus.d.ts +17 -0
  64. package/dist/channel/message-bus.js +86 -0
  65. package/dist/channel/message-bus.js.map +1 -0
  66. package/dist/channel/message-queue.d.ts +39 -0
  67. package/dist/channel/message-queue.js +248 -0
  68. package/dist/channel/message-queue.js.map +1 -0
  69. package/dist/channel/tool-router.d.ts +6 -0
  70. package/dist/channel/tool-router.js +69 -0
  71. package/dist/channel/tool-router.js.map +1 -0
  72. package/dist/channel/tool-tracker.d.ts +13 -0
  73. package/dist/channel/tool-tracker.js +58 -0
  74. package/dist/channel/tool-tracker.js.map +1 -0
  75. package/dist/channel/types.d.ts +116 -0
  76. package/dist/channel/types.js +2 -0
  77. package/dist/channel/types.js.map +1 -0
  78. package/dist/cli.d.ts +2 -0
  79. package/dist/cli.js +782 -0
  80. package/dist/cli.js.map +1 -0
  81. package/dist/config.d.ts +8 -0
  82. package/dist/config.js +85 -0
  83. package/dist/config.js.map +1 -0
  84. package/dist/container-manager.d.ts +24 -0
  85. package/dist/container-manager.js +148 -0
  86. package/dist/container-manager.js.map +1 -0
  87. package/dist/context-guardian.d.ts +29 -0
  88. package/dist/context-guardian.js +123 -0
  89. package/dist/context-guardian.js.map +1 -0
  90. package/dist/cost-guard.d.ts +21 -0
  91. package/dist/cost-guard.js +113 -0
  92. package/dist/cost-guard.js.map +1 -0
  93. package/dist/daemon-entry.d.ts +1 -0
  94. package/dist/daemon-entry.js +29 -0
  95. package/dist/daemon-entry.js.map +1 -0
  96. package/dist/daemon.d.ts +88 -0
  97. package/dist/daemon.js +820 -0
  98. package/dist/daemon.js.map +1 -0
  99. package/dist/daily-summary.d.ts +13 -0
  100. package/dist/daily-summary.js +55 -0
  101. package/dist/daily-summary.js.map +1 -0
  102. package/dist/db.d.ts +10 -0
  103. package/dist/db.js +43 -0
  104. package/dist/db.js.map +1 -0
  105. package/dist/event-log.d.ts +22 -0
  106. package/dist/event-log.js +66 -0
  107. package/dist/event-log.js.map +1 -0
  108. package/dist/export-import.d.ts +2 -0
  109. package/dist/export-import.js +110 -0
  110. package/dist/export-import.js.map +1 -0
  111. package/dist/fleet-context.d.ts +36 -0
  112. package/dist/fleet-context.js +4 -0
  113. package/dist/fleet-context.js.map +1 -0
  114. package/dist/fleet-manager.d.ts +115 -0
  115. package/dist/fleet-manager.js +1742 -0
  116. package/dist/fleet-manager.js.map +1 -0
  117. package/dist/fleet-system-prompt.d.ts +11 -0
  118. package/dist/fleet-system-prompt.js +60 -0
  119. package/dist/fleet-system-prompt.js.map +1 -0
  120. package/dist/hang-detector.d.ts +16 -0
  121. package/dist/hang-detector.js +53 -0
  122. package/dist/hang-detector.js.map +1 -0
  123. package/dist/index.d.ts +8 -0
  124. package/dist/index.js +6 -0
  125. package/dist/index.js.map +1 -0
  126. package/dist/install-recorder.d.ts +30 -0
  127. package/dist/install-recorder.js +159 -0
  128. package/dist/install-recorder.js.map +1 -0
  129. package/dist/logger.d.ts +3 -0
  130. package/dist/logger.js +63 -0
  131. package/dist/logger.js.map +1 -0
  132. package/dist/meeting/orchestrator.d.ts +30 -0
  133. package/dist/meeting/orchestrator.js +355 -0
  134. package/dist/meeting/orchestrator.js.map +1 -0
  135. package/dist/meeting/prompt-builder.d.ts +12 -0
  136. package/dist/meeting/prompt-builder.js +96 -0
  137. package/dist/meeting/prompt-builder.js.map +1 -0
  138. package/dist/meeting/role-assigner.d.ts +2 -0
  139. package/dist/meeting/role-assigner.js +25 -0
  140. package/dist/meeting/role-assigner.js.map +1 -0
  141. package/dist/meeting/types.d.ts +21 -0
  142. package/dist/meeting/types.js +2 -0
  143. package/dist/meeting/types.js.map +1 -0
  144. package/dist/meeting-manager.d.ts +10 -0
  145. package/dist/meeting-manager.js +38 -0
  146. package/dist/meeting-manager.js.map +1 -0
  147. package/dist/memory-layer.d.ts +13 -0
  148. package/dist/memory-layer.js +44 -0
  149. package/dist/memory-layer.js.map +1 -0
  150. package/dist/plugin/agend/.claude-plugin/plugin.json +5 -0
  151. package/dist/plugin/agend/.mcp.json +9 -0
  152. package/dist/plugin/ccd-channel/.claude-plugin/plugin.json +5 -0
  153. package/dist/plugin/ccd-channel/.mcp.json +9 -0
  154. package/dist/process-manager.d.ts +31 -0
  155. package/dist/process-manager.js +264 -0
  156. package/dist/process-manager.js.map +1 -0
  157. package/dist/scheduler/db.d.ts +16 -0
  158. package/dist/scheduler/db.js +132 -0
  159. package/dist/scheduler/db.js.map +1 -0
  160. package/dist/scheduler/db.test.d.ts +1 -0
  161. package/dist/scheduler/db.test.js +92 -0
  162. package/dist/scheduler/db.test.js.map +1 -0
  163. package/dist/scheduler/index.d.ts +4 -0
  164. package/dist/scheduler/index.js +4 -0
  165. package/dist/scheduler/index.js.map +1 -0
  166. package/dist/scheduler/scheduler.d.ts +25 -0
  167. package/dist/scheduler/scheduler.js +119 -0
  168. package/dist/scheduler/scheduler.js.map +1 -0
  169. package/dist/scheduler/scheduler.test.d.ts +1 -0
  170. package/dist/scheduler/scheduler.test.js +119 -0
  171. package/dist/scheduler/scheduler.test.js.map +1 -0
  172. package/dist/scheduler/types.d.ts +47 -0
  173. package/dist/scheduler/types.js +7 -0
  174. package/dist/scheduler/types.js.map +1 -0
  175. package/dist/service-installer.d.ts +14 -0
  176. package/dist/service-installer.js +91 -0
  177. package/dist/service-installer.js.map +1 -0
  178. package/dist/setup-wizard.d.ts +14 -0
  179. package/dist/setup-wizard.js +517 -0
  180. package/dist/setup-wizard.js.map +1 -0
  181. package/dist/stt.d.ts +10 -0
  182. package/dist/stt.js +33 -0
  183. package/dist/stt.js.map +1 -0
  184. package/dist/tmux-manager.d.ts +22 -0
  185. package/dist/tmux-manager.js +131 -0
  186. package/dist/tmux-manager.js.map +1 -0
  187. package/dist/topic-commands.d.ts +22 -0
  188. package/dist/topic-commands.js +176 -0
  189. package/dist/topic-commands.js.map +1 -0
  190. package/dist/transcript-monitor.d.ts +21 -0
  191. package/dist/transcript-monitor.js +149 -0
  192. package/dist/transcript-monitor.js.map +1 -0
  193. package/dist/types.d.ts +153 -0
  194. package/dist/types.js +2 -0
  195. package/dist/types.js.map +1 -0
  196. package/dist/webhook-emitter.d.ts +15 -0
  197. package/dist/webhook-emitter.js +41 -0
  198. package/dist/webhook-emitter.js.map +1 -0
  199. package/package.json +60 -4
  200. package/templates/launchd.plist.ejs +29 -0
  201. package/templates/systemd.service.ejs +15 -0
  202. package/index.js +0 -1
@@ -0,0 +1,92 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from "vitest";
2
+ import { mkdtempSync, rmSync } from "node:fs";
3
+ import { join } from "node:path";
4
+ import { tmpdir } from "node:os";
5
+ import { SchedulerDb } from "./db.js";
6
+ describe("SchedulerDb", () => {
7
+ let dir;
8
+ let db;
9
+ beforeEach(() => {
10
+ dir = mkdtempSync(join(tmpdir(), "scheduler-test-"));
11
+ db = new SchedulerDb(join(dir, "scheduler.db"));
12
+ });
13
+ afterEach(() => {
14
+ db.close();
15
+ rmSync(dir, { recursive: true, force: true });
16
+ });
17
+ it("creates tables on init", () => {
18
+ const schedules = db.list();
19
+ expect(schedules).toEqual([]);
20
+ });
21
+ it("creates and retrieves a schedule", () => {
22
+ const s = db.create({
23
+ cron: "0 7 * * *",
24
+ message: "test message",
25
+ source: "proj-a",
26
+ target: "proj-a",
27
+ reply_chat_id: "-100123",
28
+ reply_thread_id: "42",
29
+ label: "daily test",
30
+ timezone: "Asia/Taipei",
31
+ });
32
+ expect(s.id).toBeTruthy();
33
+ expect(s.cron).toBe("0 7 * * *");
34
+ expect(s.enabled).toBe(true);
35
+ const fetched = db.get(s.id);
36
+ expect(fetched).toEqual(s);
37
+ });
38
+ it("lists schedules with optional target filter", () => {
39
+ db.create({ cron: "0 7 * * *", message: "a", source: "a", target: "a", reply_chat_id: "1", reply_thread_id: null });
40
+ db.create({ cron: "0 8 * * *", message: "b", source: "a", target: "b", reply_chat_id: "1", reply_thread_id: null });
41
+ expect(db.list()).toHaveLength(2);
42
+ expect(db.list("a")).toHaveLength(1);
43
+ expect(db.list("b")).toHaveLength(1);
44
+ });
45
+ it("updates a schedule", () => {
46
+ const s = db.create({ cron: "0 7 * * *", message: "old", source: "a", target: "a", reply_chat_id: "1", reply_thread_id: null });
47
+ const updated = db.update(s.id, { message: "new", enabled: false });
48
+ expect(updated.message).toBe("new");
49
+ expect(updated.enabled).toBe(false);
50
+ expect(updated.cron).toBe("0 7 * * *");
51
+ });
52
+ it("deletes a schedule and cascades runs", () => {
53
+ const s = db.create({ cron: "0 7 * * *", message: "x", source: "a", target: "a", reply_chat_id: "1", reply_thread_id: null });
54
+ db.recordRun(s.id, "delivered");
55
+ expect(db.getRuns(s.id)).toHaveLength(1);
56
+ db.delete(s.id);
57
+ expect(db.get(s.id)).toBeNull();
58
+ expect(db.getRuns(s.id)).toHaveLength(0);
59
+ });
60
+ it("deleteByInstanceOrThread removes matching schedules", () => {
61
+ db.create({ cron: "0 7 * * *", message: "a", source: "a", target: "proj-a", reply_chat_id: "1", reply_thread_id: "42" });
62
+ db.create({ cron: "0 8 * * *", message: "b", source: "b", target: "proj-b", reply_chat_id: "1", reply_thread_id: "42" });
63
+ db.create({ cron: "0 9 * * *", message: "c", source: "c", target: "proj-c", reply_chat_id: "1", reply_thread_id: "99" });
64
+ const count = db.deleteByInstanceOrThread("proj-a", "42");
65
+ expect(count).toBe(2);
66
+ expect(db.list()).toHaveLength(1);
67
+ });
68
+ it("records and retrieves runs", () => {
69
+ const s = db.create({ cron: "0 7 * * *", message: "x", source: "a", target: "a", reply_chat_id: "1", reply_thread_id: null });
70
+ db.recordRun(s.id, "delivered");
71
+ db.recordRun(s.id, "instance_offline", "retry 3x failed");
72
+ const runs = db.getRuns(s.id);
73
+ expect(runs).toHaveLength(2);
74
+ expect(runs[0].status).toBe("instance_offline");
75
+ expect(runs[0].detail).toBe("retry 3x failed");
76
+ });
77
+ it("enforces max schedule count", () => {
78
+ for (let i = 0; i < 5; i++) {
79
+ db.create({ cron: "0 7 * * *", message: `m${i}`, source: "a", target: "a", reply_chat_id: "1", reply_thread_id: null });
80
+ }
81
+ expect(() => db.create({ cron: "0 7 * * *", message: "over", source: "a", target: "a", reply_chat_id: "1", reply_thread_id: null }, 5)).toThrow(/limit/i);
82
+ });
83
+ it("prunes old runs on init", () => {
84
+ const s = db.create({ cron: "0 7 * * *", message: "x", source: "a", target: "a", reply_chat_id: "1", reply_thread_id: null });
85
+ db["db"].prepare("INSERT INTO schedule_runs (schedule_id, triggered_at, status) VALUES (?, datetime('now', '-60 days'), 'delivered')").run(s.id);
86
+ db.recordRun(s.id, "delivered");
87
+ db.pruneOldRuns();
88
+ const runs = db.getRuns(s.id);
89
+ expect(runs).toHaveLength(1);
90
+ });
91
+ });
92
+ //# sourceMappingURL=db.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"db.test.js","sourceRoot":"","sources":["../../src/scheduler/db.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ export { Scheduler } from "./scheduler.js";
2
+ export { SchedulerDb } from "./db.js";
3
+ export type { Schedule, ScheduleRun, CreateScheduleParams, UpdateScheduleParams, SchedulerConfig } from "./types.js";
4
+ export { DEFAULT_SCHEDULER_CONFIG } from "./types.js";
@@ -0,0 +1,4 @@
1
+ export { Scheduler } from "./scheduler.js";
2
+ export { SchedulerDb } from "./db.js";
3
+ export { DEFAULT_SCHEDULER_CONFIG } from "./types.js";
4
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/scheduler/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAC3C,OAAO,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAEtC,OAAO,EAAE,wBAAwB,EAAE,MAAM,YAAY,CAAC"}
@@ -0,0 +1,25 @@
1
+ import type { Schedule, CreateScheduleParams, UpdateScheduleParams, SchedulerConfig, ScheduleRun } from "./types.js";
2
+ export declare class Scheduler {
3
+ private db;
4
+ private jobs;
5
+ private onTrigger;
6
+ private config;
7
+ private isValidInstance;
8
+ constructor(dbPath: string, onTrigger: (schedule: Schedule) => void, config: SchedulerConfig, isValidInstance: (name: string) => boolean);
9
+ init(): void;
10
+ reload(): void;
11
+ shutdown(): void;
12
+ create(params: CreateScheduleParams): Schedule;
13
+ list(target?: string): Schedule[];
14
+ get(id: string): Schedule | null;
15
+ update(id: string, params: UpdateScheduleParams): Schedule;
16
+ delete(id: string): void;
17
+ trigger(id: string): void;
18
+ deleteByInstanceOrThread(instanceName: string, threadId: string): number;
19
+ recordRun(scheduleId: string, status: string, detail?: string): void;
20
+ getRuns(scheduleId: string, limit?: number): ScheduleRun[];
21
+ private registerAllJobs;
22
+ private registerJob;
23
+ private stopJob;
24
+ private stopAllJobs;
25
+ }
@@ -0,0 +1,119 @@
1
+ import { Cron } from "croner";
2
+ import { SchedulerDb } from "./db.js";
3
+ export class Scheduler {
4
+ db;
5
+ jobs = new Map();
6
+ onTrigger;
7
+ config;
8
+ isValidInstance;
9
+ constructor(dbPath, onTrigger, config, isValidInstance) {
10
+ this.db = new SchedulerDb(dbPath);
11
+ this.onTrigger = onTrigger;
12
+ this.config = config;
13
+ this.isValidInstance = isValidInstance;
14
+ }
15
+ init() {
16
+ this.db.pruneOldRuns();
17
+ this.registerAllJobs();
18
+ }
19
+ reload() {
20
+ this.stopAllJobs();
21
+ this.registerAllJobs();
22
+ }
23
+ shutdown() {
24
+ this.stopAllJobs();
25
+ this.db.close();
26
+ }
27
+ create(params) {
28
+ try {
29
+ new Cron(params.cron, { timezone: params.timezone ?? this.config.default_timezone });
30
+ }
31
+ catch (err) {
32
+ throw new Error(`Invalid cron expression: ${err.message}`);
33
+ }
34
+ if (!this.isValidInstance(params.target)) {
35
+ throw new Error(`Instance "${params.target}" not found in fleet config.`);
36
+ }
37
+ const schedule = this.db.create(params, this.config.max_schedules);
38
+ this.registerJob(schedule);
39
+ return schedule;
40
+ }
41
+ list(target) {
42
+ return this.db.list(target);
43
+ }
44
+ get(id) {
45
+ return this.db.get(id);
46
+ }
47
+ update(id, params) {
48
+ if (params.cron !== undefined) {
49
+ try {
50
+ new Cron(params.cron, { timezone: this.db.get(id)?.timezone ?? this.config.default_timezone });
51
+ }
52
+ catch (err) {
53
+ throw new Error(`Invalid cron expression: ${err.message}`);
54
+ }
55
+ }
56
+ if (params.target !== undefined && !this.isValidInstance(params.target)) {
57
+ throw new Error(`Instance "${params.target}" not found in fleet config.`);
58
+ }
59
+ const updated = this.db.update(id, params);
60
+ this.stopJob(id);
61
+ if (updated.enabled) {
62
+ this.registerJob(updated);
63
+ }
64
+ return updated;
65
+ }
66
+ delete(id) {
67
+ this.stopJob(id);
68
+ this.db.delete(id);
69
+ }
70
+ trigger(id) {
71
+ const schedule = this.db.get(id);
72
+ if (!schedule)
73
+ throw new Error(`Schedule "${id}" not found.`);
74
+ this.onTrigger(schedule);
75
+ }
76
+ deleteByInstanceOrThread(instanceName, threadId) {
77
+ const affected = this.db.list().filter((s) => s.target === instanceName || s.reply_thread_id === threadId);
78
+ for (const s of affected) {
79
+ this.stopJob(s.id);
80
+ }
81
+ return this.db.deleteByInstanceOrThread(instanceName, threadId);
82
+ }
83
+ recordRun(scheduleId, status, detail) {
84
+ this.db.recordRun(scheduleId, status, detail);
85
+ }
86
+ getRuns(scheduleId, limit) {
87
+ return this.db.getRuns(scheduleId, limit);
88
+ }
89
+ registerAllJobs() {
90
+ for (const schedule of this.db.list()) {
91
+ if (schedule.enabled) {
92
+ this.registerJob(schedule);
93
+ }
94
+ }
95
+ }
96
+ registerJob(schedule) {
97
+ const job = new Cron(schedule.cron, { timezone: schedule.timezone }, () => {
98
+ const current = this.db.get(schedule.id);
99
+ if (current && current.enabled) {
100
+ this.onTrigger(current);
101
+ }
102
+ });
103
+ this.jobs.set(schedule.id, job);
104
+ }
105
+ stopJob(id) {
106
+ const job = this.jobs.get(id);
107
+ if (job) {
108
+ job.stop();
109
+ this.jobs.delete(id);
110
+ }
111
+ }
112
+ stopAllJobs() {
113
+ for (const [, job] of this.jobs) {
114
+ job.stop();
115
+ }
116
+ this.jobs.clear();
117
+ }
118
+ }
119
+ //# sourceMappingURL=scheduler.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scheduler.js","sourceRoot":"","sources":["../../src/scheduler/scheduler.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,119 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from "vitest";
2
+ import { mkdtempSync, rmSync } from "node:fs";
3
+ import { join } from "node:path";
4
+ import { tmpdir } from "node:os";
5
+ import { Scheduler } from "./scheduler.js";
6
+ import { DEFAULT_SCHEDULER_CONFIG } from "./types.js";
7
+ describe("Scheduler", () => {
8
+ let dir;
9
+ let scheduler;
10
+ let triggered;
11
+ beforeEach(() => {
12
+ dir = mkdtempSync(join(tmpdir(), "scheduler-engine-test-"));
13
+ triggered = [];
14
+ scheduler = new Scheduler(join(dir, "scheduler.db"), (schedule) => { triggered.push(schedule); }, DEFAULT_SCHEDULER_CONFIG, (instanceName) => true);
15
+ scheduler.init();
16
+ });
17
+ afterEach(() => {
18
+ scheduler.shutdown();
19
+ rmSync(dir, { recursive: true, force: true });
20
+ });
21
+ it("creates a schedule and registers cron job", () => {
22
+ const s = scheduler.create({
23
+ cron: "0 7 * * *",
24
+ message: "hello",
25
+ source: "proj-a",
26
+ target: "proj-a",
27
+ reply_chat_id: "1",
28
+ reply_thread_id: null,
29
+ });
30
+ expect(s.id).toBeTruthy();
31
+ expect(scheduler.list()).toHaveLength(1);
32
+ });
33
+ it("rejects invalid cron expression", () => {
34
+ expect(() => scheduler.create({
35
+ cron: "not a cron",
36
+ message: "hello",
37
+ source: "a",
38
+ target: "a",
39
+ reply_chat_id: "1",
40
+ reply_thread_id: null,
41
+ })).toThrow(/cron/i);
42
+ });
43
+ it("rejects invalid target instance", () => {
44
+ const s2 = new Scheduler(join(dir, "scheduler2.db"), () => { }, DEFAULT_SCHEDULER_CONFIG, (name) => name === "proj-a");
45
+ s2.init();
46
+ expect(() => s2.create({
47
+ cron: "0 7 * * *",
48
+ message: "hello",
49
+ source: "proj-a",
50
+ target: "nonexistent",
51
+ reply_chat_id: "1",
52
+ reply_thread_id: null,
53
+ })).toThrow(/not found/i);
54
+ s2.shutdown();
55
+ });
56
+ it("manual trigger calls onTrigger callback", () => {
57
+ const s = scheduler.create({
58
+ cron: "0 7 * * *",
59
+ message: "hello",
60
+ source: "proj-a",
61
+ target: "proj-a",
62
+ reply_chat_id: "1",
63
+ reply_thread_id: null,
64
+ });
65
+ scheduler.trigger(s.id);
66
+ expect(triggered).toHaveLength(1);
67
+ expect(triggered[0].id).toBe(s.id);
68
+ });
69
+ it("delete removes schedule and cron job", () => {
70
+ const s = scheduler.create({
71
+ cron: "0 7 * * *",
72
+ message: "hello",
73
+ source: "a",
74
+ target: "a",
75
+ reply_chat_id: "1",
76
+ reply_thread_id: null,
77
+ });
78
+ scheduler.delete(s.id);
79
+ expect(scheduler.list()).toHaveLength(0);
80
+ });
81
+ it("update reschedules cron job", () => {
82
+ const s = scheduler.create({
83
+ cron: "0 7 * * *",
84
+ message: "hello",
85
+ source: "a",
86
+ target: "a",
87
+ reply_chat_id: "1",
88
+ reply_thread_id: null,
89
+ });
90
+ const updated = scheduler.update(s.id, { cron: "0 8 * * *" });
91
+ expect(updated.cron).toBe("0 8 * * *");
92
+ });
93
+ it("reload clears and re-registers all jobs", () => {
94
+ scheduler.create({
95
+ cron: "0 7 * * *",
96
+ message: "hello",
97
+ source: "a",
98
+ target: "a",
99
+ reply_chat_id: "1",
100
+ reply_thread_id: null,
101
+ });
102
+ scheduler.reload();
103
+ expect(scheduler.list()).toHaveLength(1);
104
+ });
105
+ it("deleteByInstanceOrThread cleans up and removes cron jobs", () => {
106
+ scheduler.create({
107
+ cron: "0 7 * * *",
108
+ message: "a",
109
+ source: "a",
110
+ target: "proj-a",
111
+ reply_chat_id: "1",
112
+ reply_thread_id: "42",
113
+ });
114
+ const count = scheduler.deleteByInstanceOrThread("proj-a", "42");
115
+ expect(count).toBe(1);
116
+ expect(scheduler.list()).toHaveLength(0);
117
+ });
118
+ });
119
+ //# sourceMappingURL=scheduler.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scheduler.test.js","sourceRoot":"","sources":["../../src/scheduler/scheduler.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,47 @@
1
+ export interface Schedule {
2
+ id: string;
3
+ cron: string;
4
+ message: string;
5
+ source: string;
6
+ target: string;
7
+ reply_chat_id: string;
8
+ reply_thread_id: string | null;
9
+ label: string | null;
10
+ enabled: boolean;
11
+ timezone: string;
12
+ created_at: string;
13
+ last_triggered_at: string | null;
14
+ last_status: string | null;
15
+ }
16
+ export interface ScheduleRun {
17
+ id: number;
18
+ schedule_id: string;
19
+ triggered_at: string;
20
+ status: "delivered" | "delivered_fallback" | "retry" | "instance_offline" | "channel_dead";
21
+ detail: string | null;
22
+ }
23
+ export interface CreateScheduleParams {
24
+ cron: string;
25
+ message: string;
26
+ source: string;
27
+ target: string;
28
+ reply_chat_id: string;
29
+ reply_thread_id: string | null;
30
+ label?: string;
31
+ timezone?: string;
32
+ }
33
+ export interface UpdateScheduleParams {
34
+ cron?: string;
35
+ message?: string;
36
+ target?: string;
37
+ label?: string;
38
+ timezone?: string;
39
+ enabled?: boolean;
40
+ }
41
+ export interface SchedulerConfig {
42
+ max_schedules: number;
43
+ default_timezone: string;
44
+ retry_count: number;
45
+ retry_interval_ms: number;
46
+ }
47
+ export declare const DEFAULT_SCHEDULER_CONFIG: SchedulerConfig;
@@ -0,0 +1,7 @@
1
+ export const DEFAULT_SCHEDULER_CONFIG = {
2
+ max_schedules: 100,
3
+ default_timezone: "Asia/Taipei",
4
+ retry_count: 3,
5
+ retry_interval_ms: 30_000,
6
+ };
7
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/scheduler/types.ts"],"names":[],"mappings":"AAmDA,MAAM,CAAC,MAAM,wBAAwB,GAAoB;IACvD,aAAa,EAAE,GAAG;IAClB,gBAAgB,EAAE,aAAa;IAC/B,WAAW,EAAE,CAAC;IACd,iBAAiB,EAAE,MAAM;CAC1B,CAAC"}
@@ -0,0 +1,14 @@
1
+ interface ServiceVars {
2
+ label: string;
3
+ execPath: string;
4
+ path?: string;
5
+ workingDirectory: string;
6
+ logPath: string;
7
+ }
8
+ export declare function detectPlatform(): "macos" | "linux";
9
+ export declare function renderLaunchdPlist(vars: ServiceVars): string;
10
+ export declare function renderSystemdUnit(vars: ServiceVars): string;
11
+ export declare function uninstallService(label: string): boolean;
12
+ export declare function installService(vars: ServiceVars): string;
13
+ export declare function activateService(plistPath: string, pidPath: string): void;
14
+ export {};
@@ -0,0 +1,91 @@
1
+ import { readFileSync, writeFileSync, mkdirSync, existsSync, unlinkSync } from "node:fs";
2
+ import { join, dirname } from "node:path";
3
+ import { fileURLToPath } from "node:url";
4
+ import { execSync } from "node:child_process";
5
+ import ejs from "ejs";
6
+ const { render } = ejs;
7
+ import { platform } from "node:os";
8
+ const __dirname = dirname(fileURLToPath(import.meta.url));
9
+ const templatesDir = join(__dirname, "..", "templates");
10
+ export function detectPlatform() {
11
+ return platform() === "darwin" ? "macos" : "linux";
12
+ }
13
+ function withDefaults(vars) {
14
+ return { ...vars, path: vars.path ?? process.env.PATH ?? "" };
15
+ }
16
+ export function renderLaunchdPlist(vars) {
17
+ const template = readFileSync(join(templatesDir, "launchd.plist.ejs"), "utf-8");
18
+ return render(template, withDefaults(vars));
19
+ }
20
+ export function renderSystemdUnit(vars) {
21
+ const template = readFileSync(join(templatesDir, "systemd.service.ejs"), "utf-8");
22
+ return render(template, withDefaults(vars));
23
+ }
24
+ export function uninstallService(label) {
25
+ const plat = detectPlatform();
26
+ if (plat === "macos") {
27
+ const plistPath = join(process.env.HOME, "Library/LaunchAgents", `${label}.plist`);
28
+ if (existsSync(plistPath)) {
29
+ unlinkSync(plistPath);
30
+ return true;
31
+ }
32
+ }
33
+ else {
34
+ const unitPath = join(process.env.HOME, ".config/systemd/user", `${label}.service`);
35
+ if (existsSync(unitPath)) {
36
+ unlinkSync(unitPath);
37
+ return true;
38
+ }
39
+ }
40
+ return false;
41
+ }
42
+ export function installService(vars) {
43
+ const plat = detectPlatform();
44
+ if (plat === "macos") {
45
+ const plistPath = join(process.env.HOME, "Library/LaunchAgents", `${vars.label}.plist`);
46
+ mkdirSync(dirname(plistPath), { recursive: true });
47
+ writeFileSync(plistPath, renderLaunchdPlist(vars));
48
+ return plistPath;
49
+ }
50
+ else {
51
+ const unitPath = join(process.env.HOME, ".config/systemd/user", `${vars.label}.service`);
52
+ mkdirSync(dirname(unitPath), { recursive: true });
53
+ writeFileSync(unitPath, renderSystemdUnit(vars));
54
+ return unitPath;
55
+ }
56
+ }
57
+ export function activateService(plistPath, pidPath) {
58
+ // Kill manually-running fleet if present
59
+ if (existsSync(pidPath)) {
60
+ const pid = parseInt(readFileSync(pidPath, "utf-8").trim(), 10);
61
+ try {
62
+ process.kill(pid, "SIGTERM");
63
+ // Wait briefly for process to exit
64
+ for (let i = 0; i < 20; i++) {
65
+ try {
66
+ process.kill(pid, 0);
67
+ }
68
+ catch {
69
+ break;
70
+ }
71
+ execSync("sleep 0.1");
72
+ }
73
+ }
74
+ catch {
75
+ // Process already gone
76
+ }
77
+ try {
78
+ unlinkSync(pidPath);
79
+ }
80
+ catch { }
81
+ }
82
+ const plat = detectPlatform();
83
+ if (plat === "macos") {
84
+ execSync(`launchctl load ${plistPath}`, { stdio: "inherit" });
85
+ }
86
+ else {
87
+ const serviceName = plistPath.replace(/.*\//, "").replace(/\.service$/, "");
88
+ execSync(`systemctl --user enable --now ${serviceName}`, { stdio: "inherit" });
89
+ }
90
+ }
91
+ //# sourceMappingURL=service-installer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"service-installer.js","sourceRoot":"","sources":["../src/service-installer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,14 @@
1
+ export declare function validateBotToken(token: string): boolean;
2
+ export declare function verifyBotToken(token: string): Promise<{
3
+ valid: boolean;
4
+ username: string | null;
5
+ }>;
6
+ interface PrereqResult {
7
+ claude: boolean;
8
+ claudeVersion: string;
9
+ tmux: boolean;
10
+ tmuxVersion: string;
11
+ }
12
+ export declare function checkPrerequisites(): PrereqResult;
13
+ export declare function runSetupWizard(): Promise<void>;
14
+ export {};