2023-04-12 01:59:14 -07:00
|
|
|
import { Service } from 'typedi';
|
2023-10-04 06:32:05 -07:00
|
|
|
import {
|
|
|
|
Brackets,
|
|
|
|
DataSource,
|
|
|
|
Not,
|
|
|
|
In,
|
|
|
|
IsNull,
|
|
|
|
LessThanOrEqual,
|
|
|
|
MoreThanOrEqual,
|
|
|
|
Repository,
|
|
|
|
} from 'typeorm';
|
2023-09-20 06:21:42 -07:00
|
|
|
import { DateUtils } from 'typeorm/util/DateUtils';
|
2023-06-20 10:13:18 -07:00
|
|
|
import type {
|
|
|
|
FindManyOptions,
|
|
|
|
FindOneOptions,
|
|
|
|
FindOptionsWhere,
|
|
|
|
SelectQueryBuilder,
|
|
|
|
} from 'typeorm';
|
|
|
|
import { parse, stringify } from 'flatted';
|
2023-09-20 06:21:42 -07:00
|
|
|
import type { IExecutionsSummary, IRunExecutionData } from 'n8n-workflow';
|
2023-09-22 08:22:12 -07:00
|
|
|
import { BinaryDataService } from 'n8n-core';
|
2023-06-20 10:13:18 -07:00
|
|
|
import type {
|
2023-09-20 06:21:42 -07:00
|
|
|
ExecutionPayload,
|
2023-06-20 10:13:18 -07:00
|
|
|
IExecutionBase,
|
|
|
|
IExecutionFlattedDb,
|
|
|
|
IExecutionResponse,
|
|
|
|
} from '@/Interfaces';
|
2023-09-20 06:21:42 -07:00
|
|
|
|
|
|
|
import config from '@/config';
|
2023-06-20 10:13:18 -07:00
|
|
|
import type { IGetExecutionsQueryFilter } from '@/executions/executions.service';
|
|
|
|
import { isAdvancedExecutionFiltersEnabled } from '@/executions/executionHelpers';
|
2023-09-20 06:21:42 -07:00
|
|
|
import type { ExecutionData } from '../entities/ExecutionData';
|
|
|
|
import { ExecutionEntity } from '../entities/ExecutionEntity';
|
2023-06-20 10:13:18 -07:00
|
|
|
import { ExecutionMetadata } from '../entities/ExecutionMetadata';
|
2023-09-20 06:21:42 -07:00
|
|
|
import { ExecutionDataRepository } from './executionData.repository';
|
2023-09-27 07:22:39 -07:00
|
|
|
import { TIME, inTest } from '@/constants';
|
2023-10-25 07:35:22 -07:00
|
|
|
import { Logger } from '@/Logger';
|
2023-06-20 10:13:18 -07:00
|
|
|
|
|
|
|
function parseFiltersToQueryBuilder(
|
|
|
|
qb: SelectQueryBuilder<ExecutionEntity>,
|
|
|
|
filters?: IGetExecutionsQueryFilter,
|
|
|
|
) {
|
|
|
|
if (filters?.status) {
|
|
|
|
qb.andWhere('execution.status IN (:...workflowStatus)', {
|
|
|
|
workflowStatus: filters.status,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (filters?.finished) {
|
|
|
|
qb.andWhere({ finished: filters.finished });
|
|
|
|
}
|
|
|
|
if (filters?.metadata && isAdvancedExecutionFiltersEnabled()) {
|
|
|
|
qb.leftJoin(ExecutionMetadata, 'md', 'md.executionId = execution.id');
|
|
|
|
for (const md of filters.metadata) {
|
|
|
|
qb.andWhere('md.key = :key AND md.value = :value', md);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (filters?.startedAfter) {
|
|
|
|
qb.andWhere({
|
|
|
|
startedAt: MoreThanOrEqual(
|
|
|
|
DateUtils.mixedDateToUtcDatetimeString(new Date(filters.startedAfter)),
|
|
|
|
),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (filters?.startedBefore) {
|
|
|
|
qb.andWhere({
|
|
|
|
startedAt: LessThanOrEqual(
|
|
|
|
DateUtils.mixedDateToUtcDatetimeString(new Date(filters.startedBefore)),
|
|
|
|
),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (filters?.workflowId) {
|
|
|
|
qb.andWhere({
|
|
|
|
workflowId: filters.workflowId,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2023-04-12 01:59:14 -07:00
|
|
|
|
|
|
|
@Service()
|
|
|
|
export class ExecutionRepository extends Repository<ExecutionEntity> {
|
2023-10-24 07:16:45 -07:00
|
|
|
private hardDeletionBatchSize = 100;
|
2023-09-20 06:21:42 -07:00
|
|
|
|
|
|
|
private rates: Record<string, number> = {
|
2023-10-19 07:57:12 -07:00
|
|
|
softDeletion: config.getEnv('executions.pruneDataIntervals.softDelete') * TIME.MINUTE,
|
|
|
|
hardDeletion: config.getEnv('executions.pruneDataIntervals.hardDelete') * TIME.MINUTE,
|
2023-09-20 06:21:42 -07:00
|
|
|
};
|
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
private softDeletionInterval: NodeJS.Timer | undefined;
|
|
|
|
|
|
|
|
private hardDeletionTimeout: NodeJS.Timeout | undefined;
|
|
|
|
|
2023-09-20 06:21:42 -07:00
|
|
|
private isMainInstance = config.get('generic.instanceType') === 'main';
|
|
|
|
|
|
|
|
private isPruningEnabled = config.getEnv('executions.pruneData');
|
|
|
|
|
2023-07-18 01:51:22 -07:00
|
|
|
constructor(
|
|
|
|
dataSource: DataSource,
|
2023-10-25 07:35:22 -07:00
|
|
|
private readonly logger: Logger,
|
2023-07-18 01:51:22 -07:00
|
|
|
private readonly executionDataRepository: ExecutionDataRepository,
|
2023-09-22 08:22:12 -07:00
|
|
|
private readonly binaryDataService: BinaryDataService,
|
2023-07-18 01:51:22 -07:00
|
|
|
) {
|
2023-04-12 01:59:14 -07:00
|
|
|
super(ExecutionEntity, dataSource.manager);
|
2023-09-20 06:21:42 -07:00
|
|
|
|
2023-09-27 07:22:39 -07:00
|
|
|
if (!this.isMainInstance || inTest) return;
|
2023-09-20 06:21:42 -07:00
|
|
|
|
|
|
|
if (this.isPruningEnabled) this.setSoftDeletionInterval();
|
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
this.scheduleHardDeletion();
|
2023-09-20 06:21:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
clearTimers() {
|
|
|
|
if (!this.isMainInstance) return;
|
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
this.logger.debug('Clearing soft-deletion interval and hard-deletion timeout (pruning cycle)');
|
2023-09-20 06:21:42 -07:00
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
clearInterval(this.softDeletionInterval);
|
|
|
|
clearTimeout(this.hardDeletionTimeout);
|
2023-09-20 06:21:42 -07:00
|
|
|
}
|
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
setSoftDeletionInterval(rateMs = this.rates.softDeletion) {
|
|
|
|
const when = [(rateMs / TIME.MINUTE).toFixed(2), 'min'].join(' ');
|
|
|
|
|
|
|
|
this.logger.debug(`Setting soft-deletion interval at every ${when} (pruning cycle)`);
|
2023-09-20 06:21:42 -07:00
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
this.softDeletionInterval = setInterval(
|
2023-10-20 06:02:47 -07:00
|
|
|
async () => this.softDeleteOnPruningCycle(),
|
|
|
|
this.rates.softDeletion,
|
|
|
|
);
|
2023-09-20 06:21:42 -07:00
|
|
|
}
|
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
scheduleHardDeletion(rateMs = this.rates.hardDeletion) {
|
|
|
|
const when = [(rateMs / TIME.MINUTE).toFixed(2), 'min'].join(' ');
|
|
|
|
|
|
|
|
this.logger.debug(`Scheduling hard-deletion for next ${when} (pruning cycle)`);
|
2023-09-20 06:21:42 -07:00
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
this.hardDeletionTimeout = setTimeout(
|
2023-10-20 06:02:47 -07:00
|
|
|
async () => this.hardDeleteOnPruningCycle(),
|
2023-09-20 06:21:42 -07:00
|
|
|
this.rates.hardDeletion,
|
|
|
|
);
|
2023-06-20 10:13:18 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
async findMultipleExecutions(
|
|
|
|
queryParams: FindManyOptions<ExecutionEntity>,
|
|
|
|
options?: {
|
|
|
|
unflattenData: true;
|
|
|
|
includeData?: true;
|
|
|
|
},
|
|
|
|
): Promise<IExecutionResponse[]>;
|
|
|
|
async findMultipleExecutions(
|
|
|
|
queryParams: FindManyOptions<ExecutionEntity>,
|
|
|
|
options?: {
|
|
|
|
unflattenData?: false | undefined;
|
|
|
|
includeData?: true;
|
|
|
|
},
|
|
|
|
): Promise<IExecutionFlattedDb[]>;
|
|
|
|
async findMultipleExecutions(
|
|
|
|
queryParams: FindManyOptions<ExecutionEntity>,
|
|
|
|
options?: {
|
|
|
|
unflattenData?: boolean;
|
|
|
|
includeData?: boolean;
|
|
|
|
},
|
|
|
|
): Promise<IExecutionBase[]>;
|
|
|
|
async findMultipleExecutions(
|
|
|
|
queryParams: FindManyOptions<ExecutionEntity>,
|
|
|
|
options?: {
|
|
|
|
unflattenData?: boolean;
|
|
|
|
includeData?: boolean;
|
|
|
|
},
|
|
|
|
): Promise<IExecutionFlattedDb[] | IExecutionResponse[] | IExecutionBase[]> {
|
|
|
|
if (options?.includeData) {
|
|
|
|
if (!queryParams.relations) {
|
|
|
|
queryParams.relations = [];
|
|
|
|
}
|
|
|
|
(queryParams.relations as string[]).push('executionData');
|
|
|
|
}
|
|
|
|
|
|
|
|
const executions = await this.find(queryParams);
|
|
|
|
|
|
|
|
if (options?.includeData && options?.unflattenData) {
|
|
|
|
return executions.map((execution) => {
|
|
|
|
const { executionData, ...rest } = execution;
|
|
|
|
return {
|
|
|
|
...rest,
|
|
|
|
data: parse(executionData.data) as IRunExecutionData,
|
|
|
|
workflowData: executionData.workflowData,
|
|
|
|
} as IExecutionResponse;
|
|
|
|
});
|
|
|
|
} else if (options?.includeData) {
|
|
|
|
return executions.map((execution) => {
|
|
|
|
const { executionData, ...rest } = execution;
|
|
|
|
return {
|
|
|
|
...rest,
|
|
|
|
data: execution.executionData.data,
|
|
|
|
workflowData: execution.executionData.workflowData,
|
|
|
|
} as IExecutionFlattedDb;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return executions.map((execution) => {
|
|
|
|
const { executionData, ...rest } = execution;
|
|
|
|
return rest;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async findSingleExecution(
|
|
|
|
id: string,
|
|
|
|
options?: {
|
|
|
|
includeData: true;
|
|
|
|
unflattenData: true;
|
|
|
|
where?: FindOptionsWhere<ExecutionEntity>;
|
|
|
|
},
|
|
|
|
): Promise<IExecutionResponse | undefined>;
|
|
|
|
async findSingleExecution(
|
|
|
|
id: string,
|
|
|
|
options?: {
|
|
|
|
includeData: true;
|
|
|
|
unflattenData?: false | undefined;
|
|
|
|
where?: FindOptionsWhere<ExecutionEntity>;
|
|
|
|
},
|
|
|
|
): Promise<IExecutionFlattedDb | undefined>;
|
|
|
|
async findSingleExecution(
|
|
|
|
id: string,
|
|
|
|
options?: {
|
|
|
|
includeData?: boolean;
|
|
|
|
unflattenData?: boolean;
|
|
|
|
where?: FindOptionsWhere<ExecutionEntity>;
|
|
|
|
},
|
|
|
|
): Promise<IExecutionBase | undefined>;
|
|
|
|
async findSingleExecution(
|
|
|
|
id: string,
|
|
|
|
options?: {
|
|
|
|
includeData?: boolean;
|
|
|
|
unflattenData?: boolean;
|
|
|
|
where?: FindOptionsWhere<ExecutionEntity>;
|
|
|
|
},
|
|
|
|
): Promise<IExecutionFlattedDb | IExecutionResponse | IExecutionBase | undefined> {
|
|
|
|
const whereClause: FindOneOptions<ExecutionEntity> = {
|
|
|
|
where: {
|
|
|
|
id,
|
|
|
|
...options?.where,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
if (options?.includeData) {
|
|
|
|
whereClause.relations = ['executionData'];
|
|
|
|
}
|
|
|
|
|
|
|
|
const execution = await this.findOne(whereClause);
|
|
|
|
|
|
|
|
if (!execution) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
const { executionData, ...rest } = execution;
|
|
|
|
|
|
|
|
if (options?.includeData && options?.unflattenData) {
|
|
|
|
return {
|
|
|
|
...rest,
|
|
|
|
data: parse(execution.executionData.data) as IRunExecutionData,
|
|
|
|
workflowData: execution.executionData.workflowData,
|
|
|
|
} as IExecutionResponse;
|
|
|
|
} else if (options?.includeData) {
|
|
|
|
return {
|
|
|
|
...rest,
|
|
|
|
data: execution.executionData.data,
|
|
|
|
workflowData: execution.executionData.workflowData,
|
|
|
|
} as IExecutionFlattedDb;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rest;
|
|
|
|
}
|
|
|
|
|
2023-09-20 06:21:42 -07:00
|
|
|
async createNewExecution(execution: ExecutionPayload) {
|
2023-06-20 10:13:18 -07:00
|
|
|
const { data, workflowData, ...rest } = execution;
|
|
|
|
|
|
|
|
const newExecution = await this.save(rest);
|
|
|
|
await this.executionDataRepository.save({
|
|
|
|
execution: newExecution,
|
|
|
|
workflowData,
|
|
|
|
data: stringify(data),
|
|
|
|
});
|
|
|
|
|
|
|
|
return newExecution;
|
|
|
|
}
|
|
|
|
|
2023-09-04 10:58:36 -07:00
|
|
|
async markAsCrashed(executionIds: string[]) {
|
|
|
|
await this.update(
|
|
|
|
{ id: In(executionIds) },
|
|
|
|
{
|
|
|
|
status: 'crashed',
|
|
|
|
stoppedAt: new Date(),
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-10-20 06:02:47 -07:00
|
|
|
/**
|
|
|
|
* Permanently remove a single execution and its binary data.
|
|
|
|
*/
|
|
|
|
async hardDelete(ids: { workflowId: string; executionId: string }) {
|
2023-10-24 01:37:02 -07:00
|
|
|
return Promise.all([this.delete(ids.executionId), this.binaryDataService.deleteMany([ids])]);
|
2023-10-20 06:02:47 -07:00
|
|
|
}
|
|
|
|
|
2023-06-20 10:13:18 -07:00
|
|
|
async updateExistingExecution(executionId: string, execution: Partial<IExecutionResponse>) {
|
|
|
|
// Se isolate startedAt because it must be set when the execution starts and should never change.
|
|
|
|
// So we prevent updating it, if it's sent (it usually is and causes problems to executions that
|
|
|
|
// are resumed after waiting for some time, as a new startedAt is set)
|
|
|
|
const { id, data, workflowData, startedAt, ...executionInformation } = execution;
|
|
|
|
if (Object.keys(executionInformation).length > 0) {
|
|
|
|
await this.update({ id: executionId }, executionInformation);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data || workflowData) {
|
|
|
|
const executionData: Partial<ExecutionData> = {};
|
|
|
|
if (workflowData) {
|
|
|
|
executionData.workflowData = workflowData;
|
|
|
|
}
|
|
|
|
if (data) {
|
|
|
|
executionData.data = stringify(data);
|
|
|
|
}
|
|
|
|
// @ts-ignore
|
|
|
|
await this.executionDataRepository.update({ executionId }, executionData);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async countExecutions(
|
|
|
|
filters: IGetExecutionsQueryFilter | undefined,
|
|
|
|
accessibleWorkflowIds: string[],
|
|
|
|
currentlyRunningExecutions: string[],
|
|
|
|
isOwner: boolean,
|
|
|
|
): Promise<{ count: number; estimated: boolean }> {
|
|
|
|
const dbType = config.getEnv('database.type');
|
|
|
|
if (dbType !== 'postgresdb' || (filters && Object.keys(filters).length > 0) || !isOwner) {
|
|
|
|
const query = this.createQueryBuilder('execution').andWhere(
|
|
|
|
'execution.workflowId IN (:...accessibleWorkflowIds)',
|
|
|
|
{ accessibleWorkflowIds },
|
|
|
|
);
|
|
|
|
if (currentlyRunningExecutions.length > 0) {
|
|
|
|
query.andWhere('execution.id NOT IN (:...currentlyRunningExecutions)', {
|
|
|
|
currentlyRunningExecutions,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
parseFiltersToQueryBuilder(query, filters);
|
|
|
|
|
|
|
|
const count = await query.getCount();
|
|
|
|
return { count, estimated: false };
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Get an estimate of rows count.
|
|
|
|
const estimateRowsNumberSql =
|
|
|
|
"SELECT n_live_tup FROM pg_stat_all_tables WHERE relname = 'execution_entity';";
|
|
|
|
const rows = (await this.query(estimateRowsNumberSql)) as Array<{ n_live_tup: string }>;
|
|
|
|
|
|
|
|
const estimate = parseInt(rows[0].n_live_tup, 10);
|
|
|
|
// If over 100k, return just an estimate.
|
|
|
|
if (estimate > 100_000) {
|
|
|
|
// if less than 100k, we get the real count as even a full
|
|
|
|
// table scan should not take so long.
|
|
|
|
return { count: estimate, estimated: true };
|
|
|
|
}
|
|
|
|
} catch (error) {
|
|
|
|
if (error instanceof Error) {
|
2023-10-25 07:35:22 -07:00
|
|
|
this.logger.warn(`Failed to get executions count from Postgres: ${error.message}`, {
|
2023-06-20 10:13:18 -07:00
|
|
|
error,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const count = await this.count({
|
|
|
|
where: {
|
|
|
|
workflowId: In(accessibleWorkflowIds),
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
return { count, estimated: false };
|
|
|
|
}
|
|
|
|
|
|
|
|
async searchExecutions(
|
|
|
|
filters: IGetExecutionsQueryFilter | undefined,
|
|
|
|
limit: number,
|
|
|
|
excludedExecutionIds: string[],
|
|
|
|
accessibleWorkflowIds: string[],
|
|
|
|
additionalFilters?: { lastId?: string; firstId?: string },
|
|
|
|
): Promise<IExecutionsSummary[]> {
|
|
|
|
if (accessibleWorkflowIds.length === 0) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
const query = this.createQueryBuilder('execution')
|
|
|
|
.select([
|
|
|
|
'execution.id',
|
|
|
|
'execution.finished',
|
|
|
|
'execution.mode',
|
|
|
|
'execution.retryOf',
|
|
|
|
'execution.retrySuccessId',
|
|
|
|
'execution.status',
|
|
|
|
'execution.startedAt',
|
|
|
|
'execution.stoppedAt',
|
|
|
|
'execution.workflowId',
|
|
|
|
'execution.waitTill',
|
|
|
|
'workflow.name',
|
|
|
|
])
|
|
|
|
.innerJoin('execution.workflow', 'workflow')
|
|
|
|
.limit(limit)
|
|
|
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
|
|
|
.orderBy({ 'execution.id': 'DESC' })
|
|
|
|
.andWhere('execution.workflowId IN (:...accessibleWorkflowIds)', { accessibleWorkflowIds });
|
|
|
|
if (excludedExecutionIds.length > 0) {
|
|
|
|
query.andWhere('execution.id NOT IN (:...excludedExecutionIds)', { excludedExecutionIds });
|
|
|
|
}
|
|
|
|
|
|
|
|
if (additionalFilters?.lastId) {
|
|
|
|
query.andWhere('execution.id < :lastId', { lastId: additionalFilters.lastId });
|
|
|
|
}
|
|
|
|
if (additionalFilters?.firstId) {
|
|
|
|
query.andWhere('execution.id > :firstId', { firstId: additionalFilters.firstId });
|
|
|
|
}
|
|
|
|
|
|
|
|
parseFiltersToQueryBuilder(query, filters);
|
|
|
|
|
|
|
|
const executions = await query.getMany();
|
|
|
|
|
|
|
|
return executions.map((execution) => {
|
|
|
|
const { workflow, waitTill, ...rest } = execution;
|
|
|
|
return {
|
|
|
|
...rest,
|
|
|
|
waitTill: waitTill ?? undefined,
|
|
|
|
workflowName: workflow.name,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-09-20 06:21:42 -07:00
|
|
|
async deleteExecutionsByFilter(
|
2023-06-20 10:13:18 -07:00
|
|
|
filters: IGetExecutionsQueryFilter | undefined,
|
|
|
|
accessibleWorkflowIds: string[],
|
|
|
|
deleteConditions: {
|
|
|
|
deleteBefore?: Date;
|
|
|
|
ids?: string[];
|
|
|
|
},
|
|
|
|
) {
|
|
|
|
if (!deleteConditions?.deleteBefore && !deleteConditions?.ids) {
|
|
|
|
throw new Error('Either "deleteBefore" or "ids" must be present in the request body');
|
|
|
|
}
|
|
|
|
|
|
|
|
const query = this.createQueryBuilder('execution')
|
|
|
|
.select(['execution.id'])
|
|
|
|
.andWhere('execution.workflowId IN (:...accessibleWorkflowIds)', { accessibleWorkflowIds });
|
|
|
|
|
|
|
|
if (deleteConditions.deleteBefore) {
|
|
|
|
// delete executions by date, if user may access the underlying workflows
|
|
|
|
query.andWhere('execution.startedAt <= :deleteBefore', {
|
|
|
|
deleteBefore: deleteConditions.deleteBefore,
|
|
|
|
});
|
|
|
|
// Filters are only used when filtering by date
|
|
|
|
parseFiltersToQueryBuilder(query, filters);
|
|
|
|
} else if (deleteConditions.ids) {
|
|
|
|
// delete executions by IDs, if user may access the underlying workflows
|
|
|
|
query.andWhere('execution.id IN (:...executionIds)', { executionIds: deleteConditions.ids });
|
|
|
|
}
|
|
|
|
|
|
|
|
const executions = await query.getMany();
|
|
|
|
|
|
|
|
if (!executions.length) {
|
|
|
|
if (deleteConditions.ids) {
|
2023-10-25 07:35:22 -07:00
|
|
|
this.logger.error('Failed to delete an execution due to insufficient permissions', {
|
2023-06-20 10:13:18 -07:00
|
|
|
executionIds: deleteConditions.ids,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-07-12 01:08:29 -07:00
|
|
|
const executionIds = executions.map(({ id }) => id);
|
2023-06-20 10:13:18 -07:00
|
|
|
do {
|
|
|
|
// Delete in batches to avoid "SQLITE_ERROR: Expression tree is too large (maximum depth 1000)" error
|
2023-10-24 07:16:45 -07:00
|
|
|
const batch = executionIds.splice(0, this.hardDeletionBatchSize);
|
2023-10-20 06:02:47 -07:00
|
|
|
await this.delete(batch);
|
2023-07-12 01:08:29 -07:00
|
|
|
} while (executionIds.length > 0);
|
2023-04-12 01:59:14 -07:00
|
|
|
}
|
2023-09-20 06:21:42 -07:00
|
|
|
|
2023-10-20 06:02:47 -07:00
|
|
|
/**
|
|
|
|
* Mark executions as deleted based on age and count, in a pruning cycle.
|
|
|
|
*/
|
|
|
|
async softDeleteOnPruningCycle() {
|
2023-10-25 07:35:22 -07:00
|
|
|
this.logger.debug('Starting soft-deletion of executions (pruning cycle)');
|
2023-09-20 06:21:42 -07:00
|
|
|
|
|
|
|
const maxAge = config.getEnv('executions.pruneDataMaxAge'); // in h
|
|
|
|
const maxCount = config.getEnv('executions.pruneDataMaxCount');
|
|
|
|
|
|
|
|
// Find ids of all executions that were stopped longer that pruneDataMaxAge ago
|
|
|
|
const date = new Date();
|
|
|
|
date.setHours(date.getHours() - maxAge);
|
|
|
|
|
|
|
|
const toPrune: Array<FindOptionsWhere<ExecutionEntity>> = [
|
|
|
|
// date reformatting needed - see https://github.com/typeorm/typeorm/issues/2286
|
|
|
|
{ stoppedAt: LessThanOrEqual(DateUtils.mixedDateToUtcDatetimeString(date)) },
|
|
|
|
];
|
|
|
|
|
|
|
|
if (maxCount > 0) {
|
|
|
|
const executions = await this.find({
|
|
|
|
select: ['id'],
|
|
|
|
skip: maxCount,
|
|
|
|
take: 1,
|
|
|
|
order: { id: 'DESC' },
|
|
|
|
});
|
|
|
|
|
|
|
|
if (executions[0]) {
|
|
|
|
toPrune.push({ id: LessThanOrEqual(executions[0].id) });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const [timeBasedWhere, countBasedWhere] = toPrune;
|
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
const result = await this.createQueryBuilder()
|
2023-09-20 06:21:42 -07:00
|
|
|
.update(ExecutionEntity)
|
|
|
|
.set({ deletedAt: new Date() })
|
2023-10-04 06:32:05 -07:00
|
|
|
.where({
|
|
|
|
deletedAt: IsNull(),
|
|
|
|
// Only mark executions as deleted if they are in an end state
|
|
|
|
status: Not(In(['new', 'running', 'waiting'])),
|
|
|
|
})
|
|
|
|
.andWhere(
|
2023-09-20 06:21:42 -07:00
|
|
|
new Brackets((qb) =>
|
|
|
|
countBasedWhere
|
|
|
|
? qb.where(timeBasedWhere).orWhere(countBasedWhere)
|
|
|
|
: qb.where(timeBasedWhere),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
.execute();
|
2023-10-24 07:16:45 -07:00
|
|
|
|
|
|
|
if (result.affected === 0) {
|
2023-10-25 07:35:22 -07:00
|
|
|
this.logger.debug('Found no executions to soft-delete (pruning cycle)');
|
2023-10-24 07:16:45 -07:00
|
|
|
}
|
2023-09-20 06:21:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-10-20 06:02:47 -07:00
|
|
|
* Permanently remove all soft-deleted executions and their binary data, in a pruning cycle.
|
2023-09-20 06:21:42 -07:00
|
|
|
*/
|
2023-10-20 06:02:47 -07:00
|
|
|
private async hardDeleteOnPruningCycle() {
|
2023-09-20 06:21:42 -07:00
|
|
|
const date = new Date();
|
2023-10-19 07:57:12 -07:00
|
|
|
date.setHours(date.getHours() - config.getEnv('executions.pruneDataHardDeleteBuffer'));
|
2023-09-20 06:21:42 -07:00
|
|
|
|
2023-09-27 00:42:35 -07:00
|
|
|
const workflowIdsAndExecutionIds = (
|
2023-09-20 06:21:42 -07:00
|
|
|
await this.find({
|
2023-09-27 00:42:35 -07:00
|
|
|
select: ['workflowId', 'id'],
|
2023-09-20 06:21:42 -07:00
|
|
|
where: {
|
|
|
|
deletedAt: LessThanOrEqual(DateUtils.mixedDateToUtcDatetimeString(date)),
|
|
|
|
},
|
2023-10-24 07:16:45 -07:00
|
|
|
take: this.hardDeletionBatchSize,
|
2023-09-20 06:21:42 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @important This ensures soft-deleted executions are included,
|
|
|
|
* else `@DeleteDateColumn()` at `deletedAt` will exclude them.
|
|
|
|
*/
|
|
|
|
withDeleted: true,
|
|
|
|
})
|
2023-09-27 00:42:35 -07:00
|
|
|
).map(({ id: executionId, workflowId }) => ({ workflowId, executionId }));
|
|
|
|
|
|
|
|
const executionIds = workflowIdsAndExecutionIds.map((o) => o.executionId);
|
2023-09-20 06:21:42 -07:00
|
|
|
|
2023-09-26 07:53:38 -07:00
|
|
|
if (executionIds.length === 0) {
|
2023-10-24 07:16:45 -07:00
|
|
|
this.logger.debug('Found no executions to hard-delete (pruning cycle)');
|
|
|
|
this.scheduleHardDeletion();
|
2023-09-26 07:53:38 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
try {
|
|
|
|
this.logger.debug('Starting hard-deletion of executions (pruning cycle)', {
|
2023-10-20 06:02:47 -07:00
|
|
|
executionIds,
|
2023-10-24 07:16:45 -07:00
|
|
|
});
|
2023-09-20 06:21:42 -07:00
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
await this.binaryDataService.deleteMany(workflowIdsAndExecutionIds);
|
|
|
|
|
|
|
|
await this.delete({ id: In(executionIds) });
|
|
|
|
} catch (error) {
|
|
|
|
this.logger.error('Failed to hard-delete executions (pruning cycle)', {
|
|
|
|
executionIds,
|
|
|
|
error: error instanceof Error ? error.message : `${error}`,
|
|
|
|
});
|
|
|
|
}
|
2023-09-20 06:21:42 -07:00
|
|
|
|
|
|
|
/**
|
2023-10-24 07:16:45 -07:00
|
|
|
* For next batch, speed up hard-deletion cycle in high-volume case
|
|
|
|
* to prevent high concurrency from causing duplicate deletions.
|
2023-09-20 06:21:42 -07:00
|
|
|
*/
|
2023-10-24 07:16:45 -07:00
|
|
|
const isHighVolume = executionIds.length >= this.hardDeletionBatchSize;
|
|
|
|
const rate = isHighVolume ? 1 * TIME.SECOND : this.rates.hardDeletion;
|
2023-09-20 06:21:42 -07:00
|
|
|
|
2023-10-24 07:16:45 -07:00
|
|
|
this.scheduleHardDeletion(rate);
|
2023-09-20 06:21:42 -07:00
|
|
|
}
|
2023-04-12 01:59:14 -07:00
|
|
|
}
|