Files
gitops/ts/opsserver/handlers/sync.handler.ts

223 lines
8.1 KiB
TypeScript

import * as plugins from '../../plugins.ts';
import type { OpsServer } from '../classes.opsserver.ts';
import * as interfaces from '../../../ts_interfaces/index.ts';
import { requireValidIdentity } from '../helpers/guards.ts';
import { logger } from '../../logging.ts';
export class SyncHandler {
public typedrouter = new plugins.typedrequest.TypedRouter();
constructor(private opsServerRef: OpsServer) {
this.opsServerRef.typedrouter.addTypedRouter(this.typedrouter);
this.registerHandlers();
this.setupBroadcast();
}
/**
* Wire up the logger's broadcast function to push sync log entries
* to all connected frontends via TypedSocket.
*/
private setupBroadcast(): void {
logger.setBroadcastFn((entry) => {
try {
const typedsocket = this.opsServerRef.server?.typedserver?.typedsocket;
if (!typedsocket) return;
typedsocket.findAllTargetConnectionsByTag('allClients').then((connections) => {
for (const conn of connections) {
typedsocket
.createTypedRequest<interfaces.requests.IReq_PushSyncLog>('pushSyncLog', conn)
.fire({ entry })
.catch(() => {});
}
}).catch(() => {});
} catch {
// Server may not be ready yet — ignore
}
});
}
private get syncManager() {
return this.opsServerRef.gitopsAppRef.syncManager;
}
private get actionLog() {
return this.opsServerRef.gitopsAppRef.actionLog;
}
private registerHandlers(): void {
// Get all sync configs
this.typedrouter.addTypedHandler(
new plugins.typedrequest.TypedHandler<interfaces.requests.IReq_GetSyncConfigs>(
'getSyncConfigs',
async (dataArg) => {
await requireValidIdentity(this.opsServerRef.adminHandler, dataArg);
return { configs: this.syncManager.getConfigs() };
},
),
);
// Create sync config
this.typedrouter.addTypedHandler(
new plugins.typedrequest.TypedHandler<interfaces.requests.IReq_CreateSyncConfig>(
'createSyncConfig',
async (dataArg) => {
await requireValidIdentity(this.opsServerRef.adminHandler, dataArg);
const config = await this.syncManager.createConfig({
name: dataArg.name,
sourceConnectionId: dataArg.sourceConnectionId,
targetConnectionId: dataArg.targetConnectionId,
targetGroupOffset: dataArg.targetGroupOffset,
intervalMinutes: dataArg.intervalMinutes,
enforceDelete: dataArg.enforceDelete,
enforceGroupDelete: dataArg.enforceGroupDelete,
addMirrorHint: dataArg.addMirrorHint,
});
this.actionLog.append({
actionType: 'create',
entityType: 'sync',
entityId: config.id,
entityName: config.name,
details: `Created sync config "${config.name}" (${config.intervalMinutes}m interval)`,
username: dataArg.identity.username,
});
return { config };
},
),
);
// Update sync config
this.typedrouter.addTypedHandler(
new plugins.typedrequest.TypedHandler<interfaces.requests.IReq_UpdateSyncConfig>(
'updateSyncConfig',
async (dataArg) => {
await requireValidIdentity(this.opsServerRef.adminHandler, dataArg);
const config = await this.syncManager.updateConfig(dataArg.syncConfigId, {
name: dataArg.name,
targetGroupOffset: dataArg.targetGroupOffset,
intervalMinutes: dataArg.intervalMinutes,
enforceDelete: dataArg.enforceDelete,
enforceGroupDelete: dataArg.enforceGroupDelete,
addMirrorHint: dataArg.addMirrorHint,
});
this.actionLog.append({
actionType: 'update',
entityType: 'sync',
entityId: config.id,
entityName: config.name,
details: `Updated sync config "${config.name}"`,
username: dataArg.identity.username,
});
return { config };
},
),
);
// Delete sync config
this.typedrouter.addTypedHandler(
new plugins.typedrequest.TypedHandler<interfaces.requests.IReq_DeleteSyncConfig>(
'deleteSyncConfig',
async (dataArg) => {
await requireValidIdentity(this.opsServerRef.adminHandler, dataArg);
const config = this.syncManager.getConfig(dataArg.syncConfigId);
await this.syncManager.deleteConfig(dataArg.syncConfigId);
this.actionLog.append({
actionType: 'delete',
entityType: 'sync',
entityId: dataArg.syncConfigId,
entityName: config?.name || dataArg.syncConfigId,
details: `Deleted sync config "${config?.name || dataArg.syncConfigId}"`,
username: dataArg.identity.username,
});
return { ok: true };
},
),
);
// Pause/resume sync config
this.typedrouter.addTypedHandler(
new plugins.typedrequest.TypedHandler<interfaces.requests.IReq_PauseSyncConfig>(
'pauseSyncConfig',
async (dataArg) => {
await requireValidIdentity(this.opsServerRef.adminHandler, dataArg);
const config = await this.syncManager.pauseConfig(
dataArg.syncConfigId,
dataArg.paused,
);
this.actionLog.append({
actionType: dataArg.paused ? 'pause' : 'resume',
entityType: 'sync',
entityId: config.id,
entityName: config.name,
details: `${dataArg.paused ? 'Paused' : 'Resumed'} sync config "${config.name}"`,
username: dataArg.identity.username,
});
return { config };
},
),
);
// Trigger sync manually
this.typedrouter.addTypedHandler(
new plugins.typedrequest.TypedHandler<interfaces.requests.IReq_TriggerSync>(
'triggerSync',
async (dataArg) => {
await requireValidIdentity(this.opsServerRef.adminHandler, dataArg);
const config = this.syncManager.getConfig(dataArg.syncConfigId);
if (!config) {
return { ok: false, message: 'Sync config not found' };
}
// Fire and forget — force=true bypasses paused check for manual triggers
this.syncManager.executeSync(dataArg.syncConfigId, true).catch((err) => {
console.error(`Manual sync trigger failed: ${err}`);
});
this.actionLog.append({
actionType: 'sync',
entityType: 'sync',
entityId: config.id,
entityName: config.name,
details: `Manually triggered sync "${config.name}"`,
username: dataArg.identity.username,
});
return { ok: true, message: 'Sync triggered' };
},
),
);
// Preview sync (dry run — shows source → target mappings)
this.typedrouter.addTypedHandler(
new plugins.typedrequest.TypedHandler<interfaces.requests.IReq_PreviewSync>(
'previewSync',
async (dataArg) => {
await requireValidIdentity(this.opsServerRef.adminHandler, dataArg);
const result = await this.syncManager.previewSync(dataArg.syncConfigId);
return { mappings: result.mappings, deletions: result.deletions, groupDeletions: result.groupDeletions };
},
),
);
// Get repo statuses for a sync config
this.typedrouter.addTypedHandler(
new plugins.typedrequest.TypedHandler<interfaces.requests.IReq_GetSyncRepoStatuses>(
'getSyncRepoStatuses',
async (dataArg) => {
await requireValidIdentity(this.opsServerRef.adminHandler, dataArg);
const statuses = await this.syncManager.getRepoStatuses(dataArg.syncConfigId);
return { statuses };
},
),
);
// Get recent sync log entries
this.typedrouter.addTypedHandler(
new plugins.typedrequest.TypedHandler<interfaces.requests.IReq_GetSyncLogs>(
'getSyncLogs',
async (dataArg) => {
await requireValidIdentity(this.opsServerRef.adminHandler, dataArg);
const logs = logger.getSyncLogs(dataArg.limit || 200);
return { logs };
},
),
);
}
}