devchain-cli 0.13.1 → 0.14.1

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 (144) hide show
  1. package/dist/drizzle/0064_living_morlun.sql +59 -0
  2. package/dist/drizzle/meta/0064_snapshot.json +5609 -0
  3. package/dist/drizzle/meta/_journal.json +7 -0
  4. package/dist/node_modules/@devchain/shared/schemas/export-schema.d.ts +74 -6
  5. package/dist/node_modules/@devchain/shared/schemas/export-schema.d.ts.map +1 -1
  6. package/dist/node_modules/@devchain/shared/schemas/export-schema.js +19 -0
  7. package/dist/node_modules/@devchain/shared/schemas/export-schema.js.map +1 -1
  8. package/dist/node_modules/@devchain/shared/tsconfig.tsbuildinfo +1 -1
  9. package/dist/server/app.main.module.js +2 -0
  10. package/dist/server/app.main.module.js.map +1 -1
  11. package/dist/server/app.normal.module.js +2 -0
  12. package/dist/server/app.normal.module.js.map +1 -1
  13. package/dist/server/modules/agents/agents.module.js +2 -1
  14. package/dist/server/modules/agents/agents.module.js.map +1 -1
  15. package/dist/server/modules/agents/controllers/agents.controller.d.ts +3 -1
  16. package/dist/server/modules/agents/controllers/agents.controller.js +12 -2
  17. package/dist/server/modules/agents/controllers/agents.controller.js.map +1 -1
  18. package/dist/server/modules/events/catalog/broadcast-registry.js +21 -0
  19. package/dist/server/modules/events/catalog/broadcast-registry.js.map +1 -1
  20. package/dist/server/modules/events/catalog/guest.unregistered.d.ts +2 -2
  21. package/dist/server/modules/events/catalog/index.d.ts +50 -2
  22. package/dist/server/modules/events/catalog/index.js +5 -1
  23. package/dist/server/modules/events/catalog/index.js.map +1 -1
  24. package/dist/server/modules/events/catalog/scheduled-epic.executed.d.ts +50 -0
  25. package/dist/server/modules/events/catalog/scheduled-epic.executed.js +33 -0
  26. package/dist/server/modules/events/catalog/scheduled-epic.executed.js.map +1 -0
  27. package/dist/server/modules/events/catalog/team.member.added.d.ts +3 -0
  28. package/dist/server/modules/events/catalog/team.member.added.js +1 -0
  29. package/dist/server/modules/events/catalog/team.member.added.js.map +1 -1
  30. package/dist/server/modules/mcp/dtos/mcp.dto.d.ts +26 -51
  31. package/dist/server/modules/mcp/dtos/mcp.dto.js.map +1 -1
  32. package/dist/server/modules/mcp/services/handlers/document-tools.js +4 -2
  33. package/dist/server/modules/mcp/services/handlers/document-tools.js.map +1 -1
  34. package/dist/server/modules/mcp/services/handlers/epic-tools.js +12 -17
  35. package/dist/server/modules/mcp/services/handlers/epic-tools.js.map +1 -1
  36. package/dist/server/modules/mcp/services/handlers/record-tools.js +2 -14
  37. package/dist/server/modules/mcp/services/handlers/record-tools.js.map +1 -1
  38. package/dist/server/modules/mcp/services/handlers/review-tools.js +7 -60
  39. package/dist/server/modules/mcp/services/handlers/review-tools.js.map +1 -1
  40. package/dist/server/modules/mcp/services/handlers/session-tools.js +0 -5
  41. package/dist/server/modules/mcp/services/handlers/session-tools.js.map +1 -1
  42. package/dist/server/modules/mcp/services/handlers/teams-tools.js +5 -2
  43. package/dist/server/modules/mcp/services/handlers/teams-tools.js.map +1 -1
  44. package/dist/server/modules/mcp/services/mappers/dto-mappers.js +0 -2
  45. package/dist/server/modules/mcp/services/mappers/dto-mappers.js.map +1 -1
  46. package/dist/server/modules/projects/controllers/projects.controller.d.ts +33 -0
  47. package/dist/server/modules/projects/helpers/project-export.d.ts +14 -0
  48. package/dist/server/modules/projects/helpers/project-export.js +44 -0
  49. package/dist/server/modules/projects/helpers/project-export.js.map +1 -1
  50. package/dist/server/modules/projects/helpers/project-import.d.ts +13 -0
  51. package/dist/server/modules/projects/helpers/project-import.js +66 -2
  52. package/dist/server/modules/projects/helpers/project-import.js.map +1 -1
  53. package/dist/server/modules/projects/helpers/template-loader.d.ts +5 -0
  54. package/dist/server/modules/projects/helpers/template-loader.js +12 -0
  55. package/dist/server/modules/projects/helpers/template-loader.js.map +1 -1
  56. package/dist/server/modules/projects/projects.module.js +2 -0
  57. package/dist/server/modules/projects/projects.module.js.map +1 -1
  58. package/dist/server/modules/projects/services/projects.service.d.ts +22 -1
  59. package/dist/server/modules/projects/services/projects.service.js +11 -2
  60. package/dist/server/modules/projects/services/projects.service.js.map +1 -1
  61. package/dist/server/modules/providers/adapters/claude.adapter.js +1 -1
  62. package/dist/server/modules/providers/adapters/claude.adapter.js.map +1 -1
  63. package/dist/server/modules/registry/controllers/templates.controller.d.ts +15 -1
  64. package/dist/server/modules/scheduled-epics/controllers/scheduled-epics.controller.d.ts +17 -0
  65. package/dist/server/modules/scheduled-epics/controllers/scheduled-epics.controller.js +167 -0
  66. package/dist/server/modules/scheduled-epics/controllers/scheduled-epics.controller.js.map +1 -0
  67. package/dist/server/modules/scheduled-epics/dtos/scheduled-epic.dto.d.ts +89 -0
  68. package/dist/server/modules/scheduled-epics/dtos/scheduled-epic.dto.js +91 -0
  69. package/dist/server/modules/scheduled-epics/dtos/scheduled-epic.dto.js.map +1 -0
  70. package/dist/server/modules/scheduled-epics/helpers/cron-helpers.d.ts +8 -0
  71. package/dist/server/modules/scheduled-epics/helpers/cron-helpers.js +20 -0
  72. package/dist/server/modules/scheduled-epics/helpers/cron-helpers.js.map +1 -0
  73. package/dist/server/modules/scheduled-epics/helpers/template-helpers.d.ts +8 -0
  74. package/dist/server/modules/scheduled-epics/helpers/template-helpers.js +19 -0
  75. package/dist/server/modules/scheduled-epics/helpers/template-helpers.js.map +1 -0
  76. package/dist/server/modules/scheduled-epics/helpers/timezone-helpers.d.ts +7 -0
  77. package/dist/server/modules/scheduled-epics/helpers/timezone-helpers.js +16 -0
  78. package/dist/server/modules/scheduled-epics/helpers/timezone-helpers.js.map +1 -0
  79. package/dist/server/modules/scheduled-epics/scheduled-epics.module.d.ts +2 -0
  80. package/dist/server/modules/scheduled-epics/scheduled-epics.module.js +35 -0
  81. package/dist/server/modules/scheduled-epics/scheduled-epics.module.js.map +1 -0
  82. package/dist/server/modules/scheduled-epics/services/scheduled-epic-runner.service.d.ts +33 -0
  83. package/dist/server/modules/scheduled-epics/services/scheduled-epic-runner.service.js +434 -0
  84. package/dist/server/modules/scheduled-epics/services/scheduled-epic-runner.service.js.map +1 -0
  85. package/dist/server/modules/scheduled-epics/services/scheduled-epics.service.d.ts +23 -0
  86. package/dist/server/modules/scheduled-epics/services/scheduled-epics.service.js +136 -0
  87. package/dist/server/modules/scheduled-epics/services/scheduled-epics.service.js.map +1 -0
  88. package/dist/server/modules/seeders/seeders/0008_seed_remove_claude_no_flicker_env.d.ts +3 -0
  89. package/dist/server/modules/seeders/seeders/0008_seed_remove_claude_no_flicker_env.js +30 -0
  90. package/dist/server/modules/seeders/seeders/0008_seed_remove_claude_no_flicker_env.js.map +1 -0
  91. package/dist/server/modules/seeders/services/data-seeder.service.js +2 -0
  92. package/dist/server/modules/seeders/services/data-seeder.service.js.map +1 -1
  93. package/dist/server/modules/session-reader/data/pricing.json +65 -1
  94. package/dist/server/modules/settings/local/delegates/preset-settings.delegate.d.ts +1 -0
  95. package/dist/server/modules/settings/local/delegates/preset-settings.delegate.js +36 -0
  96. package/dist/server/modules/settings/local/delegates/preset-settings.delegate.js.map +1 -1
  97. package/dist/server/modules/settings/services/settings.service.d.ts +1 -0
  98. package/dist/server/modules/settings/services/settings.service.js +3 -0
  99. package/dist/server/modules/settings/services/settings.service.js.map +1 -1
  100. package/dist/server/modules/storage/db/schema.d.ts +616 -0
  101. package/dist/server/modules/storage/db/schema.js +62 -2
  102. package/dist/server/modules/storage/db/schema.js.map +1 -1
  103. package/dist/server/modules/storage/interfaces/storage.interface.d.ts +26 -2
  104. package/dist/server/modules/storage/interfaces/storage.interface.js.map +1 -1
  105. package/dist/server/modules/storage/local/delegates/provider.delegate.js +1 -5
  106. package/dist/server/modules/storage/local/delegates/provider.delegate.js.map +1 -1
  107. package/dist/server/modules/storage/local/delegates/scheduled-epic.delegate.d.ts +20 -0
  108. package/dist/server/modules/storage/local/delegates/scheduled-epic.delegate.js +328 -0
  109. package/dist/server/modules/storage/local/delegates/scheduled-epic.delegate.js.map +1 -0
  110. package/dist/server/modules/storage/local/local-storage.service.d.ts +15 -2
  111. package/dist/server/modules/storage/local/local-storage.service.js +38 -0
  112. package/dist/server/modules/storage/local/local-storage.service.js.map +1 -1
  113. package/dist/server/modules/storage/models/domain.models.d.ts +46 -0
  114. package/dist/server/modules/subscribers/events/event-fields-catalog.js +21 -0
  115. package/dist/server/modules/subscribers/events/event-fields-catalog.js.map +1 -1
  116. package/dist/server/modules/teams/services/teams.service.d.ts +3 -1
  117. package/dist/server/modules/teams/services/teams.service.js +21 -2
  118. package/dist/server/modules/teams/services/teams.service.js.map +1 -1
  119. package/dist/server/modules/teams/subscribers/team-membership-changed-notifier.subscriber.js +3 -1
  120. package/dist/server/modules/teams/subscribers/team-membership-changed-notifier.subscriber.js.map +1 -1
  121. package/dist/server/modules/teams/teams.module.js +2 -1
  122. package/dist/server/modules/teams/teams.module.js.map +1 -1
  123. package/dist/server/modules/terminal/dtos/ws-envelope.dto.d.ts +4 -4
  124. package/dist/server/modules/terminal/gateways/terminal.gateway.js +7 -7
  125. package/dist/server/modules/terminal/gateways/terminal.gateway.js.map +1 -1
  126. package/dist/server/modules/terminal/services/terminal-seed.service.js +1 -1
  127. package/dist/server/modules/terminal/services/terminal-seed.service.js.map +1 -1
  128. package/dist/server/modules/terminal/services/terminal-session/terminal-session.d.ts +7 -0
  129. package/dist/server/modules/terminal/services/terminal-session/terminal-session.js +18 -7
  130. package/dist/server/modules/terminal/services/terminal-session/terminal-session.js.map +1 -1
  131. package/dist/server/modules/terminal/utils/normalize-line-endings.d.ts +1 -0
  132. package/dist/server/modules/terminal/utils/normalize-line-endings.js +8 -0
  133. package/dist/server/modules/terminal/utils/normalize-line-endings.js.map +1 -1
  134. package/dist/server/templates/teams-dev.json +180 -191
  135. package/dist/server/tsconfig.tsbuildinfo +1 -1
  136. package/dist/server/ui/assets/{ReviewDetailPage-CEqKCtbS.js → ReviewDetailPage-CobRKQBn.js} +1 -1
  137. package/dist/server/ui/assets/{ReviewsPage-KUZqxdFY.js → ReviewsPage-Bb6ZmriH.js} +1 -1
  138. package/dist/server/ui/assets/{index-BFSzo9O6.js → index-BV_-Jlz8.js} +242 -232
  139. package/dist/server/ui/assets/index-C_ZOt0it.css +32 -0
  140. package/dist/server/ui/assets/{useReviewSubscription-q_e_8Dd4.js → useReviewSubscription-DzaIaXy7.js} +1 -1
  141. package/dist/server/ui/index.html +2 -2
  142. package/dist/templates/teams-dev.json +180 -191
  143. package/package.json +14 -1
  144. package/dist/server/ui/assets/index-59euy2RV.css +0 -32
@@ -0,0 +1,33 @@
1
+ import { OnModuleDestroy, OnModuleInit } from '@nestjs/common';
2
+ import { type StorageService } from '../../storage/interfaces/storage.interface';
3
+ import type { ScheduledEpicRun } from '../../storage/models/domain.models';
4
+ import { EpicsService } from '../../epics/services/epics.service';
5
+ import { EventsService } from '../../events/services/events.service';
6
+ import type { ScheduledEpicRunnerRefresh } from './scheduled-epics.service';
7
+ export declare class ScheduledEpicRunnerService implements ScheduledEpicRunnerRefresh, OnModuleInit, OnModuleDestroy {
8
+ private readonly storage;
9
+ private readonly epicsService;
10
+ private readonly eventsService;
11
+ private wakeTimer;
12
+ private scanning;
13
+ constructor(storage: StorageService, epicsService: EpicsService, eventsService: EventsService);
14
+ onModuleInit(): void;
15
+ onModuleDestroy(): void;
16
+ refreshScheduleWindow(): void;
17
+ private scheduleNextWake;
18
+ private clearWakeTimer;
19
+ private wake;
20
+ private scanAndExecute;
21
+ private scanProject;
22
+ private processSchedule;
23
+ private computeMissedSlots;
24
+ private claimAndExecute;
25
+ private atomicClaim;
26
+ private drainPendingRuns;
27
+ executeRun(run: ScheduledEpicRun): Promise<void>;
28
+ private executeClaimedRun;
29
+ private publishRunOutcome;
30
+ private recoverStaleRuns;
31
+ private advanceNextRunAt;
32
+ private buildTemplateVars;
33
+ }
@@ -0,0 +1,434 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ var __metadata = (this && this.__metadata) || function (k, v) {
9
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
10
+ };
11
+ var __param = (this && this.__param) || function (paramIndex, decorator) {
12
+ return function (target, key) { decorator(target, key, paramIndex); }
13
+ };
14
+ Object.defineProperty(exports, "__esModule", { value: true });
15
+ exports.ScheduledEpicRunnerService = void 0;
16
+ const common_1 = require("@nestjs/common");
17
+ const storage_interface_1 = require("../../storage/interfaces/storage.interface");
18
+ const epics_service_1 = require("../../epics/services/epics.service");
19
+ const events_service_1 = require("../../events/services/events.service");
20
+ const logger_1 = require("../../../common/logging/logger");
21
+ const cron_helpers_1 = require("../helpers/cron-helpers");
22
+ const template_helpers_1 = require("../helpers/template-helpers");
23
+ const logger = (0, logger_1.createLogger)('ScheduledEpicRunnerService');
24
+ const SCAN_INTERVAL_MS = 60_000;
25
+ const MIN_WAKE_MS = 1_000;
26
+ const MAX_CATCHUP_RUNS_PER_SCAN = 10;
27
+ const CATCHUP_HORIZON_MS = 24 * 60 * 60 * 1_000;
28
+ const STALE_RUNNING_THRESHOLD_MS = 15 * 60 * 1_000;
29
+ let ScheduledEpicRunnerService = class ScheduledEpicRunnerService {
30
+ constructor(storage, epicsService, eventsService) {
31
+ this.storage = storage;
32
+ this.epicsService = epicsService;
33
+ this.eventsService = eventsService;
34
+ this.wakeTimer = null;
35
+ this.scanning = false;
36
+ }
37
+ onModuleInit() {
38
+ this.scheduleNextWake(MIN_WAKE_MS);
39
+ logger.info('Runner initialized');
40
+ }
41
+ onModuleDestroy() {
42
+ this.clearWakeTimer();
43
+ logger.info('Runner destroyed');
44
+ }
45
+ refreshScheduleWindow() {
46
+ this.scheduleNextWake(MIN_WAKE_MS);
47
+ }
48
+ scheduleNextWake(delayMs) {
49
+ this.clearWakeTimer();
50
+ const clamped = Math.max(MIN_WAKE_MS, delayMs);
51
+ this.wakeTimer = setTimeout(() => this.wake(), clamped);
52
+ }
53
+ clearWakeTimer() {
54
+ if (this.wakeTimer) {
55
+ clearTimeout(this.wakeTimer);
56
+ this.wakeTimer = null;
57
+ }
58
+ }
59
+ async wake() {
60
+ if (this.scanning) {
61
+ this.scheduleNextWake(SCAN_INTERVAL_MS);
62
+ return;
63
+ }
64
+ this.scanning = true;
65
+ try {
66
+ await this.scanAndExecute();
67
+ }
68
+ catch (error) {
69
+ logger.error({ error: String(error) }, 'Scan cycle failed');
70
+ }
71
+ finally {
72
+ this.scanning = false;
73
+ this.scheduleNextWake(SCAN_INTERVAL_MS);
74
+ }
75
+ }
76
+ async scanAndExecute() {
77
+ const projects = await this.storage.listProjects({ limit: 500 });
78
+ for (const project of projects.items) {
79
+ try {
80
+ await this.scanProject(project.id);
81
+ }
82
+ catch (error) {
83
+ logger.error({ projectId: project.id, error: String(error) }, 'Failed to scan project schedules');
84
+ }
85
+ }
86
+ }
87
+ async scanProject(projectId) {
88
+ const now = new Date().toISOString();
89
+ await this.recoverStaleRuns(projectId, now);
90
+ const dueSchedules = await this.storage.listDueScheduledEpics(projectId, now);
91
+ for (const schedule of dueSchedules) {
92
+ try {
93
+ await this.processSchedule(schedule, now);
94
+ }
95
+ catch (error) {
96
+ logger.error({ scheduleId: schedule.id, error: String(error) }, 'Failed to process schedule');
97
+ }
98
+ }
99
+ const allSchedules = await this.storage.listScheduledEpics(projectId, { limit: 500 });
100
+ for (const schedule of allSchedules.items) {
101
+ try {
102
+ await this.drainPendingRuns(schedule.id);
103
+ }
104
+ catch (error) {
105
+ logger.error({ scheduleId: schedule.id, error: String(error) }, 'Failed to drain pending runs');
106
+ }
107
+ }
108
+ }
109
+ async processSchedule(schedule, now) {
110
+ if (!schedule.enabled || !schedule.nextRunAt)
111
+ return;
112
+ const missedSlots = this.computeMissedSlots(schedule, now);
113
+ if (missedSlots.length <= 1) {
114
+ await this.claimAndExecute(schedule, schedule.nextRunAt, 'scheduler');
115
+ await this.advanceNextRunAt(schedule);
116
+ return;
117
+ }
118
+ switch (schedule.missedRunPolicy) {
119
+ case 'skip':
120
+ logger.info({ scheduleId: schedule.id, missed: missedSlots.length }, 'Skipping missed runs per policy');
121
+ await this.advanceNextRunAt(schedule);
122
+ break;
123
+ case 'run_once':
124
+ await this.claimAndExecute(schedule, missedSlots[missedSlots.length - 1], 'scheduler');
125
+ await this.advanceNextRunAt(schedule);
126
+ break;
127
+ case 'run_all': {
128
+ const slotsToProcess = missedSlots.slice(0, MAX_CATCHUP_RUNS_PER_SCAN);
129
+ for (const slot of slotsToProcess) {
130
+ await this.claimAndExecute(schedule, slot, 'scheduler');
131
+ }
132
+ if (missedSlots.length > MAX_CATCHUP_RUNS_PER_SCAN) {
133
+ const firstUnprocessed = missedSlots[MAX_CATCHUP_RUNS_PER_SCAN];
134
+ await this.storage.updateScheduledEpicRuntimeState(schedule.id, {
135
+ nextRunAt: firstUnprocessed,
136
+ });
137
+ logger.info({
138
+ scheduleId: schedule.id,
139
+ processed: slotsToProcess.length,
140
+ remaining: missedSlots.length - MAX_CATCHUP_RUNS_PER_SCAN,
141
+ nextRunAt: firstUnprocessed,
142
+ }, 'Catch-up capped, preserving backlog for next scan');
143
+ }
144
+ else {
145
+ await this.advanceNextRunAt(schedule);
146
+ }
147
+ break;
148
+ }
149
+ }
150
+ }
151
+ computeMissedSlots(schedule, now) {
152
+ if (!schedule.nextRunAt)
153
+ return [];
154
+ const slots = [];
155
+ let cursor = new Date(schedule.nextRunAt);
156
+ const nowDate = new Date(now);
157
+ const horizonCutoff = new Date(nowDate.getTime() - CATCHUP_HORIZON_MS);
158
+ const enumerationLimit = MAX_CATCHUP_RUNS_PER_SCAN * 10;
159
+ while (cursor <= nowDate && slots.length < enumerationLimit) {
160
+ if (cursor >= horizonCutoff) {
161
+ slots.push(cursor.toISOString());
162
+ }
163
+ const next = (0, cron_helpers_1.getNextRunAt)(schedule.cronExpression, schedule.timezone, cursor);
164
+ if (!next || next <= cursor)
165
+ break;
166
+ cursor = next;
167
+ }
168
+ return slots;
169
+ }
170
+ async claimAndExecute(schedule, plannedFor, source) {
171
+ const insertResult = await this.storage.createScheduledEpicRun({
172
+ scheduleId: schedule.id,
173
+ plannedFor,
174
+ source,
175
+ status: 'pending',
176
+ });
177
+ const runId = insertResult.run.id;
178
+ const claimResult = await this.atomicClaim(schedule, runId);
179
+ if (!claimResult.claimed) {
180
+ logger.debug({ scheduleId: schedule.id, plannedFor, runId }, 'Claim lost, deferring to next scan');
181
+ return;
182
+ }
183
+ await this.executeClaimedRun(claimResult.run, schedule);
184
+ }
185
+ async atomicClaim(schedule, runId) {
186
+ let result = await this.storage.claimScheduledEpicRun(runId);
187
+ if (!result.claimed && result.run.status === 'pending') {
188
+ result = await this.storage.claimScheduledEpicRun(runId);
189
+ }
190
+ return result;
191
+ }
192
+ async drainPendingRuns(scheduleId) {
193
+ const runs = await this.storage.listScheduledEpicRuns(scheduleId, {
194
+ status: 'pending',
195
+ limit: 50,
196
+ });
197
+ for (const run of runs.items) {
198
+ let schedule;
199
+ try {
200
+ schedule = await this.storage.getScheduledEpic(run.scheduleId);
201
+ }
202
+ catch {
203
+ await this.storage.updateScheduledEpicRun(run.id, {
204
+ status: 'failed',
205
+ finishedAt: new Date().toISOString(),
206
+ errorMessage: 'Schedule not found',
207
+ });
208
+ continue;
209
+ }
210
+ const claimResult = await this.atomicClaim(schedule, run.id);
211
+ if (!claimResult.claimed)
212
+ continue;
213
+ await this.executeClaimedRun(claimResult.run, schedule);
214
+ }
215
+ }
216
+ async executeRun(run) {
217
+ if (run.status !== 'pending')
218
+ return;
219
+ let schedule;
220
+ try {
221
+ schedule = await this.storage.getScheduledEpic(run.scheduleId);
222
+ }
223
+ catch {
224
+ logger.warn({ runId: run.id, scheduleId: run.scheduleId }, 'Schedule not found for run, marking failed');
225
+ await this.storage.updateScheduledEpicRun(run.id, {
226
+ status: 'failed',
227
+ finishedAt: new Date().toISOString(),
228
+ errorMessage: 'Schedule not found',
229
+ });
230
+ return;
231
+ }
232
+ const claimResult = await this.atomicClaim(schedule, run.id);
233
+ if (!claimResult.claimed)
234
+ return;
235
+ await this.executeClaimedRun(claimResult.run, schedule);
236
+ }
237
+ async executeClaimedRun(run, schedule) {
238
+ const finishedAt = new Date().toISOString();
239
+ if (!schedule.enabled && run.source !== 'manual') {
240
+ await this.storage.updateScheduledEpicRun(run.id, {
241
+ status: 'skipped',
242
+ finishedAt,
243
+ errorMessage: 'Schedule disabled',
244
+ });
245
+ await this.publishRunOutcome(run, schedule, {
246
+ status: 'skipped',
247
+ finishedAt,
248
+ errorCode: 'SCHEDULE_DISABLED',
249
+ errorMessage: 'Schedule disabled',
250
+ createdEpicId: null,
251
+ createdEpicTitle: null,
252
+ });
253
+ return;
254
+ }
255
+ if (!schedule.allowOverlap) {
256
+ const runningRuns = await this.storage.listScheduledEpicRuns(schedule.id, {
257
+ status: 'running',
258
+ limit: 2,
259
+ });
260
+ const otherRunning = runningRuns.items.filter((r) => r.id !== run.id);
261
+ if (otherRunning.length > 0) {
262
+ logger.debug({ scheduleId: schedule.id, runId: run.id }, 'Overlap not allowed and another run is in progress, skipping');
263
+ await this.storage.updateScheduledEpicRun(run.id, {
264
+ status: 'skipped',
265
+ finishedAt,
266
+ errorMessage: 'Overlap not allowed',
267
+ });
268
+ await this.publishRunOutcome(run, schedule, {
269
+ status: 'skipped',
270
+ finishedAt,
271
+ errorCode: 'DUPLICATE_CLAIM',
272
+ errorMessage: 'Overlap not allowed',
273
+ createdEpicId: null,
274
+ createdEpicTitle: null,
275
+ });
276
+ return;
277
+ }
278
+ }
279
+ let errorCode = null;
280
+ try {
281
+ const templateVars = this.buildTemplateVars(schedule, run);
282
+ let title;
283
+ let description;
284
+ try {
285
+ title = (0, template_helpers_1.renderScheduledEpicTemplate)(schedule.titleTemplate, templateVars);
286
+ description = schedule.descriptionTemplate
287
+ ? (0, template_helpers_1.renderScheduledEpicTemplate)(schedule.descriptionTemplate, templateVars)
288
+ : null;
289
+ }
290
+ catch (err) {
291
+ errorCode = 'TEMPLATE_RENDER_FAILED';
292
+ throw err;
293
+ }
294
+ const epicInput = {
295
+ title,
296
+ description,
297
+ tags: schedule.templateTags,
298
+ statusId: schedule.templateStatusId ?? undefined,
299
+ agentId: schedule.templateAgentId ?? undefined,
300
+ parentId: schedule.templateParentEpicId ?? undefined,
301
+ };
302
+ let epic;
303
+ try {
304
+ epic = await this.epicsService.createEpicForProject(schedule.projectId, epicInput);
305
+ }
306
+ catch (err) {
307
+ errorCode = 'EPIC_CREATE_FAILED';
308
+ throw err;
309
+ }
310
+ await this.storage.updateScheduledEpicRun(run.id, {
311
+ status: 'completed',
312
+ createdEpicId: epic.id,
313
+ finishedAt,
314
+ });
315
+ await this.storage.updateScheduledEpicRuntimeState(schedule.id, {
316
+ lastRunAt: finishedAt,
317
+ lastRunStatus: 'completed',
318
+ lastError: null,
319
+ });
320
+ logger.info({ scheduleId: schedule.id, runId: run.id, epicId: epic.id }, 'Scheduled epic run completed');
321
+ await this.publishRunOutcome(run, schedule, {
322
+ status: 'completed',
323
+ finishedAt,
324
+ errorCode: null,
325
+ errorMessage: null,
326
+ createdEpicId: epic.id,
327
+ createdEpicTitle: epic.title,
328
+ });
329
+ }
330
+ catch (error) {
331
+ const errorMessage = error instanceof Error ? error.message : String(error);
332
+ const resolvedErrorCode = errorCode ?? 'UNKNOWN';
333
+ await this.storage.updateScheduledEpicRun(run.id, {
334
+ status: 'failed',
335
+ finishedAt,
336
+ errorMessage,
337
+ });
338
+ await this.storage.updateScheduledEpicRuntimeState(schedule.id, {
339
+ lastRunAt: finishedAt,
340
+ lastRunStatus: 'failed',
341
+ lastError: errorMessage,
342
+ });
343
+ logger.error({ scheduleId: schedule.id, runId: run.id, error: errorMessage }, 'Scheduled epic run failed');
344
+ await this.publishRunOutcome(run, schedule, {
345
+ status: 'failed',
346
+ finishedAt,
347
+ errorCode: resolvedErrorCode,
348
+ errorMessage,
349
+ createdEpicId: null,
350
+ createdEpicTitle: null,
351
+ });
352
+ }
353
+ }
354
+ async publishRunOutcome(run, schedule, outcome) {
355
+ const lagMs = new Date(outcome.finishedAt).getTime() - new Date(run.plannedFor).getTime();
356
+ try {
357
+ await this.eventsService.publish('scheduled_epic.executed', {
358
+ scheduleId: schedule.id,
359
+ runId: run.id,
360
+ projectId: schedule.projectId,
361
+ scheduleName: schedule.name,
362
+ triggerSource: run.source,
363
+ status: outcome.status,
364
+ plannedFor: run.plannedFor,
365
+ finishedAt: outcome.finishedAt,
366
+ lagMs,
367
+ createdEpicId: outcome.createdEpicId,
368
+ createdEpicTitle: outcome.createdEpicTitle,
369
+ errorCode: outcome.errorCode,
370
+ errorMessage: outcome.errorMessage,
371
+ });
372
+ }
373
+ catch (error) {
374
+ logger.warn({ error: String(error) }, 'Failed to publish scheduled_epic.executed event');
375
+ }
376
+ }
377
+ async recoverStaleRuns(projectId, now) {
378
+ const allSchedules = await this.storage.listScheduledEpics(projectId, { limit: 500 });
379
+ const staleCutoff = new Date(new Date(now).getTime() - STALE_RUNNING_THRESHOLD_MS).toISOString();
380
+ for (const schedule of allSchedules.items) {
381
+ const runningRuns = await this.storage.listScheduledEpicRuns(schedule.id, {
382
+ status: 'running',
383
+ limit: 50,
384
+ });
385
+ for (const run of runningRuns.items) {
386
+ if (run.startedAt && run.startedAt < staleCutoff) {
387
+ logger.warn({ runId: run.id, scheduleId: schedule.id, startedAt: run.startedAt }, 'Recovering stale running run');
388
+ await this.storage.updateScheduledEpicRun(run.id, {
389
+ status: 'failed',
390
+ finishedAt: now,
391
+ errorMessage: 'STALE_RUNNING_RECOVERED',
392
+ });
393
+ await this.storage.updateScheduledEpicRuntimeState(schedule.id, {
394
+ lastRunStatus: 'failed',
395
+ lastError: 'STALE_RUNNING_RECOVERED',
396
+ });
397
+ await this.publishRunOutcome(run, schedule, {
398
+ status: 'failed',
399
+ finishedAt: now,
400
+ errorCode: 'STALE_RUNNING_RECOVERED',
401
+ errorMessage: 'STALE_RUNNING_RECOVERED',
402
+ createdEpicId: null,
403
+ createdEpicTitle: null,
404
+ });
405
+ }
406
+ }
407
+ }
408
+ }
409
+ async advanceNextRunAt(schedule) {
410
+ const nextRunAt = (0, cron_helpers_1.getNextRunAt)(schedule.cronExpression, schedule.timezone);
411
+ await this.storage.updateScheduledEpicRuntimeState(schedule.id, {
412
+ nextRunAt: nextRunAt?.toISOString() ?? null,
413
+ });
414
+ }
415
+ buildTemplateVars(schedule, run) {
416
+ const plannedDate = new Date(run.plannedFor);
417
+ return {
418
+ schedule_name: schedule.name,
419
+ date: plannedDate.toISOString().split('T')[0],
420
+ datetime: run.plannedFor,
421
+ timestamp: plannedDate.getTime(),
422
+ run_source: run.source,
423
+ project_id: schedule.projectId,
424
+ };
425
+ }
426
+ };
427
+ exports.ScheduledEpicRunnerService = ScheduledEpicRunnerService;
428
+ exports.ScheduledEpicRunnerService = ScheduledEpicRunnerService = __decorate([
429
+ (0, common_1.Injectable)(),
430
+ __param(0, (0, common_1.Inject)(storage_interface_1.STORAGE_SERVICE)),
431
+ __metadata("design:paramtypes", [Object, epics_service_1.EpicsService,
432
+ events_service_1.EventsService])
433
+ ], ScheduledEpicRunnerService);
434
+ //# sourceMappingURL=scheduled-epic-runner.service.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scheduled-epic-runner.service.js","sourceRoot":"","sources":["../../../../src/modules/scheduled-epics/services/scheduled-epic-runner.service.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,23 @@
1
+ import { type ScheduledEpicStorage, type ListScheduledEpicsOptions, type ListScheduledEpicRunsOptions, type ListResult } from '../../storage/interfaces/storage.interface';
2
+ import type { ScheduledEpic, ScheduledEpicRun } from '../../storage/models/domain.models';
3
+ import type { ClaimRunResult } from '../../storage/interfaces/storage.interface';
4
+ import { type CreateScheduledEpicDto, type UpdateScheduledEpicDto } from '../dtos/scheduled-epic.dto';
5
+ export declare const SCHEDULED_EPIC_RUNNER_REFRESH = "SCHEDULED_EPIC_RUNNER_REFRESH";
6
+ export interface ScheduledEpicRunnerRefresh {
7
+ refreshScheduleWindow(): void;
8
+ }
9
+ export declare class ScheduledEpicsService {
10
+ private readonly storage;
11
+ private readonly runnerRefresh?;
12
+ constructor(storage: ScheduledEpicStorage, runnerRefresh?: ScheduledEpicRunnerRefresh | undefined);
13
+ create(dto: CreateScheduledEpicDto): Promise<ScheduledEpic>;
14
+ get(id: string): Promise<ScheduledEpic>;
15
+ list(projectId: string, options?: ListScheduledEpicsOptions): Promise<ListResult<ScheduledEpic>>;
16
+ update(id: string, dto: UpdateScheduledEpicDto, configVersion: number): Promise<ScheduledEpic>;
17
+ delete(id: string): Promise<void>;
18
+ toggle(id: string, enabled: boolean, configVersion: number): Promise<ScheduledEpic>;
19
+ runNow(id: string): Promise<ClaimRunResult>;
20
+ listRuns(scheduleId: string, options?: ListScheduledEpicRunsOptions): Promise<ListResult<ScheduledEpicRun>>;
21
+ private computeNextRunAt;
22
+ private notifyRunner;
23
+ }
@@ -0,0 +1,136 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ var __metadata = (this && this.__metadata) || function (k, v) {
9
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
10
+ };
11
+ var __param = (this && this.__param) || function (paramIndex, decorator) {
12
+ return function (target, key) { decorator(target, key, paramIndex); }
13
+ };
14
+ Object.defineProperty(exports, "__esModule", { value: true });
15
+ exports.ScheduledEpicsService = exports.SCHEDULED_EPIC_RUNNER_REFRESH = void 0;
16
+ const common_1 = require("@nestjs/common");
17
+ const storage_interface_1 = require("../../storage/interfaces/storage.interface");
18
+ const error_types_1 = require("../../../common/errors/error-types");
19
+ const logger_1 = require("../../../common/logging/logger");
20
+ const cron_helpers_1 = require("../helpers/cron-helpers");
21
+ const scheduled_epic_dto_1 = require("../dtos/scheduled-epic.dto");
22
+ exports.SCHEDULED_EPIC_RUNNER_REFRESH = 'SCHEDULED_EPIC_RUNNER_REFRESH';
23
+ const logger = (0, logger_1.createLogger)('ScheduledEpicsService');
24
+ let ScheduledEpicsService = class ScheduledEpicsService {
25
+ constructor(storage, runnerRefresh) {
26
+ this.storage = storage;
27
+ this.runnerRefresh = runnerRefresh;
28
+ }
29
+ async create(dto) {
30
+ const parsed = scheduled_epic_dto_1.CreateScheduledEpicDtoSchema.parse(dto);
31
+ const nextRunAt = this.computeNextRunAt(parsed.cronExpression, parsed.timezone);
32
+ const schedule = await this.storage.createScheduledEpic({
33
+ projectId: parsed.projectId,
34
+ name: parsed.name,
35
+ cronExpression: parsed.cronExpression,
36
+ timezone: parsed.timezone,
37
+ enabled: parsed.enabled,
38
+ titleTemplate: parsed.titleTemplate,
39
+ descriptionTemplate: parsed.descriptionTemplate ?? null,
40
+ templateStatusId: parsed.templateStatusId ?? null,
41
+ templateParentEpicId: parsed.templateParentEpicId ?? null,
42
+ templateAgentId: parsed.templateAgentId ?? null,
43
+ templateTags: parsed.templateTags,
44
+ allowOverlap: parsed.allowOverlap,
45
+ missedRunPolicy: parsed.missedRunPolicy,
46
+ nextRunAt: nextRunAt?.toISOString() ?? null,
47
+ });
48
+ logger.info({ scheduleId: schedule.id, name: schedule.name }, 'Scheduled epic created');
49
+ this.notifyRunner();
50
+ return schedule;
51
+ }
52
+ async get(id) {
53
+ return this.storage.getScheduledEpic(id);
54
+ }
55
+ async list(projectId, options) {
56
+ return this.storage.listScheduledEpics(projectId, options);
57
+ }
58
+ async update(id, dto, configVersion) {
59
+ const parsed = scheduled_epic_dto_1.UpdateScheduledEpicDtoSchema.parse(dto);
60
+ if (Object.keys(parsed).length === 0) {
61
+ throw new error_types_1.ValidationError('At least one field must be provided for update');
62
+ }
63
+ const current = await this.storage.getScheduledEpic(id);
64
+ const configUpdate = { ...parsed };
65
+ let schedule = await this.storage.updateScheduledEpic(id, configUpdate, configVersion);
66
+ const cronOrTzChanged = parsed.cronExpression !== undefined || parsed.timezone !== undefined;
67
+ if (cronOrTzChanged) {
68
+ const cronExpression = parsed.cronExpression ?? current.cronExpression;
69
+ const timezone = parsed.timezone ?? current.timezone;
70
+ const nextRunAt = this.computeNextRunAt(cronExpression, timezone);
71
+ schedule = await this.storage.updateScheduledEpicRuntimeState(id, {
72
+ nextRunAt: nextRunAt?.toISOString() ?? null,
73
+ });
74
+ }
75
+ logger.info({ scheduleId: id }, 'Scheduled epic updated');
76
+ this.notifyRunner();
77
+ return schedule;
78
+ }
79
+ async delete(id) {
80
+ await this.storage.getScheduledEpic(id);
81
+ await this.storage.deleteScheduledEpic(id);
82
+ logger.info({ scheduleId: id }, 'Scheduled epic deleted');
83
+ this.notifyRunner();
84
+ }
85
+ async toggle(id, enabled, configVersion) {
86
+ let schedule = await this.storage.updateScheduledEpic(id, { enabled }, configVersion);
87
+ if (enabled && !schedule.nextRunAt) {
88
+ const nextRunAt = this.computeNextRunAt(schedule.cronExpression, schedule.timezone);
89
+ schedule = await this.storage.updateScheduledEpicRuntimeState(id, {
90
+ nextRunAt: nextRunAt?.toISOString() ?? null,
91
+ });
92
+ }
93
+ logger.info({ scheduleId: id, enabled }, 'Scheduled epic toggled');
94
+ this.notifyRunner();
95
+ return schedule;
96
+ }
97
+ async runNow(id) {
98
+ await this.storage.getScheduledEpic(id);
99
+ const plannedFor = new Date().toISOString();
100
+ const result = await this.storage.createScheduledEpicRun({
101
+ scheduleId: id,
102
+ plannedFor,
103
+ source: 'manual',
104
+ status: 'pending',
105
+ });
106
+ logger.info({ scheduleId: id, runId: result.run.id, claimed: result.claimed }, 'Manual run requested');
107
+ this.notifyRunner();
108
+ return result;
109
+ }
110
+ async listRuns(scheduleId, options) {
111
+ await this.storage.getScheduledEpic(scheduleId);
112
+ return this.storage.listScheduledEpicRuns(scheduleId, options);
113
+ }
114
+ computeNextRunAt(cronExpression, timezone) {
115
+ return (0, cron_helpers_1.getNextRunAt)(cronExpression, timezone);
116
+ }
117
+ notifyRunner() {
118
+ if (this.runnerRefresh) {
119
+ try {
120
+ this.runnerRefresh.refreshScheduleWindow();
121
+ }
122
+ catch (error) {
123
+ logger.warn({ error: String(error) }, 'Failed to notify runner of schedule change');
124
+ }
125
+ }
126
+ }
127
+ };
128
+ exports.ScheduledEpicsService = ScheduledEpicsService;
129
+ exports.ScheduledEpicsService = ScheduledEpicsService = __decorate([
130
+ (0, common_1.Injectable)(),
131
+ __param(0, (0, common_1.Inject)(storage_interface_1.STORAGE_SERVICE)),
132
+ __param(1, (0, common_1.Optional)()),
133
+ __param(1, (0, common_1.Inject)(exports.SCHEDULED_EPIC_RUNNER_REFRESH)),
134
+ __metadata("design:paramtypes", [Object, Object])
135
+ ], ScheduledEpicsService);
136
+ //# sourceMappingURL=scheduled-epics.service.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scheduled-epics.service.js","sourceRoot":"","sources":["../../../../src/modules/scheduled-epics/services/scheduled-epics.service.ts"],"names":[],"mappings":"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"}