From afda049491de1e1c5853767a4251a7a3e1d2e295 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Iv=C3=A1n=20Ovejero?= Date: Fri, 27 Sep 2024 13:32:12 +0200 Subject: [PATCH] refactor(core): Separate execution `startedAt` from `createdAt` (#10810) --- packages/cli/src/active-executions.ts | 11 +++-- .../concurrency-control.service.ts | 1 - .../databases/entities/execution-entity.ts | 9 +++- ...0136-SeparateExecutionCreationFromStart.ts | 27 ++++++++++++ .../src/databases/migrations/mysqldb/index.ts | 2 + .../databases/migrations/postgresdb/index.ts | 2 + .../src/databases/migrations/sqlite/index.ts | 2 + .../repositories/execution.repository.ts | 42 +++++++++++++------ .../shared/shared-hook-functions.ts | 4 +- .../cli/src/executions/execution.service.ts | 5 +-- packages/cli/src/interfaces.ts | 10 +++-- packages/cli/src/scaling/job-processor.ts | 4 +- .../src/workflow-execute-additional-data.ts | 8 +++- .../workflows/workflow-execution.service.ts | 5 +-- .../execution.service.integration.test.ts | 2 + .../credentials-risk-reporter.test.ts | 2 + .../test/integration/shared/db/executions.ts | 1 + .../design-system/src/css/_tokens.dark.scss | 1 + packages/design-system/src/css/_tokens.scss | 1 + .../global/GlobalExecutionsListItem.vue | 4 +- .../workflow/WorkflowExecutionsCard.test.ts | 21 ++++++++++ .../workflow/WorkflowExecutionsCard.vue | 26 ++++++++++-- .../src/composables/useExecutionHelpers.ts | 2 + .../src/plugins/i18n/locales/en.json | 3 ++ .../src/utils/formatters/dateFormatter.ts | 4 ++ packages/workflow/src/Interfaces.ts | 2 + 26 files changed, 163 insertions(+), 38 deletions(-) create mode 100644 packages/cli/src/databases/migrations/common/1727427440136-SeparateExecutionCreationFromStart.ts diff --git a/packages/cli/src/active-executions.ts b/packages/cli/src/active-executions.ts index 8f7661925b..933bf9a944 100644 --- a/packages/cli/src/active-executions.ts +++ b/packages/cli/src/active-executions.ts @@ -13,7 +13,7 @@ import { Service } from 'typedi'; import { ExecutionRepository } from '@/databases/repositories/execution.repository'; import { ExecutionNotFoundError } from '@/errors/execution-not-found-error'; import type { - ExecutionPayload, + CreateExecutionPayload, IExecutingWorkflowData, IExecutionDb, IExecutionsCurrentSummary, @@ -52,11 +52,10 @@ export class ActiveExecutions { if (executionId === undefined) { // Is a new execution so save in DB - const fullExecutionData: ExecutionPayload = { + const fullExecutionData: CreateExecutionPayload = { data: executionData.executionData!, mode, finished: false, - startedAt: new Date(), workflowData: executionData.workflowData, status: executionStatus, workflowId: executionData.workflowData.id, @@ -74,7 +73,10 @@ export class ActiveExecutions { executionId = await this.executionRepository.createNewExecution(fullExecutionData); assert(executionId); - await this.concurrencyControl.throttle({ mode, executionId }); + if (config.getEnv('executions.mode') === 'regular') { + await this.concurrencyControl.throttle({ mode, executionId }); + await this.executionRepository.setRunning(executionId); + } executionStatus = 'running'; } else { // Is an existing execution we want to finish so update in DB @@ -86,6 +88,7 @@ export class ActiveExecutions { data: executionData.executionData!, waitTill: null, status: executionStatus, + // this is resuming, so keep `startedAt` as it was }; await this.executionRepository.updateExistingExecution(executionId, execution); diff --git a/packages/cli/src/concurrency/concurrency-control.service.ts b/packages/cli/src/concurrency/concurrency-control.service.ts index 45ef2e1206..db7826a258 100644 --- a/packages/cli/src/concurrency/concurrency-control.service.ts +++ b/packages/cli/src/concurrency/concurrency-control.service.ts @@ -70,7 +70,6 @@ export class ConcurrencyControlService { this.productionQueue.on('execution-released', async (executionId) => { this.log('Execution released', { executionId }); - await this.executionRepository.resetStartedAt(executionId); }); } diff --git a/packages/cli/src/databases/entities/execution-entity.ts b/packages/cli/src/databases/entities/execution-entity.ts index f481bb97f4..7b63b63eaf 100644 --- a/packages/cli/src/databases/entities/execution-entity.ts +++ b/packages/cli/src/databases/entities/execution-entity.ts @@ -47,7 +47,14 @@ export class ExecutionEntity { status: ExecutionStatus; @Column(datetimeColumnType) - startedAt: Date; + createdAt: Date; + + /** + * Time when the processing of the execution actually started. This column + * is `null` when an execution is enqueued but has not started yet. + */ + @Column({ type: datetimeColumnType, nullable: true }) + startedAt: Date | null; @Index() @Column({ type: datetimeColumnType, nullable: true }) diff --git a/packages/cli/src/databases/migrations/common/1727427440136-SeparateExecutionCreationFromStart.ts b/packages/cli/src/databases/migrations/common/1727427440136-SeparateExecutionCreationFromStart.ts new file mode 100644 index 0000000000..a44450fa2f --- /dev/null +++ b/packages/cli/src/databases/migrations/common/1727427440136-SeparateExecutionCreationFromStart.ts @@ -0,0 +1,27 @@ +import type { MigrationContext, ReversibleMigration } from '@/databases/types'; + +export class SeparateExecutionCreationFromStart1727427440136 implements ReversibleMigration { + async up({ + schemaBuilder: { addColumns, column, dropNotNull }, + runQuery, + escape, + }: MigrationContext) { + await addColumns('execution_entity', [ + column('createdAt').notNull.timestamp().default('NOW()'), + ]); + + await dropNotNull('execution_entity', 'startedAt'); + + const executionEntity = escape.tableName('execution_entity'); + const createdAt = escape.columnName('createdAt'); + const startedAt = escape.columnName('startedAt'); + + // inaccurate for pre-migration rows but prevents `createdAt` from being nullable + await runQuery(`UPDATE ${executionEntity} SET ${createdAt} = ${startedAt};`); + } + + async down({ schemaBuilder: { dropColumns, addNotNull } }: MigrationContext) { + await dropColumns('execution_entity', ['createdAt']); + await addNotNull('execution_entity', 'startedAt'); + } +} diff --git a/packages/cli/src/databases/migrations/mysqldb/index.ts b/packages/cli/src/databases/migrations/mysqldb/index.ts index 288f18edbe..07b910b949 100644 --- a/packages/cli/src/databases/migrations/mysqldb/index.ts +++ b/packages/cli/src/databases/migrations/mysqldb/index.ts @@ -64,6 +64,7 @@ import { CreateInvalidAuthTokenTable1723627610222 } from '../common/172362761022 import { RefactorExecutionIndices1723796243146 } from '../common/1723796243146-RefactorExecutionIndices'; import { CreateAnnotationTables1724753530828 } from '../common/1724753530828-CreateExecutionAnnotationTables'; import { AddApiKeysTable1724951148974 } from '../common/1724951148974-AddApiKeysTable'; +import { SeparateExecutionCreationFromStart1727427440136 } from '../common/1727427440136-SeparateExecutionCreationFromStart'; export const mysqlMigrations: Migration[] = [ InitialMigration1588157391238, @@ -130,4 +131,5 @@ export const mysqlMigrations: Migration[] = [ RefactorExecutionIndices1723796243146, CreateAnnotationTables1724753530828, AddApiKeysTable1724951148974, + SeparateExecutionCreationFromStart1727427440136, ]; diff --git a/packages/cli/src/databases/migrations/postgresdb/index.ts b/packages/cli/src/databases/migrations/postgresdb/index.ts index 077d686b7e..21b90e201d 100644 --- a/packages/cli/src/databases/migrations/postgresdb/index.ts +++ b/packages/cli/src/databases/migrations/postgresdb/index.ts @@ -64,6 +64,7 @@ import { CreateInvalidAuthTokenTable1723627610222 } from '../common/172362761022 import { RefactorExecutionIndices1723796243146 } from '../common/1723796243146-RefactorExecutionIndices'; import { CreateAnnotationTables1724753530828 } from '../common/1724753530828-CreateExecutionAnnotationTables'; import { AddApiKeysTable1724951148974 } from '../common/1724951148974-AddApiKeysTable'; +import { SeparateExecutionCreationFromStart1727427440136 } from '../common/1727427440136-SeparateExecutionCreationFromStart'; export const postgresMigrations: Migration[] = [ InitialMigration1587669153312, @@ -130,4 +131,5 @@ export const postgresMigrations: Migration[] = [ RefactorExecutionIndices1723796243146, CreateAnnotationTables1724753530828, AddApiKeysTable1724951148974, + SeparateExecutionCreationFromStart1727427440136, ]; diff --git a/packages/cli/src/databases/migrations/sqlite/index.ts b/packages/cli/src/databases/migrations/sqlite/index.ts index 62fda4b7d0..2828bb3f59 100644 --- a/packages/cli/src/databases/migrations/sqlite/index.ts +++ b/packages/cli/src/databases/migrations/sqlite/index.ts @@ -61,6 +61,7 @@ import { AddConstraintToExecutionMetadata1720101653148 } from '../common/1720101 import { CreateInvalidAuthTokenTable1723627610222 } from '../common/1723627610222-CreateInvalidAuthTokenTable'; import { RefactorExecutionIndices1723796243146 } from '../common/1723796243146-RefactorExecutionIndices'; import { CreateAnnotationTables1724753530828 } from '../common/1724753530828-CreateExecutionAnnotationTables'; +import { SeparateExecutionCreationFromStart1727427440136 } from '../common/1727427440136-SeparateExecutionCreationFromStart'; const sqliteMigrations: Migration[] = [ InitialMigration1588102412422, @@ -124,6 +125,7 @@ const sqliteMigrations: Migration[] = [ RefactorExecutionIndices1723796243146, CreateAnnotationTables1724753530828, AddApiKeysTable1724951148974, + SeparateExecutionCreationFromStart1727427440136, ]; export { sqliteMigrations }; diff --git a/packages/cli/src/databases/repositories/execution.repository.ts b/packages/cli/src/databases/repositories/execution.repository.ts index d76d78c99a..7a67b71ace 100644 --- a/packages/cli/src/databases/repositories/execution.repository.ts +++ b/packages/cli/src/databases/repositories/execution.repository.ts @@ -42,7 +42,7 @@ import { ExecutionAnnotation } from '@/databases/entities/execution-annotation.e import { PostgresLiveRowsRetrievalError } from '@/errors/postgres-live-rows-retrieval.error'; import type { ExecutionSummaries } from '@/executions/execution.types'; import type { - ExecutionPayload, + CreateExecutionPayload, IExecutionBase, IExecutionFlattedDb, IExecutionResponse, @@ -198,7 +198,7 @@ export class ExecutionRepository extends Repository { return executions.map((execution) => { const { executionData, ...rest } = execution; return rest; - }); + }) as IExecutionFlattedDb[] | IExecutionResponse[] | IExecutionBase[]; } reportInvalidExecutions(executions: ExecutionEntity[]) { @@ -297,15 +297,15 @@ export class ExecutionRepository extends Repository { }), ...(options?.includeAnnotation && serializedAnnotation && { annotation: serializedAnnotation }), - }; + } as IExecutionFlattedDb | IExecutionResponse | IExecutionBase; } /** * Insert a new execution and its execution data using a transaction. */ - async createNewExecution(execution: ExecutionPayload): Promise { + async createNewExecution(execution: CreateExecutionPayload): Promise { const { data, workflowData, ...rest } = execution; - const { identifiers: inserted } = await this.insert(rest); + const { identifiers: inserted } = await this.insert({ ...rest, createdAt: new Date() }); const { id: executionId } = inserted[0] as { id: string }; const { connections, nodes, name, settings } = workflowData ?? {}; await this.executionDataRepository.insert({ @@ -344,16 +344,25 @@ export class ExecutionRepository extends Repository { await this.update({ id: executionId }, { status }); } - async resetStartedAt(executionId: string) { - await this.update({ id: executionId }, { startedAt: new Date() }); + async setRunning(executionId: string) { + const startedAt = new Date(); + + await this.update({ id: executionId }, { status: 'running', startedAt }); + + return startedAt; } async updateExistingExecution(executionId: string, execution: Partial) { - // 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, workflowId, workflowData, startedAt, customData, ...executionInformation } = - execution; + const { + id, + data, + workflowId, + workflowData, + createdAt, // must never change + startedAt, // must never change + customData, + ...executionInformation + } = execution; if (Object.keys(executionInformation).length > 0) { await this.update({ id: executionId }, executionInformation); } @@ -721,6 +730,7 @@ export class ExecutionRepository extends Repository { mode: true, retryOf: true, status: true, + createdAt: true, startedAt: true, stoppedAt: true, }; @@ -806,6 +816,7 @@ export class ExecutionRepository extends Repository { // @tech_debt: These transformations should not be needed private toSummary(execution: { id: number | string; + createdAt?: Date | string; startedAt?: Date | string; stoppedAt?: Date | string; waitTill?: Date | string | null; @@ -817,6 +828,13 @@ export class ExecutionRepository extends Repository { return date; }; + if (execution.createdAt) { + execution.createdAt = + execution.createdAt instanceof Date + ? execution.createdAt.toISOString() + : normalizeDateString(execution.createdAt); + } + if (execution.startedAt) { execution.startedAt = execution.startedAt instanceof Date diff --git a/packages/cli/src/execution-lifecycle-hooks/shared/shared-hook-functions.ts b/packages/cli/src/execution-lifecycle-hooks/shared/shared-hook-functions.ts index d6d55e63e5..2888f7bd28 100644 --- a/packages/cli/src/execution-lifecycle-hooks/shared/shared-hook-functions.ts +++ b/packages/cli/src/execution-lifecycle-hooks/shared/shared-hook-functions.ts @@ -3,7 +3,7 @@ import type { ExecutionStatus, IRun, IWorkflowBase } from 'n8n-workflow'; import { Container } from 'typedi'; import { ExecutionRepository } from '@/databases/repositories/execution.repository'; -import type { ExecutionPayload, IExecutionDb } from '@/interfaces'; +import type { IExecutionDb, UpdateExecutionPayload } from '@/interfaces'; import { Logger } from '@/logger'; import { ExecutionMetadataService } from '@/services/execution-metadata.service'; import { isWorkflowIdValid } from '@/utils'; @@ -46,7 +46,7 @@ export function prepareExecutionDataForDbUpdate(parameters: { 'pinData', ]); - const fullExecutionData: ExecutionPayload = { + const fullExecutionData: UpdateExecutionPayload = { data: runData.data, mode: runData.mode, finished: runData.finished ? runData.finished : false, diff --git a/packages/cli/src/executions/execution.service.ts b/packages/cli/src/executions/execution.service.ts index 53023fce9a..be685befc9 100644 --- a/packages/cli/src/executions/execution.service.ts +++ b/packages/cli/src/executions/execution.service.ts @@ -32,7 +32,7 @@ import { QueuedExecutionRetryError } from '@/errors/queued-execution-retry.error import { InternalServerError } from '@/errors/response-errors/internal-server.error'; import { NotFoundError } from '@/errors/response-errors/not-found.error'; import type { - ExecutionPayload, + CreateExecutionPayload, IExecutionFlattedResponse, IExecutionResponse, IWorkflowDb, @@ -321,11 +321,10 @@ export class ExecutionService { }, }; - const fullExecutionData: ExecutionPayload = { + const fullExecutionData: CreateExecutionPayload = { data: executionData, mode, finished: false, - startedAt: new Date(), workflowData, workflowId: workflow.id, stoppedAt: new Date(), diff --git a/packages/cli/src/interfaces.ts b/packages/cli/src/interfaces.ts index 4d767862bb..5c29eea093 100644 --- a/packages/cli/src/interfaces.ts +++ b/packages/cli/src/interfaces.ts @@ -115,6 +115,7 @@ export type SaveExecutionDataType = 'all' | 'none'; export interface IExecutionBase { id: string; mode: WorkflowExecuteMode; + createdAt: Date; // set by DB startedAt: Date; stoppedAt?: Date; // empty value means execution is still running workflowId: string; @@ -131,10 +132,11 @@ export interface IExecutionDb extends IExecutionBase { workflowData: IWorkflowBase; } -/** - * Payload for creating or updating an execution. - */ -export type ExecutionPayload = Omit; +/** Payload for creating an execution. */ +export type CreateExecutionPayload = Omit; + +/** Payload for updating an execution. */ +export type UpdateExecutionPayload = Omit; export interface IExecutionResponse extends IExecutionBase { id: string; diff --git a/packages/cli/src/scaling/job-processor.ts b/packages/cli/src/scaling/job-processor.ts index 3155b0d90f..1d0bccc6bf 100644 --- a/packages/cli/src/scaling/job-processor.ts +++ b/packages/cli/src/scaling/job-processor.ts @@ -47,7 +47,7 @@ export class JobProcessor { this.logger.info(`[JobProcessor] Starting job ${job.id} (execution ${executionId})`); - await this.executionRepository.updateStatus(executionId, 'running'); + const startedAt = await this.executionRepository.setRunning(executionId); let { staticData } = execution.workflowData; @@ -137,7 +137,7 @@ export class JobProcessor { workflowId: execution.workflowId, workflowName: execution.workflowData.name, mode: execution.mode, - startedAt: execution.startedAt, + startedAt, retryOf: execution.retryOf ?? '', status: execution.status, }; diff --git a/packages/cli/src/workflow-execute-additional-data.ts b/packages/cli/src/workflow-execute-additional-data.ts index 4d3bd7a223..bc39620c20 100644 --- a/packages/cli/src/workflow-execute-additional-data.ts +++ b/packages/cli/src/workflow-execute-additional-data.ts @@ -41,7 +41,11 @@ import config from '@/config'; import { CredentialsHelper } from '@/credentials-helper'; import { ExecutionRepository } from '@/databases/repositories/execution.repository'; import { ExternalHooks } from '@/external-hooks'; -import type { IWorkflowExecuteProcess, IWorkflowErrorData, ExecutionPayload } from '@/interfaces'; +import type { + IWorkflowExecuteProcess, + IWorkflowErrorData, + UpdateExecutionPayload, +} from '@/interfaces'; import { NodeTypes } from '@/node-types'; import { Push } from '@/push'; import { WorkflowStatisticsService } from '@/services/workflow-statistics.service'; @@ -865,7 +869,7 @@ async function executeWorkflow( // Therefore, database might not contain finished errors. // Force an update to db as there should be no harm doing this - const fullExecutionData: ExecutionPayload = { + const fullExecutionData: UpdateExecutionPayload = { data: fullRunData.data, mode: fullRunData.mode, finished: fullRunData.finished ? fullRunData.finished : false, diff --git a/packages/cli/src/workflows/workflow-execution.service.ts b/packages/cli/src/workflows/workflow-execution.service.ts index 4dc6d00f34..8b1206b22d 100644 --- a/packages/cli/src/workflows/workflow-execution.service.ts +++ b/packages/cli/src/workflows/workflow-execution.service.ts @@ -22,7 +22,7 @@ import type { Project } from '@/databases/entities/project'; import type { User } from '@/databases/entities/user'; import { ExecutionRepository } from '@/databases/repositories/execution.repository'; import { WorkflowRepository } from '@/databases/repositories/workflow.repository'; -import type { ExecutionPayload, IWorkflowDb, IWorkflowErrorData } from '@/interfaces'; +import type { CreateExecutionPayload, IWorkflowDb, IWorkflowErrorData } from '@/interfaces'; import { Logger } from '@/logger'; import { NodeTypes } from '@/node-types'; import { SubworkflowPolicyChecker } from '@/subworkflows/subworkflow-policy-checker.service'; @@ -206,11 +206,10 @@ export class WorkflowExecutionService { initialNode, ); - const fullExecutionData: ExecutionPayload = { + const fullExecutionData: CreateExecutionPayload = { data: fakeExecution.data, mode: fakeExecution.mode, finished: false, - startedAt: new Date(), stoppedAt: new Date(), workflowData, waitTill: null, diff --git a/packages/cli/test/integration/execution.service.integration.test.ts b/packages/cli/test/integration/execution.service.integration.test.ts index 15d97f69ab..22d0d65754 100644 --- a/packages/cli/test/integration/execution.service.integration.test.ts +++ b/packages/cli/test/integration/execution.service.integration.test.ts @@ -70,6 +70,7 @@ describe('ExecutionService', () => { mode: expect.any(String), retryOf: null, status: expect.any(String), + createdAt: expect.any(String), startedAt: expect.any(String), stoppedAt: expect.any(String), waitTill: null, @@ -510,6 +511,7 @@ describe('ExecutionService', () => { mode: expect.any(String), retryOf: null, status: expect.any(String), + createdAt: expect.any(String), startedAt: expect.any(String), stoppedAt: expect.any(String), waitTill: null, diff --git a/packages/cli/test/integration/security-audit/credentials-risk-reporter.test.ts b/packages/cli/test/integration/security-audit/credentials-risk-reporter.test.ts index 8da1f3e1bf..4513beb6bb 100644 --- a/packages/cli/test/integration/security-audit/credentials-risk-reporter.test.ts +++ b/packages/cli/test/integration/security-audit/credentials-risk-reporter.test.ts @@ -159,6 +159,7 @@ test('should report credential in not recently executed workflow', async () => { const savedExecution = await Container.get(ExecutionRepository).save({ finished: true, mode: 'manual', + createdAt: date, startedAt: date, stoppedAt: date, workflowId: workflow.id, @@ -227,6 +228,7 @@ test('should not report credentials in recently executed workflow', async () => const savedExecution = await Container.get(ExecutionRepository).save({ finished: true, mode: 'manual', + createdAt: date, startedAt: date, stoppedAt: date, workflowId: workflow.id, diff --git a/packages/cli/test/integration/shared/db/executions.ts b/packages/cli/test/integration/shared/db/executions.ts index e09ca44dfb..4dd0b4fa76 100644 --- a/packages/cli/test/integration/shared/db/executions.ts +++ b/packages/cli/test/integration/shared/db/executions.ts @@ -39,6 +39,7 @@ export async function createExecution( const execution = await Container.get(ExecutionRepository).save({ finished: finished ?? true, mode: mode ?? 'manual', + createdAt: new Date(), startedAt: startedAt ?? new Date(), ...(workflow !== undefined && { workflowId: workflow.id }), stoppedAt: stoppedAt ?? new Date(), diff --git a/packages/design-system/src/css/_tokens.dark.scss b/packages/design-system/src/css/_tokens.dark.scss index 8377f524c6..72963efcf5 100644 --- a/packages/design-system/src/css/_tokens.dark.scss +++ b/packages/design-system/src/css/_tokens.dark.scss @@ -212,6 +212,7 @@ --execution-selector-background: var(--prim-gray-740); --execution-selector-text: var(--color-text-base); --execution-select-all-text: var(--color-text-base); + --execution-card-text-waiting: var(--prim-color-secondary-tint-100); // NDV --color-run-data-background: var(--prim-gray-800); diff --git a/packages/design-system/src/css/_tokens.scss b/packages/design-system/src/css/_tokens.scss index 1690926232..56d5142c87 100644 --- a/packages/design-system/src/css/_tokens.scss +++ b/packages/design-system/src/css/_tokens.scss @@ -273,6 +273,7 @@ --execution-card-border-running: var(--prim-color-alt-b-tint-250); --execution-card-border-unknown: var(--prim-gray-120); --execution-card-background-hover: var(--color-foreground-light); + --execution-card-text-waiting: var(--color-secondary); --execution-selector-background: var(--color-background-dark); --execution-selector-text: var(--color-text-xlight); --execution-select-all-text: var(--color-danger); diff --git a/packages/editor-ui/src/components/executions/global/GlobalExecutionsListItem.vue b/packages/editor-ui/src/components/executions/global/GlobalExecutionsListItem.vue index 43d85e8e27..f23dbbc4ac 100644 --- a/packages/editor-ui/src/components/executions/global/GlobalExecutionsListItem.vue +++ b/packages/editor-ui/src/components/executions/global/GlobalExecutionsListItem.vue @@ -62,7 +62,9 @@ const classes = computed(() => { }); const formattedStartedAtDate = computed(() => { - return props.execution.startedAt ? formatDate(props.execution.startedAt) : ''; + return props.execution.startedAt + ? formatDate(props.execution.startedAt) + : i18n.baseText('executionsList.startingSoon'); }); const formattedWaitTillDate = computed(() => { diff --git a/packages/editor-ui/src/components/executions/workflow/WorkflowExecutionsCard.test.ts b/packages/editor-ui/src/components/executions/workflow/WorkflowExecutionsCard.test.ts index e1a34eaef6..b9b9645b0a 100644 --- a/packages/editor-ui/src/components/executions/workflow/WorkflowExecutionsCard.test.ts +++ b/packages/editor-ui/src/components/executions/workflow/WorkflowExecutionsCard.test.ts @@ -150,4 +150,25 @@ describe('WorkflowExecutionsCard', () => { } }, ); + + test('displays correct text for new execution', () => { + const createdAt = new Date('2024-09-27T12:00:00Z'); + const props = { + execution: { + id: '1', + mode: 'manual', + status: 'new', + createdAt: createdAt.toISOString(), + }, + workflowPermissions: { + execute: true, + }, + }; + + const { getByTestId } = renderComponent({ props }); + + const executionTimeElement = getByTestId('execution-time'); + expect(executionTimeElement).toBeVisible(); + expect(executionTimeElement.textContent).toBe('27 Sep - Starting soon'); + }); }); diff --git a/packages/editor-ui/src/components/executions/workflow/WorkflowExecutionsCard.vue b/packages/editor-ui/src/components/executions/workflow/WorkflowExecutionsCard.vue index 5c74cc2594..18ba430cb9 100644 --- a/packages/editor-ui/src/components/executions/workflow/WorkflowExecutionsCard.vue +++ b/packages/editor-ui/src/components/executions/workflow/WorkflowExecutionsCard.vue @@ -11,6 +11,7 @@ import { useI18n } from '@/composables/useI18n'; import type { PermissionsRecord } from '@/permissions'; import { usePostHog } from '@/stores/posthog.store'; import { useSettingsStore } from '@/stores/settings.store'; +import { toDayMonth, toTime } from '@/utils/formatters/dateFormatter'; const props = defineProps<{ execution: ExecutionSummary; @@ -87,7 +88,17 @@ function onRetryMenuItemSelect(action: string): void { :data-test-execution-status="executionUIDetails.name" >
- + + {{ toDayMonth(executionUIDetails.createdAt) }} - + {{ locale.baseText('executionDetails.startingSoon') }} + + {{ executionUIDetails.startTime }}
@@ -106,6 +117,15 @@ function onRetryMenuItemSelect(action: string): void { {{ locale.baseText('executionDetails.runningTimeRunning') }} + + {{ locale.baseText('executionDetails.at') }} {{ toTime(execution.createdAt) }} + dateformat(fullDate, 'd mmm'); + +export const toTime = (fullDate: Date | string) => dateformat(fullDate, 'HH:MM:ss'); diff --git a/packages/workflow/src/Interfaces.ts b/packages/workflow/src/Interfaces.ts index 62cf8c5bed..15802fe0b0 100644 --- a/packages/workflow/src/Interfaces.ts +++ b/packages/workflow/src/Interfaces.ts @@ -2122,6 +2122,7 @@ export interface IWorkflowBase { name: string; active: boolean; createdAt: Date; + startedAt?: Date; updatedAt: Date; nodes: INode[]; connections: IConnections; @@ -2463,6 +2464,7 @@ export interface ExecutionSummary { retryOf?: string | null; retrySuccessId?: string | null; waitTill?: Date; + createdAt?: Date; startedAt: Date; stoppedAt?: Date; workflowId: string;