mirror of
https://github.com/khoaliber/dockhand.git
synced 2026-03-06 05:39:05 +00:00
Initial commit
This commit is contained in:
632
lib/server/scheduler/index.ts
Normal file
632
lib/server/scheduler/index.ts
Normal file
@@ -0,0 +1,632 @@
|
||||
/**
|
||||
* Unified Scheduler Service
|
||||
*
|
||||
* Manages all scheduled tasks using croner with automatic job lifecycle:
|
||||
* - System cleanup jobs (static cron schedules)
|
||||
* - Container auto-updates (dynamic schedules from database)
|
||||
* - Git stack auto-sync (dynamic schedules from database)
|
||||
*
|
||||
* All execution logic is in separate task files for clean architecture.
|
||||
*/
|
||||
|
||||
import { Cron } from 'croner';
|
||||
import {
|
||||
getEnabledAutoUpdateSettings,
|
||||
getEnabledAutoUpdateGitStacks,
|
||||
getAutoUpdateSettingById,
|
||||
getGitStack,
|
||||
getScheduleCleanupCron,
|
||||
getEventCleanupCron,
|
||||
getScheduleRetentionDays,
|
||||
getEventRetentionDays,
|
||||
getScheduleCleanupEnabled,
|
||||
getEventCleanupEnabled,
|
||||
getEnvironments,
|
||||
getEnvUpdateCheckSettings,
|
||||
getAllEnvUpdateCheckSettings,
|
||||
getEnvironment,
|
||||
getEnvironmentTimezone,
|
||||
getDefaultTimezone
|
||||
} from '../db';
|
||||
import {
|
||||
cleanupStaleVolumeHelpers,
|
||||
cleanupExpiredVolumeHelpers
|
||||
} from '../docker';
|
||||
|
||||
// Import task execution functions
|
||||
import { runContainerUpdate } from './tasks/container-update';
|
||||
import { runGitStackSync } from './tasks/git-stack-sync';
|
||||
import { runEnvUpdateCheckJob } from './tasks/env-update-check';
|
||||
import {
|
||||
runScheduleCleanupJob,
|
||||
runEventCleanupJob,
|
||||
runVolumeHelperCleanupJob,
|
||||
SYSTEM_SCHEDULE_CLEANUP_ID,
|
||||
SYSTEM_EVENT_CLEANUP_ID,
|
||||
SYSTEM_VOLUME_HELPER_CLEANUP_ID
|
||||
} from './tasks/system-cleanup';
|
||||
|
||||
// Store all active cron jobs
|
||||
const activeJobs: Map<string, Cron> = new Map();
|
||||
|
||||
// System cleanup jobs
|
||||
let cleanupJob: Cron | null = null;
|
||||
let eventCleanupJob: Cron | null = null;
|
||||
let volumeHelperCleanupJob: Cron | null = null;
|
||||
|
||||
// Scheduler state
|
||||
let isRunning = false;
|
||||
|
||||
/**
|
||||
* Start the unified scheduler service.
|
||||
* Registers all schedules with croner for automatic execution.
|
||||
*/
|
||||
export async function startScheduler(): Promise<void> {
|
||||
if (isRunning) {
|
||||
console.log('[Scheduler] Already running');
|
||||
return;
|
||||
}
|
||||
|
||||
console.log('[Scheduler] Starting scheduler service...');
|
||||
isRunning = true;
|
||||
|
||||
// Get cron expressions and default timezone from database
|
||||
const scheduleCleanupCron = await getScheduleCleanupCron();
|
||||
const eventCleanupCron = await getEventCleanupCron();
|
||||
const defaultTimezone = await getDefaultTimezone();
|
||||
|
||||
// Start system cleanup jobs (static schedules with default timezone)
|
||||
cleanupJob = new Cron(scheduleCleanupCron, { timezone: defaultTimezone }, async () => {
|
||||
await runScheduleCleanupJob();
|
||||
});
|
||||
|
||||
eventCleanupJob = new Cron(eventCleanupCron, { timezone: defaultTimezone }, async () => {
|
||||
await runEventCleanupJob();
|
||||
});
|
||||
|
||||
// Cleanup functions to pass to the job (avoids dynamic import issues in production)
|
||||
// Wrap cleanupStaleVolumeHelpers to pre-fetch environments
|
||||
const wrappedCleanupStale = async () => {
|
||||
const envs = await getEnvironments();
|
||||
await cleanupStaleVolumeHelpers(envs);
|
||||
};
|
||||
const volumeCleanupFns = {
|
||||
cleanupStaleVolumeHelpers: wrappedCleanupStale,
|
||||
cleanupExpiredVolumeHelpers
|
||||
};
|
||||
|
||||
// Volume helper cleanup runs every 30 minutes to clean up expired browse containers
|
||||
volumeHelperCleanupJob = new Cron('*/30 * * * *', { timezone: defaultTimezone }, async () => {
|
||||
await runVolumeHelperCleanupJob('cron', volumeCleanupFns);
|
||||
});
|
||||
|
||||
// Run volume helper cleanup immediately on startup to clean up stale containers
|
||||
runVolumeHelperCleanupJob('startup', volumeCleanupFns).catch(err => {
|
||||
console.error('[Scheduler] Error during startup volume helper cleanup:', err);
|
||||
});
|
||||
|
||||
console.log(`[Scheduler] System schedule cleanup: ${scheduleCleanupCron} [${defaultTimezone}]`);
|
||||
console.log(`[Scheduler] System event cleanup: ${eventCleanupCron} [${defaultTimezone}]`);
|
||||
console.log(`[Scheduler] Volume helper cleanup: every 30 minutes [${defaultTimezone}]`);
|
||||
|
||||
// Register all dynamic schedules from database
|
||||
await refreshAllSchedules();
|
||||
|
||||
console.log('[Scheduler] Service started');
|
||||
}
|
||||
|
||||
/**
|
||||
* Stop the scheduler service and cleanup all jobs.
|
||||
*/
|
||||
export function stopScheduler(): void {
|
||||
if (!isRunning) return;
|
||||
|
||||
console.log('[Scheduler] Stopping scheduler...');
|
||||
isRunning = false;
|
||||
|
||||
// Stop system jobs
|
||||
if (cleanupJob) {
|
||||
cleanupJob.stop();
|
||||
cleanupJob = null;
|
||||
}
|
||||
if (eventCleanupJob) {
|
||||
eventCleanupJob.stop();
|
||||
eventCleanupJob = null;
|
||||
}
|
||||
if (volumeHelperCleanupJob) {
|
||||
volumeHelperCleanupJob.stop();
|
||||
volumeHelperCleanupJob = null;
|
||||
}
|
||||
|
||||
// Stop all dynamic jobs
|
||||
for (const [key, job] of activeJobs.entries()) {
|
||||
job.stop();
|
||||
}
|
||||
activeJobs.clear();
|
||||
|
||||
console.log('[Scheduler] Service stopped');
|
||||
}
|
||||
|
||||
/**
|
||||
* Refresh all dynamic schedules from database.
|
||||
* Called on startup and optionally for recovery.
|
||||
*/
|
||||
export async function refreshAllSchedules(): Promise<void> {
|
||||
console.log('[Scheduler] Refreshing all schedules...');
|
||||
|
||||
// Clear existing dynamic jobs
|
||||
for (const [key, job] of activeJobs.entries()) {
|
||||
job.stop();
|
||||
}
|
||||
activeJobs.clear();
|
||||
|
||||
let containerCount = 0;
|
||||
let gitStackCount = 0;
|
||||
|
||||
// Register container auto-update schedules
|
||||
try {
|
||||
const containerSettings = await getEnabledAutoUpdateSettings();
|
||||
for (const setting of containerSettings) {
|
||||
if (setting.cronExpression) {
|
||||
const registered = await registerSchedule(
|
||||
setting.id,
|
||||
'container_update',
|
||||
setting.environmentId
|
||||
);
|
||||
if (registered) containerCount++;
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[Scheduler] Error loading container schedules:', error);
|
||||
}
|
||||
|
||||
// Register git stack auto-sync schedules
|
||||
try {
|
||||
const gitStacks = await getEnabledAutoUpdateGitStacks();
|
||||
for (const stack of gitStacks) {
|
||||
if (stack.autoUpdateCron) {
|
||||
const registered = await registerSchedule(
|
||||
stack.id,
|
||||
'git_stack_sync',
|
||||
stack.environmentId
|
||||
);
|
||||
if (registered) gitStackCount++;
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[Scheduler] Error loading git stack schedules:', error);
|
||||
}
|
||||
|
||||
// Register environment update check schedules
|
||||
let envUpdateCheckCount = 0;
|
||||
try {
|
||||
const envConfigs = await getAllEnvUpdateCheckSettings();
|
||||
for (const { envId, settings } of envConfigs) {
|
||||
if (settings.enabled && settings.cron) {
|
||||
const registered = await registerSchedule(
|
||||
envId,
|
||||
'env_update_check',
|
||||
envId
|
||||
);
|
||||
if (registered) envUpdateCheckCount++;
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[Scheduler] Error loading env update check schedules:', error);
|
||||
}
|
||||
|
||||
console.log(`[Scheduler] Registered ${containerCount} container schedules, ${gitStackCount} git stack schedules, ${envUpdateCheckCount} env update check schedules`);
|
||||
}
|
||||
|
||||
/**
|
||||
* Register or update a schedule with automatic croner execution.
|
||||
* Idempotent - can be called multiple times safely.
|
||||
*/
|
||||
export async function registerSchedule(
|
||||
scheduleId: number,
|
||||
type: 'container_update' | 'git_stack_sync' | 'env_update_check',
|
||||
environmentId: number | null
|
||||
): Promise<boolean> {
|
||||
const key = `${type}-${scheduleId}`;
|
||||
|
||||
try {
|
||||
// Unregister existing job if present
|
||||
unregisterSchedule(scheduleId, type);
|
||||
|
||||
// Fetch schedule data from database
|
||||
let cronExpression: string | null = null;
|
||||
let entityName: string | null = null;
|
||||
let enabled = false;
|
||||
|
||||
if (type === 'container_update') {
|
||||
const setting = await getAutoUpdateSettingById(scheduleId);
|
||||
if (!setting) return false;
|
||||
cronExpression = setting.cronExpression;
|
||||
entityName = setting.containerName;
|
||||
enabled = setting.enabled;
|
||||
} else if (type === 'git_stack_sync') {
|
||||
const stack = await getGitStack(scheduleId);
|
||||
if (!stack) return false;
|
||||
cronExpression = stack.autoUpdateCron;
|
||||
entityName = stack.stackName;
|
||||
enabled = stack.autoUpdate;
|
||||
} else if (type === 'env_update_check') {
|
||||
const config = await getEnvUpdateCheckSettings(scheduleId);
|
||||
if (!config) return false;
|
||||
const env = await getEnvironment(scheduleId);
|
||||
if (!env) return false;
|
||||
cronExpression = config.cron;
|
||||
entityName = `Update: ${env.name}`;
|
||||
enabled = config.enabled;
|
||||
}
|
||||
|
||||
// Don't create job if disabled or no cron expression
|
||||
if (!enabled || !cronExpression) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Get timezone for this environment
|
||||
const timezone = environmentId ? await getEnvironmentTimezone(environmentId) : 'UTC';
|
||||
|
||||
// Create new Cron instance with timezone
|
||||
const job = new Cron(cronExpression, { timezone }, async () => {
|
||||
// Defensive check: verify schedule still exists and is enabled
|
||||
if (type === 'container_update') {
|
||||
const setting = await getAutoUpdateSettingById(scheduleId);
|
||||
if (!setting || !setting.enabled) return;
|
||||
await runContainerUpdate(scheduleId, setting.containerName, environmentId, 'cron');
|
||||
} else if (type === 'git_stack_sync') {
|
||||
const stack = await getGitStack(scheduleId);
|
||||
if (!stack || !stack.autoUpdate) return;
|
||||
await runGitStackSync(scheduleId, stack.stackName, environmentId, 'cron');
|
||||
} else if (type === 'env_update_check') {
|
||||
const config = await getEnvUpdateCheckSettings(scheduleId);
|
||||
if (!config || !config.enabled) return;
|
||||
await runEnvUpdateCheckJob(scheduleId, 'cron');
|
||||
}
|
||||
});
|
||||
|
||||
// Store in active jobs map
|
||||
activeJobs.set(key, job);
|
||||
console.log(`[Scheduler] Registered ${type} schedule ${scheduleId} (${entityName}): ${cronExpression} [${timezone}]`);
|
||||
return true;
|
||||
} catch (error: any) {
|
||||
console.error(`[Scheduler] Failed to register ${type} schedule ${scheduleId}:`, error.message);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Unregister a schedule and stop its croner job.
|
||||
* Idempotent - safe to call even if not registered.
|
||||
*/
|
||||
export function unregisterSchedule(
|
||||
scheduleId: number,
|
||||
type: 'container_update' | 'git_stack_sync' | 'env_update_check'
|
||||
): void {
|
||||
const key = `${type}-${scheduleId}`;
|
||||
const job = activeJobs.get(key);
|
||||
|
||||
if (job) {
|
||||
job.stop();
|
||||
activeJobs.delete(key);
|
||||
console.log(`[Scheduler] Unregistered ${type} schedule ${scheduleId}`);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Refresh all schedules for a specific environment.
|
||||
* Called when an environment's timezone changes to re-register jobs with the new timezone.
|
||||
*/
|
||||
export async function refreshSchedulesForEnvironment(environmentId: number): Promise<void> {
|
||||
console.log(`[Scheduler] Refreshing schedules for environment ${environmentId} (timezone changed)`);
|
||||
|
||||
let refreshedCount = 0;
|
||||
|
||||
// Re-register container auto-update schedules for this environment
|
||||
try {
|
||||
const containerSettings = await getEnabledAutoUpdateSettings();
|
||||
for (const setting of containerSettings) {
|
||||
if (setting.environmentId === environmentId && setting.cronExpression) {
|
||||
const registered = await registerSchedule(
|
||||
setting.id,
|
||||
'container_update',
|
||||
setting.environmentId
|
||||
);
|
||||
if (registered) refreshedCount++;
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[Scheduler] Error refreshing container schedules:', error);
|
||||
}
|
||||
|
||||
// Re-register git stack auto-sync schedules for this environment
|
||||
try {
|
||||
const gitStacks = await getEnabledAutoUpdateGitStacks();
|
||||
for (const stack of gitStacks) {
|
||||
if (stack.environmentId === environmentId && stack.autoUpdateCron) {
|
||||
const registered = await registerSchedule(
|
||||
stack.id,
|
||||
'git_stack_sync',
|
||||
stack.environmentId
|
||||
);
|
||||
if (registered) refreshedCount++;
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[Scheduler] Error refreshing git stack schedules:', error);
|
||||
}
|
||||
|
||||
// Re-register environment update check schedule for this environment
|
||||
try {
|
||||
const config = await getEnvUpdateCheckSettings(environmentId);
|
||||
if (config && config.enabled && config.cron) {
|
||||
const registered = await registerSchedule(
|
||||
environmentId,
|
||||
'env_update_check',
|
||||
environmentId
|
||||
);
|
||||
if (registered) refreshedCount++;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[Scheduler] Error refreshing env update check schedule:', error);
|
||||
}
|
||||
|
||||
console.log(`[Scheduler] Refreshed ${refreshedCount} schedules for environment ${environmentId}`);
|
||||
}
|
||||
|
||||
/**
|
||||
* Refresh system cleanup jobs with the new default timezone.
|
||||
* Called when the default timezone setting changes.
|
||||
*/
|
||||
export async function refreshSystemJobs(): Promise<void> {
|
||||
console.log('[Scheduler] Refreshing system jobs (default timezone changed)');
|
||||
|
||||
// Get current settings
|
||||
const scheduleCleanupCron = await getScheduleCleanupCron();
|
||||
const eventCleanupCron = await getEventCleanupCron();
|
||||
const defaultTimezone = await getDefaultTimezone();
|
||||
|
||||
// Cleanup functions to pass to the job
|
||||
const wrappedCleanupStale = async () => {
|
||||
const envs = await getEnvironments();
|
||||
await cleanupStaleVolumeHelpers(envs);
|
||||
};
|
||||
const volumeCleanupFns = {
|
||||
cleanupStaleVolumeHelpers: wrappedCleanupStale,
|
||||
cleanupExpiredVolumeHelpers
|
||||
};
|
||||
|
||||
// Stop existing system jobs
|
||||
if (cleanupJob) {
|
||||
cleanupJob.stop();
|
||||
}
|
||||
if (eventCleanupJob) {
|
||||
eventCleanupJob.stop();
|
||||
}
|
||||
if (volumeHelperCleanupJob) {
|
||||
volumeHelperCleanupJob.stop();
|
||||
}
|
||||
|
||||
// Re-create with new timezone
|
||||
cleanupJob = new Cron(scheduleCleanupCron, { timezone: defaultTimezone }, async () => {
|
||||
await runScheduleCleanupJob();
|
||||
});
|
||||
|
||||
eventCleanupJob = new Cron(eventCleanupCron, { timezone: defaultTimezone }, async () => {
|
||||
await runEventCleanupJob();
|
||||
});
|
||||
|
||||
volumeHelperCleanupJob = new Cron('*/30 * * * *', { timezone: defaultTimezone }, async () => {
|
||||
await runVolumeHelperCleanupJob('cron', volumeCleanupFns);
|
||||
});
|
||||
|
||||
console.log(`[Scheduler] System schedule cleanup: ${scheduleCleanupCron} [${defaultTimezone}]`);
|
||||
console.log(`[Scheduler] System event cleanup: ${eventCleanupCron} [${defaultTimezone}]`);
|
||||
console.log(`[Scheduler] Volume helper cleanup: every 30 minutes [${defaultTimezone}]`);
|
||||
}
|
||||
|
||||
// =============================================================================
|
||||
// MANUAL TRIGGER FUNCTIONS (for API endpoints)
|
||||
// =============================================================================
|
||||
|
||||
/**
|
||||
* Manually trigger a container update.
|
||||
*/
|
||||
export async function triggerContainerUpdate(settingId: number): Promise<{ success: boolean; executionId?: number; error?: string }> {
|
||||
try {
|
||||
const setting = await getAutoUpdateSettingById(settingId);
|
||||
if (!setting) {
|
||||
return { success: false, error: 'Auto-update setting not found' };
|
||||
}
|
||||
|
||||
// Run in background
|
||||
runContainerUpdate(settingId, setting.containerName, setting.environmentId, 'manual');
|
||||
|
||||
return { success: true };
|
||||
} catch (error: any) {
|
||||
return { success: false, error: error.message };
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Manually trigger a git stack sync.
|
||||
*/
|
||||
export async function triggerGitStackSync(stackId: number): Promise<{ success: boolean; executionId?: number; error?: string }> {
|
||||
try {
|
||||
const stack = await getGitStack(stackId);
|
||||
if (!stack) {
|
||||
return { success: false, error: 'Git stack not found' };
|
||||
}
|
||||
|
||||
// Run in background
|
||||
runGitStackSync(stackId, stack.stackName, stack.environmentId, 'manual');
|
||||
|
||||
return { success: true };
|
||||
} catch (error: any) {
|
||||
return { success: false, error: error.message };
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Trigger git stack sync from webhook (called from webhook endpoint).
|
||||
*/
|
||||
export async function triggerGitStackSyncFromWebhook(stackId: number): Promise<{ success: boolean; executionId?: number; error?: string }> {
|
||||
try {
|
||||
const stack = await getGitStack(stackId);
|
||||
if (!stack) {
|
||||
return { success: false, error: 'Git stack not found' };
|
||||
}
|
||||
|
||||
// Run in background
|
||||
runGitStackSync(stackId, stack.stackName, stack.environmentId, 'webhook');
|
||||
|
||||
return { success: true };
|
||||
} catch (error: any) {
|
||||
return { success: false, error: error.message };
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Manually trigger an environment update check.
|
||||
*/
|
||||
export async function triggerEnvUpdateCheck(environmentId: number): Promise<{ success: boolean; executionId?: number; error?: string }> {
|
||||
try {
|
||||
const config = await getEnvUpdateCheckSettings(environmentId);
|
||||
if (!config) {
|
||||
return { success: false, error: 'Update check settings not found for this environment' };
|
||||
}
|
||||
|
||||
const env = await getEnvironment(environmentId);
|
||||
if (!env) {
|
||||
return { success: false, error: 'Environment not found' };
|
||||
}
|
||||
|
||||
// Run in background
|
||||
runEnvUpdateCheckJob(environmentId, 'manual');
|
||||
|
||||
return { success: true };
|
||||
} catch (error: any) {
|
||||
return { success: false, error: error.message };
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Manually trigger a system job (schedule cleanup, event cleanup, etc.).
|
||||
*/
|
||||
export async function triggerSystemJob(jobId: string): Promise<{ success: boolean; executionId?: number; error?: string }> {
|
||||
try {
|
||||
if (jobId === String(SYSTEM_SCHEDULE_CLEANUP_ID) || jobId === 'schedule-cleanup') {
|
||||
runScheduleCleanupJob('manual');
|
||||
return { success: true };
|
||||
} else if (jobId === String(SYSTEM_EVENT_CLEANUP_ID) || jobId === 'event-cleanup') {
|
||||
runEventCleanupJob('manual');
|
||||
return { success: true };
|
||||
} else if (jobId === String(SYSTEM_VOLUME_HELPER_CLEANUP_ID) || jobId === 'volume-helper-cleanup') {
|
||||
// Wrap to pre-fetch environments (avoids dynamic import in production)
|
||||
const wrappedCleanupStale = async () => {
|
||||
const envs = await getEnvironments();
|
||||
await cleanupStaleVolumeHelpers(envs);
|
||||
};
|
||||
runVolumeHelperCleanupJob('manual', {
|
||||
cleanupStaleVolumeHelpers: wrappedCleanupStale,
|
||||
cleanupExpiredVolumeHelpers
|
||||
});
|
||||
return { success: true };
|
||||
} else {
|
||||
return { success: false, error: 'Unknown system job ID' };
|
||||
}
|
||||
} catch (error: any) {
|
||||
return { success: false, error: error.message };
|
||||
}
|
||||
}
|
||||
|
||||
// =============================================================================
|
||||
// UTILITY FUNCTIONS
|
||||
// =============================================================================
|
||||
|
||||
/**
|
||||
* Get the next run time for a cron expression.
|
||||
* @param cronExpression - The cron expression
|
||||
* @param timezone - Optional IANA timezone (e.g., 'Europe/Warsaw'). Defaults to local timezone.
|
||||
*/
|
||||
export function getNextRun(cronExpression: string, timezone?: string): Date | null {
|
||||
try {
|
||||
const options = timezone ? { timezone } : undefined;
|
||||
const job = new Cron(cronExpression, options);
|
||||
const next = job.nextRun();
|
||||
job.stop();
|
||||
return next;
|
||||
} catch {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if a cron expression is valid.
|
||||
*/
|
||||
export function isValidCron(cronExpression: string): boolean {
|
||||
try {
|
||||
const job = new Cron(cronExpression);
|
||||
job.stop();
|
||||
return true;
|
||||
} catch {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get system schedules info for the API.
|
||||
*/
|
||||
export async function getSystemSchedules(): Promise<SystemScheduleInfo[]> {
|
||||
const scheduleRetention = await getScheduleRetentionDays();
|
||||
const eventRetention = await getEventRetentionDays();
|
||||
const scheduleCleanupCron = await getScheduleCleanupCron();
|
||||
const eventCleanupCron = await getEventCleanupCron();
|
||||
const scheduleCleanupEnabled = await getScheduleCleanupEnabled();
|
||||
const eventCleanupEnabled = await getEventCleanupEnabled();
|
||||
|
||||
return [
|
||||
{
|
||||
id: SYSTEM_SCHEDULE_CLEANUP_ID,
|
||||
type: 'system_cleanup' as const,
|
||||
name: 'Schedule execution cleanup',
|
||||
description: `Removes execution logs older than ${scheduleRetention} days`,
|
||||
cronExpression: scheduleCleanupCron,
|
||||
nextRun: scheduleCleanupEnabled ? getNextRun(scheduleCleanupCron)?.toISOString() ?? null : null,
|
||||
isSystem: true,
|
||||
enabled: scheduleCleanupEnabled
|
||||
},
|
||||
{
|
||||
id: SYSTEM_EVENT_CLEANUP_ID,
|
||||
type: 'system_cleanup' as const,
|
||||
name: 'Container event cleanup',
|
||||
description: `Removes container events older than ${eventRetention} days`,
|
||||
cronExpression: eventCleanupCron,
|
||||
nextRun: eventCleanupEnabled ? getNextRun(eventCleanupCron)?.toISOString() ?? null : null,
|
||||
isSystem: true,
|
||||
enabled: eventCleanupEnabled
|
||||
},
|
||||
{
|
||||
id: SYSTEM_VOLUME_HELPER_CLEANUP_ID,
|
||||
type: 'system_cleanup' as const,
|
||||
name: 'Volume helper cleanup',
|
||||
description: 'Cleans up temporary volume browser containers',
|
||||
cronExpression: '*/30 * * * *',
|
||||
nextRun: getNextRun('*/30 * * * *')?.toISOString() ?? null,
|
||||
isSystem: true,
|
||||
enabled: true
|
||||
}
|
||||
];
|
||||
}
|
||||
|
||||
export interface SystemScheduleInfo {
|
||||
id: number;
|
||||
type: 'system_cleanup';
|
||||
name: string;
|
||||
description: string;
|
||||
cronExpression: string;
|
||||
nextRun: string | null;
|
||||
isSystem: true;
|
||||
enabled: boolean;
|
||||
}
|
||||
Reference in New Issue
Block a user