2019-06-23 03:35:23 -07:00
|
|
|
<template>
|
2022-01-21 09:00:00 -08:00
|
|
|
<Modal
|
|
|
|
:name="EXECUTIONS_MODAL_KEY"
|
|
|
|
width="80%"
|
2022-12-14 01:04:10 -08:00
|
|
|
:title="`${$locale.baseText('executionsList.workflowExecutions')} ${
|
|
|
|
combinedExecutions.length
|
|
|
|
}/${finishedExecutionsCountEstimated === true ? '~' : ''}${combinedExecutionsCount}`"
|
2022-01-21 09:00:00 -08:00
|
|
|
:eventBus="modalBus"
|
|
|
|
>
|
2022-11-18 05:59:31 -08:00
|
|
|
<template #content>
|
2022-09-21 01:20:29 -07:00
|
|
|
<div class="filters">
|
2019-06-23 03:35:23 -07:00
|
|
|
<el-row>
|
2021-08-29 04:36:17 -07:00
|
|
|
<el-col :span="2" class="filter-headline">
|
2021-12-15 04:16:53 -08:00
|
|
|
{{ $locale.baseText('executionsList.filters') }}:
|
2019-06-23 03:35:23 -07:00
|
|
|
</el-col>
|
2021-08-29 04:36:17 -07:00
|
|
|
<el-col :span="7">
|
2022-12-14 01:04:10 -08:00
|
|
|
<n8n-select
|
|
|
|
v-model="filter.workflowId"
|
|
|
|
:placeholder="$locale.baseText('executionsList.selectWorkflow')"
|
|
|
|
size="medium"
|
|
|
|
filterable
|
|
|
|
@change="handleFilterChanged"
|
|
|
|
>
|
2022-09-21 01:20:29 -07:00
|
|
|
<div class="ph-no-capture">
|
|
|
|
<n8n-option
|
|
|
|
v-for="item in workflows"
|
|
|
|
:key="item.id"
|
|
|
|
:label="item.name"
|
2022-12-14 01:04:10 -08:00
|
|
|
:value="item.id"
|
|
|
|
>
|
2022-09-21 01:20:29 -07:00
|
|
|
</n8n-option>
|
|
|
|
</div>
|
2021-08-29 04:36:17 -07:00
|
|
|
</n8n-select>
|
2019-06-23 03:35:23 -07:00
|
|
|
</el-col>
|
2021-08-29 04:36:17 -07:00
|
|
|
<el-col :span="5" :offset="1">
|
2022-12-14 01:04:10 -08:00
|
|
|
<n8n-select
|
|
|
|
v-model="filter.status"
|
|
|
|
:placeholder="$locale.baseText('executionsList.selectStatus')"
|
|
|
|
size="medium"
|
|
|
|
filterable
|
|
|
|
@change="handleFilterChanged"
|
|
|
|
>
|
2021-08-29 04:36:17 -07:00
|
|
|
<n8n-option
|
2019-12-12 15:39:56 -08:00
|
|
|
v-for="item in statuses"
|
|
|
|
:key="item.id"
|
|
|
|
:label="item.name"
|
2022-12-14 01:04:10 -08:00
|
|
|
:value="item.id"
|
|
|
|
>
|
2021-08-29 04:36:17 -07:00
|
|
|
</n8n-option>
|
|
|
|
</n8n-select>
|
2021-02-09 14:32:40 -08:00
|
|
|
</el-col>
|
2021-08-29 04:36:17 -07:00
|
|
|
<el-col :span="4" :offset="5" class="autorefresh">
|
2022-12-14 01:04:10 -08:00
|
|
|
<el-checkbox v-model="autoRefresh" @change="handleAutoRefreshToggle">{{
|
|
|
|
$locale.baseText('executionsList.autoRefresh')
|
|
|
|
}}</el-checkbox>
|
2019-06-23 03:35:23 -07:00
|
|
|
</el-col>
|
|
|
|
</el-row>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="selection-options">
|
|
|
|
<span v-if="checkAll === true || isIndeterminate === true">
|
2022-12-14 01:04:10 -08:00
|
|
|
{{ $locale.baseText('executionsList.selected') }}: {{ numSelected }} /
|
|
|
|
<span v-if="finishedExecutionsCountEstimated === true">~</span
|
|
|
|
>{{ finishedExecutionsCount }}
|
|
|
|
<n8n-icon-button
|
|
|
|
:title="$locale.baseText('executionsList.deleteSelected')"
|
|
|
|
icon="trash"
|
|
|
|
size="mini"
|
|
|
|
@click="handleDeleteSelected"
|
|
|
|
/>
|
2019-06-23 03:35:23 -07:00
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
<el-table
|
|
|
|
:data="combinedExecutions"
|
|
|
|
stripe
|
|
|
|
v-loading="isDataLoading"
|
|
|
|
:row-class-name="getRowClass"
|
|
|
|
>
|
2019-06-23 03:35:23 -07:00
|
|
|
<el-table-column label="" width="30">
|
|
|
|
<!-- eslint-disable-next-line vue/no-unused-vars -->
|
2022-12-14 01:04:10 -08:00
|
|
|
<template #header="scope">
|
|
|
|
<el-checkbox
|
|
|
|
:indeterminate="isIndeterminate"
|
|
|
|
v-model="checkAll"
|
|
|
|
@change="handleCheckAllChange"
|
|
|
|
label=" "
|
|
|
|
></el-checkbox>
|
2019-06-23 03:35:23 -07:00
|
|
|
</template>
|
2022-11-18 05:59:31 -08:00
|
|
|
<template #default="scope">
|
2022-12-14 01:04:10 -08:00
|
|
|
<el-checkbox
|
|
|
|
v-if="scope.row.stoppedAt !== undefined && scope.row.id"
|
|
|
|
:value="selectedItems[scope.row.id.toString()] || checkAll"
|
|
|
|
@change="handleCheckboxChanged(scope.row.id)"
|
|
|
|
label=" "
|
|
|
|
></el-checkbox>
|
2019-06-23 03:35:23 -07:00
|
|
|
</template>
|
|
|
|
</el-table-column>
|
2022-12-14 01:04:10 -08:00
|
|
|
<el-table-column
|
|
|
|
property="startedAt"
|
|
|
|
:label="$locale.baseText('executionsList.startedAtId')"
|
|
|
|
width="205"
|
|
|
|
>
|
2022-11-18 05:59:31 -08:00
|
|
|
<template #default="scope">
|
2022-12-14 01:04:10 -08:00
|
|
|
{{ convertToDisplayDate(scope.row.startedAt) }}<br />
|
|
|
|
<small v-if="scope.row.id">ID: {{ scope.row.id }}</small>
|
2019-06-23 03:35:23 -07:00
|
|
|
</template>
|
|
|
|
</el-table-column>
|
2021-12-15 04:16:53 -08:00
|
|
|
<el-table-column property="workflowName" :label="$locale.baseText('executionsList.name')">
|
2022-11-18 05:59:31 -08:00
|
|
|
<template #default="scope">
|
2022-09-21 01:20:29 -07:00
|
|
|
<div class="ph-no-capture">
|
|
|
|
<span class="workflow-name">
|
|
|
|
{{ scope.row.workflowName || $locale.baseText('executionsList.unsavedWorkflow') }}
|
|
|
|
</span>
|
|
|
|
</div>
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
<span v-if="scope.row.stoppedAt === undefined">
|
2021-12-15 04:16:53 -08:00
|
|
|
({{ $locale.baseText('executionsList.running') }})
|
2019-06-23 03:35:23 -07:00
|
|
|
</span>
|
|
|
|
<span v-if="scope.row.retryOf !== undefined">
|
2022-12-14 01:04:10 -08:00
|
|
|
<br /><small
|
|
|
|
>{{ $locale.baseText('executionsList.retryOf') }} "{{ scope.row.retryOf }}"</small
|
|
|
|
>
|
2019-06-23 03:35:23 -07:00
|
|
|
</span>
|
|
|
|
<span v-else-if="scope.row.retrySuccessId !== undefined">
|
2022-12-14 01:04:10 -08:00
|
|
|
<br /><small
|
|
|
|
>{{ $locale.baseText('executionsList.successRetry') }} "{{
|
|
|
|
scope.row.retrySuccessId
|
|
|
|
}}"</small
|
|
|
|
>
|
2019-06-23 03:35:23 -07:00
|
|
|
</span>
|
|
|
|
</template>
|
|
|
|
</el-table-column>
|
2022-12-14 01:04:10 -08:00
|
|
|
<el-table-column
|
|
|
|
:label="$locale.baseText('executionsList.status')"
|
|
|
|
width="122"
|
|
|
|
align="center"
|
|
|
|
>
|
2022-11-18 05:59:31 -08:00
|
|
|
<template #default="scope" align="center">
|
2022-12-14 01:04:10 -08:00
|
|
|
<n8n-tooltip placement="top">
|
2022-11-18 05:59:31 -08:00
|
|
|
<template #content>
|
|
|
|
<div v-html="statusTooltipText(scope.row)"></div>
|
|
|
|
</template>
|
2021-12-20 06:30:27 -08:00
|
|
|
<span class="status-badge running" v-if="scope.row.waitTill">
|
|
|
|
{{ $locale.baseText('executionsList.waiting') }}
|
|
|
|
</span>
|
2021-12-20 08:18:28 -08:00
|
|
|
<span class="status-badge running" v-else-if="scope.row.stoppedAt === undefined">
|
2021-12-15 04:16:53 -08:00
|
|
|
{{ $locale.baseText('executionsList.running') }}
|
2019-06-23 03:35:23 -07:00
|
|
|
</span>
|
|
|
|
<span class="status-badge success" v-else-if="scope.row.finished">
|
2021-12-15 04:16:53 -08:00
|
|
|
{{ $locale.baseText('executionsList.success') }}
|
2019-06-23 03:35:23 -07:00
|
|
|
</span>
|
2021-03-02 23:31:55 -08:00
|
|
|
<span class="status-badge error" v-else-if="scope.row.stoppedAt !== null">
|
2021-12-15 04:16:53 -08:00
|
|
|
{{ $locale.baseText('executionsList.error') }}
|
2019-06-23 03:35:23 -07:00
|
|
|
</span>
|
2021-03-02 23:31:55 -08:00
|
|
|
<span class="status-badge warning" v-else>
|
2021-12-15 04:16:53 -08:00
|
|
|
{{ $locale.baseText('executionsList.unknown') }}
|
2021-03-02 23:31:55 -08:00
|
|
|
</span>
|
2021-08-29 04:36:17 -07:00
|
|
|
</n8n-tooltip>
|
2019-06-23 03:35:23 -07:00
|
|
|
|
2019-12-12 16:12:38 -08:00
|
|
|
<el-dropdown trigger="click" @command="handleRetryClick">
|
2022-08-12 06:29:25 -07:00
|
|
|
<span class="retry-button">
|
|
|
|
<n8n-icon-button
|
2022-12-14 01:04:10 -08:00
|
|
|
v-if="
|
|
|
|
scope.row.stoppedAt !== undefined &&
|
|
|
|
!scope.row.finished &&
|
|
|
|
scope.row.retryOf === undefined &&
|
|
|
|
scope.row.retrySuccessId === undefined &&
|
|
|
|
!scope.row.waitTill
|
|
|
|
"
|
|
|
|
:type="scope.row.stoppedAt === null ? 'warning' : 'danger'"
|
2022-08-12 06:29:25 -07:00
|
|
|
class="ml-3xs"
|
|
|
|
size="mini"
|
|
|
|
:title="$locale.baseText('executionsList.retryExecution')"
|
|
|
|
icon="redo"
|
|
|
|
/>
|
|
|
|
</span>
|
2022-11-18 05:59:31 -08:00
|
|
|
<template #dropdown>
|
|
|
|
<el-dropdown-menu>
|
2022-12-14 01:04:10 -08:00
|
|
|
<el-dropdown-item :command="{ command: 'currentlySaved', row: scope.row }">
|
2022-11-18 05:59:31 -08:00
|
|
|
{{ $locale.baseText('executionsList.retryWithCurrentlySavedWorkflow') }}
|
|
|
|
</el-dropdown-item>
|
2022-12-14 01:04:10 -08:00
|
|
|
<el-dropdown-item :command="{ command: 'original', row: scope.row }">
|
2022-12-15 21:22:31 -08:00
|
|
|
{{ $locale.baseText('executionsList.retryWithOriginalWorkflow') }}
|
2022-11-18 05:59:31 -08:00
|
|
|
</el-dropdown-item>
|
|
|
|
</el-dropdown-menu>
|
|
|
|
</template>
|
2019-12-12 16:12:38 -08:00
|
|
|
</el-dropdown>
|
2019-06-23 03:35:23 -07:00
|
|
|
</template>
|
|
|
|
</el-table-column>
|
2022-12-14 01:04:10 -08:00
|
|
|
<el-table-column
|
|
|
|
property="mode"
|
|
|
|
:label="$locale.baseText('executionsList.mode')"
|
|
|
|
width="100"
|
|
|
|
align="center"
|
|
|
|
>
|
2022-11-18 05:59:31 -08:00
|
|
|
<template #default="scope">
|
2021-12-15 04:16:53 -08:00
|
|
|
{{ $locale.baseText(`executionsList.modes.${scope.row.mode}`) }}
|
2021-11-20 09:28:05 -08:00
|
|
|
</template>
|
2021-11-18 02:32:13 -08:00
|
|
|
</el-table-column>
|
2022-12-14 01:04:10 -08:00
|
|
|
<el-table-column
|
|
|
|
:label="$locale.baseText('executionsList.runningTime')"
|
|
|
|
width="150"
|
|
|
|
align="center"
|
|
|
|
>
|
2022-11-18 05:59:31 -08:00
|
|
|
<template #default="scope">
|
2019-06-23 03:35:23 -07:00
|
|
|
<span v-if="scope.row.stoppedAt === undefined">
|
|
|
|
<font-awesome-icon icon="spinner" spin />
|
2022-12-14 01:04:10 -08:00
|
|
|
<execution-time :start-time="scope.row.startedAt" />
|
2019-06-23 03:35:23 -07:00
|
|
|
</span>
|
2021-02-08 23:59:32 -08:00
|
|
|
<!-- stoppedAt will be null if process crashed -->
|
2022-12-14 01:04:10 -08:00
|
|
|
<span v-else-if="scope.row.stoppedAt === null"> -- </span>
|
2019-06-23 03:35:23 -07:00
|
|
|
<span v-else>
|
2022-12-14 01:04:10 -08:00
|
|
|
{{
|
|
|
|
displayTimer(
|
|
|
|
new Date(scope.row.stoppedAt).getTime() - new Date(scope.row.startedAt).getTime(),
|
|
|
|
true,
|
|
|
|
)
|
|
|
|
}}
|
2019-06-23 03:35:23 -07:00
|
|
|
</span>
|
|
|
|
</template>
|
|
|
|
</el-table-column>
|
|
|
|
<el-table-column label="" width="100" align="center">
|
2022-11-18 05:59:31 -08:00
|
|
|
<template #default="scope">
|
2021-08-29 04:36:17 -07:00
|
|
|
<div class="actions-container">
|
|
|
|
<span v-if="scope.row.stoppedAt === undefined || scope.row.waitTill">
|
2022-12-14 01:04:10 -08:00
|
|
|
<n8n-icon-button
|
|
|
|
icon="stop"
|
|
|
|
size="small"
|
|
|
|
:title="$locale.baseText('executionsList.stopExecution')"
|
|
|
|
@click.stop="stopExecution(scope.row.id)"
|
|
|
|
:loading="stoppingExecutions.includes(scope.row.id)"
|
|
|
|
/>
|
2021-08-29 04:36:17 -07:00
|
|
|
</span>
|
2022-12-14 01:04:10 -08:00
|
|
|
<span v-if="scope.row.stoppedAt !== undefined && scope.row.id">
|
|
|
|
<n8n-icon-button
|
|
|
|
icon="folder-open"
|
|
|
|
size="small"
|
|
|
|
:title="$locale.baseText('executionsList.openPastExecution')"
|
|
|
|
@click.stop="(e) => displayExecution(scope.row, e)"
|
|
|
|
/>
|
2021-08-29 04:36:17 -07:00
|
|
|
</span>
|
|
|
|
</div>
|
2019-06-23 03:35:23 -07:00
|
|
|
</template>
|
|
|
|
</el-table-column>
|
|
|
|
</el-table>
|
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
<div
|
|
|
|
class="load-more"
|
|
|
|
v-if="
|
|
|
|
finishedExecutionsCount > finishedExecutions.length ||
|
|
|
|
finishedExecutionsCountEstimated === true
|
|
|
|
"
|
|
|
|
>
|
|
|
|
<n8n-button
|
|
|
|
icon="sync"
|
|
|
|
:title="$locale.baseText('executionsList.loadMore')"
|
|
|
|
:label="$locale.baseText('executionsList.loadMore')"
|
|
|
|
@click="loadMore()"
|
|
|
|
:loading="isDataLoading"
|
|
|
|
/>
|
2019-06-23 03:35:23 -07:00
|
|
|
</div>
|
2022-01-21 09:00:00 -08:00
|
|
|
</template>
|
|
|
|
</Modal>
|
2019-06-23 03:35:23 -07:00
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts">
|
2022-09-23 07:14:28 -07:00
|
|
|
/* eslint-disable prefer-spread */
|
2019-06-23 03:35:23 -07:00
|
|
|
import Vue from 'vue';
|
|
|
|
|
2019-07-24 05:25:30 -07:00
|
|
|
import ExecutionTime from '@/components/ExecutionTime.vue';
|
2019-06-23 03:35:23 -07:00
|
|
|
import WorkflowActivator from '@/components/WorkflowActivator.vue';
|
2022-01-21 09:00:00 -08:00
|
|
|
import Modal from '@/components/Modal.vue';
|
2019-06-23 03:35:23 -07:00
|
|
|
|
2022-11-23 04:41:53 -08:00
|
|
|
import { externalHooks } from '@/mixins/externalHooks';
|
2022-12-23 04:37:32 -08:00
|
|
|
import {
|
|
|
|
WAIT_TIME_UNLIMITED,
|
|
|
|
EXECUTIONS_MODAL_KEY,
|
|
|
|
VIEWS,
|
|
|
|
PLACEHOLDER_EMPTY_WORKFLOW_ID,
|
|
|
|
} from '@/constants';
|
2021-08-21 05:11:32 -07:00
|
|
|
|
2022-11-23 04:41:53 -08:00
|
|
|
import { restApi } from '@/mixins/restApi';
|
|
|
|
import { genericHelpers } from '@/mixins/genericHelpers';
|
|
|
|
import { showMessage } from '@/mixins/showMessage';
|
2019-06-23 03:35:23 -07:00
|
|
|
import {
|
|
|
|
IExecutionsCurrentSummaryExtended,
|
|
|
|
IExecutionDeleteFilter,
|
|
|
|
IExecutionsListResponse,
|
|
|
|
IExecutionShortResponse,
|
|
|
|
IExecutionsSummary,
|
|
|
|
IWorkflowShortResponse,
|
|
|
|
} from '@/Interface';
|
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
import { convertToDisplayDate } from '@/utils';
|
2021-09-11 01:15:36 -07:00
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
import { IDataObject } from 'n8n-workflow';
|
2019-06-23 03:35:23 -07:00
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
import { range as _range } from 'lodash';
|
2021-03-10 06:51:18 -08:00
|
|
|
|
2019-06-23 03:35:23 -07:00
|
|
|
import mixins from 'vue-typed-mixins';
|
2022-11-04 06:04:31 -07:00
|
|
|
import { mapStores } from 'pinia';
|
|
|
|
import { useUIStore } from '@/stores/ui';
|
|
|
|
import { useWorkflowsStore } from '@/stores/workflows';
|
2019-06-23 03:35:23 -07:00
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
export default mixins(externalHooks, genericHelpers, restApi, showMessage).extend({
|
2019-06-23 03:35:23 -07:00
|
|
|
name: 'ExecutionsList',
|
|
|
|
components: {
|
2019-07-24 05:25:30 -07:00
|
|
|
ExecutionTime,
|
2019-06-23 03:35:23 -07:00
|
|
|
WorkflowActivator,
|
2022-01-21 09:00:00 -08:00
|
|
|
Modal,
|
2019-06-23 03:35:23 -07:00
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
data() {
|
2019-06-23 03:35:23 -07:00
|
|
|
return {
|
|
|
|
finishedExecutions: [] as IExecutionsSummary[],
|
|
|
|
finishedExecutionsCount: 0,
|
2021-07-06 14:25:25 -07:00
|
|
|
finishedExecutionsCountEstimated: false,
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
checkAll: false,
|
2021-02-09 14:32:40 -08:00
|
|
|
autoRefresh: true,
|
|
|
|
autoRefreshInterval: undefined as undefined | NodeJS.Timer,
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
filter: {
|
2019-12-12 15:39:56 -08:00
|
|
|
status: 'ALL',
|
2019-06-23 03:35:23 -07:00
|
|
|
workflowId: 'ALL',
|
|
|
|
},
|
|
|
|
|
|
|
|
isDataLoading: false,
|
|
|
|
|
|
|
|
requestItemsPerRequest: 10,
|
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
selectedItems: {} as { [key: string]: boolean },
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
stoppingExecutions: [] as string[],
|
|
|
|
workflows: [] as IWorkflowShortResponse[],
|
2022-01-21 09:00:00 -08:00
|
|
|
modalBus: new Vue(),
|
|
|
|
EXECUTIONS_MODAL_KEY,
|
2021-11-10 10:41:40 -08:00
|
|
|
};
|
|
|
|
},
|
2022-01-21 09:00:00 -08:00
|
|
|
async created() {
|
|
|
|
await this.loadWorkflows();
|
|
|
|
await this.refreshData();
|
|
|
|
this.handleAutoRefreshToggle();
|
|
|
|
|
|
|
|
this.$externalHooks().run('executionsList.openDialog');
|
2022-12-14 01:04:10 -08:00
|
|
|
this.$telemetry.track('User opened Executions log', {
|
|
|
|
workflow_id: this.workflowsStore.workflowId,
|
|
|
|
});
|
2022-01-21 09:00:00 -08:00
|
|
|
},
|
|
|
|
beforeDestroy() {
|
|
|
|
if (this.autoRefreshInterval) {
|
|
|
|
clearInterval(this.autoRefreshInterval);
|
|
|
|
this.autoRefreshInterval = undefined;
|
|
|
|
}
|
|
|
|
},
|
2021-11-10 10:41:40 -08:00
|
|
|
computed: {
|
2022-12-14 01:04:10 -08:00
|
|
|
...mapStores(useUIStore, useWorkflowsStore),
|
|
|
|
statuses() {
|
2021-11-10 10:41:40 -08:00
|
|
|
return [
|
2019-12-12 15:39:56 -08:00
|
|
|
{
|
|
|
|
id: 'ALL',
|
2021-12-15 04:16:53 -08:00
|
|
|
name: this.$locale.baseText('executionsList.anyStatus'),
|
2019-12-12 15:39:56 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'error',
|
2021-12-15 04:16:53 -08:00
|
|
|
name: this.$locale.baseText('executionsList.error'),
|
2019-12-12 15:39:56 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'running',
|
2021-12-15 04:16:53 -08:00
|
|
|
name: this.$locale.baseText('executionsList.running'),
|
2019-12-12 15:39:56 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'success',
|
2021-12-15 04:16:53 -08:00
|
|
|
name: this.$locale.baseText('executionsList.success'),
|
2019-12-12 15:39:56 -08:00
|
|
|
},
|
2021-08-21 05:11:32 -07:00
|
|
|
{
|
|
|
|
id: 'waiting',
|
2021-12-15 04:16:53 -08:00
|
|
|
name: this.$locale.baseText('executionsList.waiting'),
|
2021-08-21 05:11:32 -07:00
|
|
|
},
|
2021-11-10 10:41:40 -08:00
|
|
|
];
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
activeExecutions(): IExecutionsCurrentSummaryExtended[] {
|
2022-11-04 06:04:31 -07:00
|
|
|
return this.workflowsStore.activeExecutions;
|
2019-07-24 05:25:30 -07:00
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
combinedExecutions(): IExecutionsSummary[] {
|
2019-06-23 03:35:23 -07:00
|
|
|
const returnData: IExecutionsSummary[] = [];
|
|
|
|
|
2019-12-12 15:39:56 -08:00
|
|
|
if (['ALL', 'running'].includes(this.filter.status)) {
|
|
|
|
returnData.push.apply(returnData, this.activeExecutions);
|
|
|
|
}
|
2021-08-21 05:11:32 -07:00
|
|
|
if (['ALL', 'error', 'success', 'waiting'].includes(this.filter.status)) {
|
2019-12-12 15:39:56 -08:00
|
|
|
returnData.push.apply(returnData, this.finishedExecutions);
|
|
|
|
}
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
return returnData;
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
combinedExecutionsCount(): number {
|
2021-07-06 14:25:25 -07:00
|
|
|
return 0 + this.activeExecutions.length + this.finishedExecutionsCount;
|
2019-06-23 03:35:23 -07:00
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
numSelected(): number {
|
2019-06-23 03:35:23 -07:00
|
|
|
if (this.checkAll === true) {
|
|
|
|
return this.finishedExecutionsCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Object.keys(this.selectedItems).length;
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
isIndeterminate(): boolean {
|
2019-06-23 03:35:23 -07:00
|
|
|
if (this.checkAll === true) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.numSelected > 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
workflowFilterCurrent(): IDataObject {
|
2019-06-23 03:35:23 -07:00
|
|
|
const filter: IDataObject = {};
|
|
|
|
if (this.filter.workflowId !== 'ALL') {
|
|
|
|
filter.workflowId = this.filter.workflowId;
|
|
|
|
}
|
|
|
|
return filter;
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
workflowFilterPast(): IDataObject {
|
2019-12-12 15:39:56 -08:00
|
|
|
const filter: IDataObject = {};
|
|
|
|
if (this.filter.workflowId !== 'ALL') {
|
|
|
|
filter.workflowId = this.filter.workflowId;
|
|
|
|
}
|
2021-08-21 05:11:32 -07:00
|
|
|
if (this.filter.status === 'waiting') {
|
|
|
|
filter.waitTill = true;
|
|
|
|
} else if (['error', 'success'].includes(this.filter.status)) {
|
2019-12-12 15:39:56 -08:00
|
|
|
filter.finished = this.filter.status === 'success';
|
|
|
|
}
|
|
|
|
return filter;
|
|
|
|
},
|
2019-06-23 03:35:23 -07:00
|
|
|
},
|
|
|
|
methods: {
|
2022-01-21 09:00:00 -08:00
|
|
|
closeDialog() {
|
|
|
|
this.modalBus.$emit('close');
|
2019-06-23 03:35:23 -07:00
|
|
|
},
|
2021-09-11 01:15:36 -07:00
|
|
|
convertToDisplayDate,
|
2022-12-22 03:40:33 -08:00
|
|
|
displayExecution(execution: IExecutionsSummary, e: PointerEvent) {
|
2022-12-23 04:37:32 -08:00
|
|
|
if (
|
|
|
|
!this.workflowsStore.workflowId ||
|
|
|
|
this.workflowsStore.workflowId === PLACEHOLDER_EMPTY_WORKFLOW_ID ||
|
|
|
|
execution.workflowId !== this.workflowsStore.workflowId
|
|
|
|
) {
|
|
|
|
const workflowExecutions: IExecutionsSummary[] = this.combinedExecutions.filter(
|
|
|
|
(ex) => ex.workflowId === execution.workflowId,
|
|
|
|
);
|
2022-12-22 03:40:33 -08:00
|
|
|
this.workflowsStore.currentWorkflowExecutions = workflowExecutions;
|
|
|
|
this.workflowsStore.activeWorkflowExecution = execution;
|
|
|
|
}
|
|
|
|
|
2021-12-03 04:59:15 -08:00
|
|
|
if (e.metaKey || e.ctrlKey) {
|
2022-12-14 01:04:10 -08:00
|
|
|
const route = this.$router.resolve({
|
|
|
|
name: VIEWS.EXECUTION_PREVIEW,
|
|
|
|
params: { name: execution.workflowId, executionId: execution.id },
|
|
|
|
});
|
2021-12-03 04:59:15 -08:00
|
|
|
window.open(route.href, '_blank');
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
this.$router
|
|
|
|
.push({
|
|
|
|
name: VIEWS.EXECUTION_PREVIEW,
|
|
|
|
params: { name: execution.workflowId, executionId: execution.id },
|
|
|
|
})
|
|
|
|
.catch(() => {});
|
2022-01-21 09:00:00 -08:00
|
|
|
this.modalBus.$emit('closeAll');
|
2019-06-23 03:35:23 -07:00
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
handleAutoRefreshToggle() {
|
2021-02-09 14:32:40 -08:00
|
|
|
if (this.autoRefreshInterval) {
|
|
|
|
// Clear any previously existing intervals (if any - there shouldn't)
|
|
|
|
clearInterval(this.autoRefreshInterval);
|
|
|
|
this.autoRefreshInterval = undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.autoRefresh) {
|
2022-09-23 07:14:28 -07:00
|
|
|
this.autoRefreshInterval = setInterval(() => this.loadAutoRefresh(), 4 * 1000); // refresh data every 4 secs
|
2021-02-09 14:32:40 -08:00
|
|
|
}
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
handleCheckAllChange() {
|
2019-06-23 03:35:23 -07:00
|
|
|
if (this.checkAll === false) {
|
|
|
|
Vue.set(this, 'selectedItems', {});
|
|
|
|
}
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
handleCheckboxChanged(executionId: string) {
|
2019-06-23 03:35:23 -07:00
|
|
|
if (this.selectedItems[executionId]) {
|
|
|
|
Vue.delete(this.selectedItems, executionId);
|
|
|
|
} else {
|
|
|
|
Vue.set(this.selectedItems, executionId, true);
|
|
|
|
}
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
async handleDeleteSelected() {
|
2021-11-10 10:41:40 -08:00
|
|
|
const deleteExecutions = await this.confirmMessage(
|
2022-12-14 01:04:10 -08:00
|
|
|
this.$locale.baseText('executionsList.confirmMessage.message', {
|
|
|
|
interpolate: { numSelected: this.numSelected.toString() },
|
|
|
|
}),
|
2021-12-15 04:16:53 -08:00
|
|
|
this.$locale.baseText('executionsList.confirmMessage.headline'),
|
2021-11-10 10:41:40 -08:00
|
|
|
'warning',
|
2021-12-15 04:16:53 -08:00
|
|
|
this.$locale.baseText('executionsList.confirmMessage.confirmButtonText'),
|
|
|
|
this.$locale.baseText('executionsList.confirmMessage.cancelButtonText'),
|
2021-11-10 10:41:40 -08:00
|
|
|
);
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
if (deleteExecutions === false) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.isDataLoading = true;
|
|
|
|
|
|
|
|
const sendData: IExecutionDeleteFilter = {};
|
|
|
|
if (this.checkAll === true) {
|
2019-07-22 11:29:06 -07:00
|
|
|
sendData.deleteBefore = this.finishedExecutions[0].startedAt as Date;
|
2019-06-23 03:35:23 -07:00
|
|
|
} else {
|
|
|
|
sendData.ids = Object.keys(this.selectedItems);
|
|
|
|
}
|
|
|
|
|
2019-12-12 15:39:56 -08:00
|
|
|
sendData.filters = this.workflowFilterPast;
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
try {
|
|
|
|
await this.restApi().deleteExecutions(sendData);
|
2022-10-26 01:02:56 -07:00
|
|
|
let removedCurrentlyLoadedExecution = false;
|
|
|
|
let removedActiveExecution = false;
|
2022-11-04 06:04:31 -07:00
|
|
|
const currentWorkflow: string = this.workflowsStore.workflowId;
|
2022-12-14 01:04:10 -08:00
|
|
|
const activeExecution: IExecutionsSummary | null =
|
|
|
|
this.workflowsStore.activeWorkflowExecution;
|
2022-10-26 01:02:56 -07:00
|
|
|
// Also update current workflow executions view if needed
|
|
|
|
for (const selectedId of Object.keys(this.selectedItems)) {
|
2022-12-14 01:04:10 -08:00
|
|
|
const execution: IExecutionsSummary | undefined =
|
|
|
|
this.workflowsStore.getExecutionDataById(selectedId);
|
2022-10-26 01:02:56 -07:00
|
|
|
if (execution && execution.workflowId === currentWorkflow) {
|
2022-11-04 06:04:31 -07:00
|
|
|
this.workflowsStore.deleteExecution(execution);
|
2022-10-26 01:02:56 -07:00
|
|
|
removedCurrentlyLoadedExecution = true;
|
|
|
|
}
|
2022-12-14 01:04:10 -08:00
|
|
|
if (
|
|
|
|
execution !== undefined &&
|
|
|
|
activeExecution !== null &&
|
|
|
|
execution.id === activeExecution.id
|
|
|
|
) {
|
2022-10-26 01:02:56 -07:00
|
|
|
removedActiveExecution = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Also update route if needed
|
|
|
|
if (removedCurrentlyLoadedExecution) {
|
2022-12-14 01:04:10 -08:00
|
|
|
const currentWorkflowExecutions: IExecutionsSummary[] =
|
|
|
|
this.workflowsStore.currentWorkflowExecutions;
|
2022-10-26 01:02:56 -07:00
|
|
|
if (currentWorkflowExecutions.length === 0) {
|
2022-11-04 06:04:31 -07:00
|
|
|
this.workflowsStore.activeWorkflowExecution = null;
|
|
|
|
|
2022-10-26 01:02:56 -07:00
|
|
|
this.$router.push({ name: VIEWS.EXECUTION_HOME, params: { name: currentWorkflow } });
|
|
|
|
} else if (removedActiveExecution) {
|
2022-11-04 06:04:31 -07:00
|
|
|
this.workflowsStore.activeWorkflowExecution = currentWorkflowExecutions[0];
|
2022-12-14 01:04:10 -08:00
|
|
|
this.$router
|
|
|
|
.push({
|
|
|
|
name: VIEWS.EXECUTION_PREVIEW,
|
|
|
|
params: { name: currentWorkflow, executionId: currentWorkflowExecutions[0].id },
|
|
|
|
})
|
|
|
|
.catch(() => {});
|
2022-10-26 01:02:56 -07:00
|
|
|
}
|
|
|
|
}
|
2019-06-23 03:35:23 -07:00
|
|
|
} catch (error) {
|
|
|
|
this.isDataLoading = false;
|
2021-11-10 10:41:40 -08:00
|
|
|
this.$showError(
|
|
|
|
error,
|
2021-12-15 04:16:53 -08:00
|
|
|
this.$locale.baseText('executionsList.showError.handleDeleteSelected.title'),
|
2021-11-10 10:41:40 -08:00
|
|
|
);
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.isDataLoading = false;
|
|
|
|
|
|
|
|
this.$showMessage({
|
2021-12-15 04:16:53 -08:00
|
|
|
title: this.$locale.baseText('executionsList.showMessage.handleDeleteSelected.title'),
|
2019-06-23 03:35:23 -07:00
|
|
|
type: 'success',
|
|
|
|
});
|
|
|
|
|
|
|
|
Vue.set(this, 'selectedItems', {});
|
|
|
|
this.checkAll = false;
|
|
|
|
|
|
|
|
this.refreshData();
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
handleFilterChanged() {
|
2019-06-23 03:35:23 -07:00
|
|
|
this.refreshData();
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
handleRetryClick(commandData: { command: string; row: IExecutionShortResponse }) {
|
2019-12-12 16:12:38 -08:00
|
|
|
let loadWorkflow = false;
|
|
|
|
if (commandData.command === 'currentlySaved') {
|
|
|
|
loadWorkflow = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.retryExecution(commandData.row, loadWorkflow);
|
2022-07-09 23:53:04 -07:00
|
|
|
|
|
|
|
this.$telemetry.track('User clicked retry execution button', {
|
2022-11-04 06:04:31 -07:00
|
|
|
workflow_id: this.workflowsStore.workflowId,
|
2022-07-09 23:53:04 -07:00
|
|
|
execution_id: commandData.row.id,
|
|
|
|
retry_type: loadWorkflow ? 'current' : 'original',
|
|
|
|
});
|
2019-12-12 16:12:38 -08:00
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
getRowClass(data: IDataObject): string {
|
2019-07-24 05:25:30 -07:00
|
|
|
const classes: string[] = [];
|
2019-06-23 03:35:23 -07:00
|
|
|
if ((data.row as IExecutionsSummary).stoppedAt === undefined) {
|
|
|
|
classes.push('currently-running');
|
|
|
|
}
|
|
|
|
|
|
|
|
return classes.join(' ');
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
getWorkflowName(workflowId: string): string | undefined {
|
2019-06-23 03:35:23 -07:00
|
|
|
const workflow = this.workflows.find((data) => data.id === workflowId);
|
|
|
|
if (workflow === undefined) {
|
2019-07-24 05:25:30 -07:00
|
|
|
return undefined;
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return workflow.name;
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
async loadActiveExecutions(): Promise<void> {
|
|
|
|
const activeExecutions = await this.restApi().getCurrentExecutions(
|
|
|
|
this.workflowFilterCurrent,
|
|
|
|
);
|
2019-07-24 05:25:30 -07:00
|
|
|
for (const activeExecution of activeExecutions) {
|
2022-12-14 01:04:10 -08:00
|
|
|
if (
|
|
|
|
activeExecution.workflowId !== undefined &&
|
|
|
|
activeExecution.workflowName === undefined
|
|
|
|
) {
|
2019-07-24 05:25:30 -07:00
|
|
|
activeExecution.workflowName = this.getWorkflowName(activeExecution.workflowId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-04 06:04:31 -07:00
|
|
|
this.workflowsStore.activeExecutions = activeExecutions;
|
2022-12-19 05:36:25 -08:00
|
|
|
this.workflowsStore.addToCurrentExecutions(activeExecutions);
|
2019-06-23 03:35:23 -07:00
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
async loadAutoRefresh(): Promise<void> {
|
2021-02-17 15:09:39 -08:00
|
|
|
const filter = this.workflowFilterPast;
|
2021-03-02 23:31:55 -08:00
|
|
|
// We cannot use firstId here as some executions finish out of order. Let's say
|
|
|
|
// You have execution ids 500 to 505 running.
|
|
|
|
// Suppose 504 finishes before 500, 501, 502 and 503.
|
|
|
|
// iF you use firstId, filtering id >= 504 you won't
|
|
|
|
// ever get ids 500, 501, 502 and 503 when they finish
|
2022-12-14 01:04:10 -08:00
|
|
|
const pastExecutionsPromise: Promise<IExecutionsListResponse> =
|
|
|
|
this.restApi().getPastExecutions(filter, 30);
|
|
|
|
const currentExecutionsPromise: Promise<IExecutionsCurrentSummaryExtended[]> =
|
|
|
|
this.restApi().getCurrentExecutions({});
|
2021-02-09 14:32:40 -08:00
|
|
|
|
2021-02-17 15:09:39 -08:00
|
|
|
const results = await Promise.all([pastExecutionsPromise, currentExecutionsPromise]);
|
2021-02-09 14:32:40 -08:00
|
|
|
|
|
|
|
for (const activeExecution of results[1]) {
|
2022-12-14 01:04:10 -08:00
|
|
|
if (
|
|
|
|
activeExecution.workflowId !== undefined &&
|
|
|
|
activeExecution.workflowName === undefined
|
|
|
|
) {
|
2021-02-09 14:32:40 -08:00
|
|
|
activeExecution.workflowName = this.getWorkflowName(activeExecution.workflowId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-04 06:04:31 -07:00
|
|
|
this.workflowsStore.activeExecutions = results[1];
|
2021-02-09 14:32:40 -08:00
|
|
|
|
2021-05-12 08:51:54 -07:00
|
|
|
// execution IDs are typed as string, int conversion is necessary so we can order.
|
2022-12-14 01:04:10 -08:00
|
|
|
const alreadyPresentExecutionIds = this.finishedExecutions.map((exec) =>
|
|
|
|
parseInt(exec.id, 10),
|
|
|
|
);
|
2021-03-10 06:51:18 -08:00
|
|
|
let lastId = 0;
|
|
|
|
const gaps = [] as number[];
|
2022-12-14 01:04:10 -08:00
|
|
|
for (let i = results[0].results.length - 1; i >= 0; i--) {
|
2021-03-02 23:31:55 -08:00
|
|
|
const currentItem = results[0].results[i];
|
2021-03-10 06:51:18 -08:00
|
|
|
const currentId = parseInt(currentItem.id, 10);
|
|
|
|
if (lastId !== 0 && isNaN(currentId) === false) {
|
|
|
|
// We are doing this iteration to detect possible gaps.
|
|
|
|
// The gaps are used to remove executions that finished
|
|
|
|
// and were deleted from database but were displaying
|
|
|
|
// in this list while running.
|
|
|
|
if (currentId - lastId > 1) {
|
|
|
|
// We have some gaps.
|
|
|
|
const range = _range(lastId + 1, currentId);
|
|
|
|
gaps.push(...range);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
lastId = parseInt(currentItem.id, 10) || 0;
|
|
|
|
|
2021-03-02 23:31:55 -08:00
|
|
|
// Check new results from end to start
|
|
|
|
// Add new items accordingly.
|
2021-05-12 08:51:54 -07:00
|
|
|
const executionIndex = alreadyPresentExecutionIds.indexOf(currentId);
|
2021-03-02 23:31:55 -08:00
|
|
|
if (executionIndex !== -1) {
|
|
|
|
// Execution that we received is already present.
|
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
if (
|
|
|
|
this.finishedExecutions[executionIndex].finished === false &&
|
|
|
|
currentItem.finished === true
|
|
|
|
) {
|
2021-03-02 23:31:55 -08:00
|
|
|
// Concurrency stuff. This might happen if the execution finishes
|
|
|
|
// prior to saving all information to database. Somewhat rare but
|
|
|
|
// With auto refresh and several executions, it happens sometimes.
|
|
|
|
// So we replace the execution data so it displays correctly.
|
|
|
|
this.finishedExecutions[executionIndex] = currentItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the correct position to place this newcomer
|
|
|
|
let j;
|
|
|
|
for (j = this.finishedExecutions.length - 1; j >= 0; j--) {
|
2021-05-12 08:51:54 -07:00
|
|
|
if (currentId < parseInt(this.finishedExecutions[j].id, 10)) {
|
2021-03-02 23:31:55 -08:00
|
|
|
this.finishedExecutions.splice(j + 1, 0, currentItem);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (j === -1) {
|
|
|
|
this.finishedExecutions.unshift(currentItem);
|
|
|
|
}
|
|
|
|
}
|
2022-12-14 01:04:10 -08:00
|
|
|
this.finishedExecutions = this.finishedExecutions.filter(
|
|
|
|
(execution) =>
|
|
|
|
!gaps.includes(parseInt(execution.id, 10)) && lastId >= parseInt(execution.id, 10),
|
|
|
|
);
|
2021-02-09 14:32:40 -08:00
|
|
|
this.finishedExecutionsCount = results[0].count;
|
2021-07-06 14:25:25 -07:00
|
|
|
this.finishedExecutionsCountEstimated = results[0].estimated;
|
2022-12-19 05:36:25 -08:00
|
|
|
this.workflowsStore.addToCurrentExecutions(this.finishedExecutions);
|
2021-02-09 14:32:40 -08:00
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
async loadFinishedExecutions(): Promise<void> {
|
2019-12-12 15:39:56 -08:00
|
|
|
if (this.filter.status === 'running') {
|
|
|
|
this.finishedExecutions = [];
|
|
|
|
this.finishedExecutionsCount = 0;
|
2021-07-06 14:25:25 -07:00
|
|
|
this.finishedExecutionsCountEstimated = false;
|
2019-12-12 15:39:56 -08:00
|
|
|
return;
|
|
|
|
}
|
2022-12-14 01:04:10 -08:00
|
|
|
const data = await this.restApi().getPastExecutions(
|
|
|
|
this.workflowFilterPast,
|
|
|
|
this.requestItemsPerRequest,
|
|
|
|
);
|
2021-12-02 00:43:59 -08:00
|
|
|
this.finishedExecutions = data.results;
|
2019-06-23 03:35:23 -07:00
|
|
|
this.finishedExecutionsCount = data.count;
|
2021-07-06 14:25:25 -07:00
|
|
|
this.finishedExecutionsCountEstimated = data.estimated;
|
2022-12-19 05:36:25 -08:00
|
|
|
|
|
|
|
this.workflowsStore.addToCurrentExecutions(data.results);
|
2019-06-23 03:35:23 -07:00
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
async loadMore() {
|
2019-12-12 15:39:56 -08:00
|
|
|
if (this.filter.status === 'running') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-06-23 03:35:23 -07:00
|
|
|
this.isDataLoading = true;
|
|
|
|
|
2019-12-12 15:39:56 -08:00
|
|
|
const filter = this.workflowFilterPast;
|
2023-01-02 08:42:32 -08:00
|
|
|
let lastId: string | undefined;
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
if (this.finishedExecutions.length !== 0) {
|
|
|
|
const lastItem = this.finishedExecutions.slice(-1)[0];
|
2019-12-12 11:57:11 -08:00
|
|
|
lastId = lastItem.id;
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
let data: IExecutionsListResponse;
|
|
|
|
try {
|
2019-12-12 11:57:11 -08:00
|
|
|
data = await this.restApi().getPastExecutions(filter, this.requestItemsPerRequest, lastId);
|
2019-06-23 03:35:23 -07:00
|
|
|
} catch (error) {
|
|
|
|
this.isDataLoading = false;
|
2022-12-14 01:04:10 -08:00
|
|
|
this.$showError(error, this.$locale.baseText('executionsList.showError.loadMore.title'));
|
2019-06-23 03:35:23 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-11-10 10:41:40 -08:00
|
|
|
data.results = data.results.map((execution) => {
|
|
|
|
// @ts-ignore
|
2021-11-20 09:28:05 -08:00
|
|
|
return { ...execution, mode: execution.mode };
|
2021-11-10 10:41:40 -08:00
|
|
|
});
|
|
|
|
|
2019-06-23 03:35:23 -07:00
|
|
|
this.finishedExecutions.push.apply(this.finishedExecutions, data.results);
|
|
|
|
this.finishedExecutionsCount = data.count;
|
2021-07-06 14:25:25 -07:00
|
|
|
this.finishedExecutionsCountEstimated = data.estimated;
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
this.isDataLoading = false;
|
2022-12-19 05:36:25 -08:00
|
|
|
|
|
|
|
this.workflowsStore.addToCurrentExecutions(data.results);
|
2019-06-23 03:35:23 -07:00
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
async loadWorkflows() {
|
2019-06-23 03:35:23 -07:00
|
|
|
try {
|
|
|
|
const workflows = await this.restApi().getWorkflows();
|
|
|
|
workflows.sort((a, b) => {
|
|
|
|
if (a.name.toLowerCase() < b.name.toLowerCase()) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (a.name.toLowerCase() > b.name.toLowerCase()) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
});
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
workflows.unshift({
|
|
|
|
id: 'ALL',
|
2021-12-15 04:16:53 -08:00
|
|
|
name: this.$locale.baseText('executionsList.allWorkflows'),
|
2019-06-23 03:35:23 -07:00
|
|
|
});
|
|
|
|
|
|
|
|
Vue.set(this, 'workflows', workflows);
|
|
|
|
} catch (error) {
|
2021-11-10 10:41:40 -08:00
|
|
|
this.$showError(
|
|
|
|
error,
|
2021-12-15 04:16:53 -08:00
|
|
|
this.$locale.baseText('executionsList.showError.loadWorkflows.title'),
|
2021-11-10 10:41:40 -08:00
|
|
|
);
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
async retryExecution(execution: IExecutionShortResponse, loadWorkflow?: boolean) {
|
2019-06-23 03:35:23 -07:00
|
|
|
this.isDataLoading = true;
|
|
|
|
|
|
|
|
try {
|
2019-12-12 16:12:38 -08:00
|
|
|
const retrySuccessful = await this.restApi().retryExecution(execution.id, loadWorkflow);
|
2019-06-23 03:35:23 -07:00
|
|
|
|
2019-08-08 22:37:10 -07:00
|
|
|
if (retrySuccessful === true) {
|
2019-06-23 03:35:23 -07:00
|
|
|
this.$showMessage({
|
2021-12-15 04:16:53 -08:00
|
|
|
title: this.$locale.baseText('executionsList.showMessage.retrySuccessfulTrue.title'),
|
2019-06-23 03:35:23 -07:00
|
|
|
type: 'success',
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
this.$showMessage({
|
2021-12-15 04:16:53 -08:00
|
|
|
title: this.$locale.baseText('executionsList.showMessage.retrySuccessfulFalse.title'),
|
2019-06-23 03:35:23 -07:00
|
|
|
type: 'error',
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
this.isDataLoading = false;
|
|
|
|
} catch (error) {
|
2021-11-10 10:41:40 -08:00
|
|
|
this.$showError(
|
|
|
|
error,
|
2021-12-15 04:16:53 -08:00
|
|
|
this.$locale.baseText('executionsList.showError.retryExecution.title'),
|
2021-11-10 10:41:40 -08:00
|
|
|
);
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
this.isDataLoading = false;
|
|
|
|
}
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
async refreshData() {
|
2019-06-23 03:35:23 -07:00
|
|
|
this.isDataLoading = true;
|
|
|
|
|
|
|
|
try {
|
|
|
|
const activeExecutionsPromise = this.loadActiveExecutions();
|
|
|
|
const finishedExecutionsPromise = this.loadFinishedExecutions();
|
|
|
|
await Promise.all([activeExecutionsPromise, finishedExecutionsPromise]);
|
|
|
|
} catch (error) {
|
2022-12-14 01:04:10 -08:00
|
|
|
this.$showError(error, this.$locale.baseText('executionsList.showError.refreshData.title'));
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
this.isDataLoading = false;
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
statusTooltipText(entry: IExecutionsSummary): string {
|
2021-08-21 05:11:32 -07:00
|
|
|
if (entry.waitTill) {
|
|
|
|
const waitDate = new Date(entry.waitTill);
|
|
|
|
if (waitDate.toISOString() === WAIT_TIME_UNLIMITED) {
|
2022-12-14 01:04:10 -08:00
|
|
|
return this.$locale.baseText(
|
|
|
|
'executionsList.statusTooltipText.theWorkflowIsWaitingIndefinitely',
|
|
|
|
);
|
2021-08-21 05:11:32 -07:00
|
|
|
}
|
2021-11-10 10:41:40 -08:00
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
return this.$locale.baseText('executionsList.statusTooltipText.theWorkflowIsWaitingTill', {
|
|
|
|
interpolate: {
|
|
|
|
waitDateDate: waitDate.toLocaleDateString(),
|
|
|
|
waitDateTime: waitDate.toLocaleTimeString(),
|
2021-11-10 10:41:40 -08:00
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
});
|
2021-08-21 05:11:32 -07:00
|
|
|
} else if (entry.stoppedAt === undefined) {
|
2022-12-14 01:04:10 -08:00
|
|
|
return this.$locale.baseText(
|
|
|
|
'executionsList.statusTooltipText.theWorkflowIsCurrentlyExecuting',
|
|
|
|
);
|
2019-08-08 11:24:37 -07:00
|
|
|
} else if (entry.finished === true && entry.retryOf !== undefined) {
|
2021-12-15 04:16:53 -08:00
|
|
|
return this.$locale.baseText(
|
2021-11-10 10:41:40 -08:00
|
|
|
'executionsList.statusTooltipText.theWorkflowExecutionWasARetryOfAndItWasSuccessful',
|
2022-12-14 01:04:10 -08:00
|
|
|
{ interpolate: { entryRetryOf: entry.retryOf } },
|
2021-11-10 10:41:40 -08:00
|
|
|
);
|
2019-06-23 03:35:23 -07:00
|
|
|
} else if (entry.finished === true) {
|
2022-12-14 01:04:10 -08:00
|
|
|
return this.$locale.baseText(
|
|
|
|
'executionsList.statusTooltipText.theWorkflowExecutionWasSuccessful',
|
|
|
|
);
|
2019-06-23 03:35:23 -07:00
|
|
|
} else if (entry.retryOf !== undefined) {
|
2021-12-15 04:16:53 -08:00
|
|
|
return this.$locale.baseText(
|
2021-11-10 10:41:40 -08:00
|
|
|
'executionsList.statusTooltipText.theWorkflowExecutionWasARetryOfAndFailed',
|
2022-12-14 01:04:10 -08:00
|
|
|
{ interpolate: { entryRetryOf: entry.retryOf } },
|
2021-11-10 10:41:40 -08:00
|
|
|
);
|
2019-06-23 03:35:23 -07:00
|
|
|
} else if (entry.retrySuccessId !== undefined) {
|
2021-12-15 04:16:53 -08:00
|
|
|
return this.$locale.baseText(
|
2021-11-10 10:41:40 -08:00
|
|
|
'executionsList.statusTooltipText.theWorkflowExecutionFailedButTheRetryWasSuccessful',
|
2022-12-14 01:04:10 -08:00
|
|
|
{ interpolate: { entryRetrySuccessId: entry.retrySuccessId } },
|
2021-11-10 10:41:40 -08:00
|
|
|
);
|
2021-03-02 23:31:55 -08:00
|
|
|
} else if (entry.stoppedAt === null) {
|
2022-12-14 01:04:10 -08:00
|
|
|
return this.$locale.baseText(
|
|
|
|
'executionsList.statusTooltipText.theWorkflowExecutionIsProbablyStillRunning',
|
|
|
|
);
|
2019-06-23 03:35:23 -07:00
|
|
|
} else {
|
2021-12-15 04:16:53 -08:00
|
|
|
return this.$locale.baseText('executionsList.statusTooltipText.theWorkflowExecutionFailed');
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
},
|
2022-12-14 01:04:10 -08:00
|
|
|
async stopExecution(activeExecutionId: string) {
|
2019-06-23 03:35:23 -07:00
|
|
|
try {
|
|
|
|
// Add it to the list of currently stopping executions that we
|
|
|
|
// can show the user in the UI that it is in progress
|
2019-07-24 05:25:30 -07:00
|
|
|
this.stoppingExecutions.push(activeExecutionId);
|
2019-06-23 03:35:23 -07:00
|
|
|
|
2021-07-23 08:50:47 -07:00
|
|
|
await this.restApi().stopCurrentExecution(activeExecutionId);
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
// Remove it from the list of currently stopping executions
|
2019-07-24 05:25:30 -07:00
|
|
|
const index = this.stoppingExecutions.indexOf(activeExecutionId);
|
2019-06-23 03:35:23 -07:00
|
|
|
this.stoppingExecutions.splice(index, 1);
|
|
|
|
|
|
|
|
this.$showMessage({
|
2021-12-15 04:16:53 -08:00
|
|
|
title: this.$locale.baseText('executionsList.showMessage.stopExecution.title'),
|
2022-12-14 01:04:10 -08:00
|
|
|
message: this.$locale.baseText('executionsList.showMessage.stopExecution.message', {
|
|
|
|
interpolate: { activeExecutionId },
|
|
|
|
}),
|
2019-06-23 03:35:23 -07:00
|
|
|
type: 'success',
|
|
|
|
});
|
|
|
|
|
|
|
|
this.refreshData();
|
|
|
|
} catch (error) {
|
2021-11-10 10:41:40 -08:00
|
|
|
this.$showError(
|
|
|
|
error,
|
2021-12-15 04:16:53 -08:00
|
|
|
this.$locale.baseText('executionsList.showError.stopExecution.title'),
|
2021-11-10 10:41:40 -08:00
|
|
|
);
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style scoped lang="scss">
|
2021-02-09 14:32:40 -08:00
|
|
|
.autorefresh {
|
|
|
|
padding-right: 0.5em;
|
|
|
|
text-align: right;
|
|
|
|
}
|
|
|
|
|
2021-08-21 05:11:32 -07:00
|
|
|
.execution-actions {
|
|
|
|
button {
|
|
|
|
margin: 0 0.25em;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-23 03:35:23 -07:00
|
|
|
.filters {
|
|
|
|
line-height: 2em;
|
|
|
|
.refresh-button {
|
|
|
|
position: absolute;
|
|
|
|
right: 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.load-more {
|
|
|
|
margin: 2em 0 0 0;
|
|
|
|
width: 100%;
|
|
|
|
text-align: center;
|
|
|
|
}
|
|
|
|
|
|
|
|
.selection-options {
|
|
|
|
height: 2em;
|
|
|
|
}
|
|
|
|
|
|
|
|
.status-badge {
|
|
|
|
position: relative;
|
|
|
|
display: inline-block;
|
|
|
|
padding: 0 10px;
|
2021-08-29 04:36:17 -07:00
|
|
|
line-height: 22.6px;
|
2019-06-23 03:35:23 -07:00
|
|
|
border-radius: 15px;
|
|
|
|
text-align: center;
|
2021-10-27 12:55:37 -07:00
|
|
|
font-size: var(--font-size-s);
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
&.error {
|
2021-08-29 04:36:17 -07:00
|
|
|
background-color: var(--color-danger-tint-1);
|
|
|
|
color: var(--color-danger);
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
&.success {
|
2021-08-29 04:36:17 -07:00
|
|
|
background-color: var(--color-success-tint-1);
|
|
|
|
color: var(--color-success);
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
2021-03-02 23:31:55 -08:00
|
|
|
|
2022-12-14 01:04:10 -08:00
|
|
|
&.running,
|
|
|
|
&.warning {
|
2021-08-29 04:36:17 -07:00
|
|
|
background-color: var(--color-warning-tint-2);
|
|
|
|
color: var(--color-warning);
|
2021-03-02 23:31:55 -08:00
|
|
|
}
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
.workflow-name {
|
|
|
|
font-weight: bold;
|
|
|
|
}
|
|
|
|
|
2021-08-29 04:36:17 -07:00
|
|
|
.actions-container > * {
|
|
|
|
margin-left: 5px;
|
|
|
|
}
|
2019-06-23 03:35:23 -07:00
|
|
|
</style>
|
|
|
|
|
|
|
|
<style lang="scss">
|
|
|
|
.currently-running {
|
2022-07-26 03:45:55 -07:00
|
|
|
background-color: var(--color-primary-tint-3) !important;
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
.el-table tr:hover.currently-running td {
|
2022-07-26 03:45:55 -07:00
|
|
|
background-color: var(--color-primary-tint-2) !important;
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
</style>
|