mirror of
https://github.com/n8n-io/n8n.git
synced 2025-03-05 20:50:17 -08:00
fix(editor): Turn off executions list auto-refresh after leaving the page (#8005)
## Summary Fixes the bug when users leave the executions page but there is still an ongoing request for executions. --------- Co-authored-by: Alex Grozav <alex@grozav.com>
This commit is contained in:
parent
b29b4d442b
commit
e3c363d72c
|
@ -1,18 +1,20 @@
|
||||||
import { WorkflowPage } from '../pages';
|
import { WorkflowPage } from '../pages';
|
||||||
import { WorkflowExecutionsTab } from '../pages/workflow-executions-tab';
|
import { WorkflowExecutionsTab } from '../pages/workflow-executions-tab';
|
||||||
|
import type { RouteHandler } from 'cypress/types/net-stubbing';
|
||||||
|
|
||||||
const workflowPage = new WorkflowPage();
|
const workflowPage = new WorkflowPage();
|
||||||
const executionsTab = new WorkflowExecutionsTab();
|
const executionsTab = new WorkflowExecutionsTab();
|
||||||
|
const executionsRefreshInterval = 4000;
|
||||||
|
|
||||||
// Test suite for executions tab
|
// Test suite for executions tab
|
||||||
describe('Current Workflow Executions', () => {
|
describe('Current Workflow Executions', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
workflowPage.actions.visit();
|
workflowPage.actions.visit();
|
||||||
cy.createFixtureWorkflow('Test_workflow_4_executions_view.json', `My test workflow`);
|
cy.createFixtureWorkflow('Test_workflow_4_executions_view.json', `My test workflow`);
|
||||||
createMockExecutions();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should render executions tab correctly', () => {
|
it('should render executions tab correctly', () => {
|
||||||
|
createMockExecutions();
|
||||||
cy.intercept('GET', '/rest/executions?filter=*').as('getExecutions');
|
cy.intercept('GET', '/rest/executions?filter=*').as('getExecutions');
|
||||||
cy.intercept('GET', '/rest/executions-current?filter=*').as('getCurrentExecutions');
|
cy.intercept('GET', '/rest/executions-current?filter=*').as('getCurrentExecutions');
|
||||||
|
|
||||||
|
@ -29,6 +31,45 @@ describe('Current Workflow Executions', () => {
|
||||||
.invoke('attr', 'class')
|
.invoke('attr', 'class')
|
||||||
.should('match', /_active_/);
|
.should('match', /_active_/);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should not redirect back to execution tab when request is not done before leaving the page', () => {
|
||||||
|
cy.intercept('GET', '/rest/executions?filter=*');
|
||||||
|
cy.intercept('GET', '/rest/executions-current?filter=*');
|
||||||
|
|
||||||
|
executionsTab.actions.switchToExecutionsTab();
|
||||||
|
executionsTab.actions.switchToEditorTab();
|
||||||
|
cy.wait(executionsRefreshInterval);
|
||||||
|
cy.url().should('not.include', '/executions');
|
||||||
|
executionsTab.actions.switchToExecutionsTab();
|
||||||
|
executionsTab.actions.switchToEditorTab();
|
||||||
|
executionsTab.actions.switchToExecutionsTab();
|
||||||
|
executionsTab.actions.switchToEditorTab();
|
||||||
|
executionsTab.actions.switchToExecutionsTab();
|
||||||
|
executionsTab.actions.switchToEditorTab();
|
||||||
|
cy.wait(executionsRefreshInterval);
|
||||||
|
cy.url().should('not.include', '/executions');
|
||||||
|
executionsTab.actions.switchToExecutionsTab();
|
||||||
|
cy.wait(1000);
|
||||||
|
executionsTab.actions.switchToEditorTab();
|
||||||
|
cy.wait(executionsRefreshInterval);
|
||||||
|
cy.url().should('not.include', '/executions');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not redirect back to execution tab when slow request is not done before leaving the page', () => {
|
||||||
|
const throttleResponse: RouteHandler = (req) => {
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
setTimeout(() => resolve(req.continue()), 2000);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
cy.intercept('GET', '/rest/executions?filter=*', throttleResponse);
|
||||||
|
cy.intercept('GET', '/rest/executions-current?filter=*', throttleResponse);
|
||||||
|
|
||||||
|
executionsTab.actions.switchToExecutionsTab();
|
||||||
|
executionsTab.actions.switchToEditorTab();
|
||||||
|
cy.wait(executionsRefreshInterval);
|
||||||
|
cy.url().should('not.include', '/executions');
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
const createMockExecutions = () => {
|
const createMockExecutions = () => {
|
||||||
|
|
|
@ -39,9 +39,11 @@ export class WorkflowExecutionsTab extends BasePage {
|
||||||
},
|
},
|
||||||
switchToExecutionsTab: () => {
|
switchToExecutionsTab: () => {
|
||||||
this.getters.executionsTabButton().click();
|
this.getters.executionsTabButton().click();
|
||||||
|
cy.url().should('include', '/executions');
|
||||||
},
|
},
|
||||||
switchToEditorTab: () => {
|
switchToEditorTab: () => {
|
||||||
workflowPage.getters.editorTabButton().click();
|
workflowPage.getters.editorTabButton().click();
|
||||||
|
cy.url().should('match', /\/workflow\/[^\/]+$/);
|
||||||
},
|
},
|
||||||
deleteExecutionInPreview: () => {
|
deleteExecutionInPreview: () => {
|
||||||
this.getters.executionPreviewDeleteButton().click();
|
this.getters.executionPreviewDeleteButton().click();
|
||||||
|
|
|
@ -310,6 +310,7 @@ import { isEmpty } from '@/utils/typesUtils';
|
||||||
import { setPageTitle } from '@/utils/htmlUtils';
|
import { setPageTitle } from '@/utils/htmlUtils';
|
||||||
import { executionFilterToQueryFilter } from '@/utils/executionUtils';
|
import { executionFilterToQueryFilter } from '@/utils/executionUtils';
|
||||||
import { useExternalHooks } from '@/composables/useExternalHooks';
|
import { useExternalHooks } from '@/composables/useExternalHooks';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: 'ExecutionsList',
|
name: 'ExecutionsList',
|
||||||
|
@ -328,11 +329,13 @@ export default defineComponent({
|
||||||
const i18n = useI18n();
|
const i18n = useI18n();
|
||||||
const telemetry = useTelemetry();
|
const telemetry = useTelemetry();
|
||||||
const externalHooks = useExternalHooks();
|
const externalHooks = useExternalHooks();
|
||||||
|
const route = useRoute();
|
||||||
|
|
||||||
return {
|
return {
|
||||||
i18n,
|
i18n,
|
||||||
telemetry,
|
telemetry,
|
||||||
externalHooks,
|
externalHooks,
|
||||||
|
route,
|
||||||
...useToast(),
|
...useToast(),
|
||||||
...useMessage(),
|
...useMessage(),
|
||||||
};
|
};
|
||||||
|
@ -346,7 +349,7 @@ export default defineComponent({
|
||||||
|
|
||||||
allVisibleSelected: false,
|
allVisibleSelected: false,
|
||||||
allExistingSelected: false,
|
allExistingSelected: false,
|
||||||
autoRefresh: this.autoRefreshEnabled,
|
autoRefresh: false,
|
||||||
autoRefreshTimeout: undefined as undefined | NodeJS.Timer,
|
autoRefreshTimeout: undefined as undefined | NodeJS.Timer,
|
||||||
|
|
||||||
filter: {} as ExecutionFilterType,
|
filter: {} as ExecutionFilterType,
|
||||||
|
@ -361,6 +364,9 @@ export default defineComponent({
|
||||||
workflows: [] as IWorkflowShortResponse[],
|
workflows: [] as IWorkflowShortResponse[],
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
created() {
|
||||||
|
this.autoRefresh = this.autoRefreshEnabled;
|
||||||
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
setPageTitle(`n8n - ${this.pageTitle}`);
|
setPageTitle(`n8n - ${this.pageTitle}`);
|
||||||
|
|
||||||
|
@ -376,6 +382,7 @@ export default defineComponent({
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
beforeUnmount() {
|
beforeUnmount() {
|
||||||
|
this.autoRefresh = false;
|
||||||
this.stopAutoRefreshInterval();
|
this.stopAutoRefreshInterval();
|
||||||
document.removeEventListener('visibilitychange', this.onDocumentVisibilityChange);
|
document.removeEventListener('visibilitychange', this.onDocumentVisibilityChange);
|
||||||
},
|
},
|
||||||
|
@ -938,7 +945,7 @@ export default defineComponent({
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
async startAutoRefreshInterval() {
|
async startAutoRefreshInterval() {
|
||||||
if (this.autoRefresh) {
|
if (this.autoRefresh && this.route.name === VIEWS.WORKFLOW_EXECUTIONS) {
|
||||||
await this.loadAutoRefresh();
|
await this.loadAutoRefresh();
|
||||||
this.autoRefreshTimeout = setTimeout(() => {
|
this.autoRefreshTimeout = setTimeout(() => {
|
||||||
void this.startAutoRefreshInterval();
|
void this.startAutoRefreshInterval();
|
||||||
|
@ -946,10 +953,8 @@ export default defineComponent({
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
stopAutoRefreshInterval() {
|
stopAutoRefreshInterval() {
|
||||||
if (this.autoRefreshTimeout) {
|
clearTimeout(this.autoRefreshTimeout);
|
||||||
clearTimeout(this.autoRefreshTimeout);
|
this.autoRefreshTimeout = undefined;
|
||||||
this.autoRefreshTimeout = undefined;
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
onDocumentVisibilityChange() {
|
onDocumentVisibilityChange() {
|
||||||
if (document.visibilityState === 'hidden') {
|
if (document.visibilityState === 'hidden') {
|
||||||
|
|
|
@ -145,7 +145,6 @@ export default defineComponent({
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
async beforeRouteLeave(to, from, next) {
|
async beforeRouteLeave(to, from, next) {
|
||||||
this.stopAutoRefreshInterval();
|
|
||||||
if (getNodeViewTab(to) === MAIN_HEADER_TABS.WORKFLOW) {
|
if (getNodeViewTab(to) === MAIN_HEADER_TABS.WORKFLOW) {
|
||||||
next();
|
next();
|
||||||
return;
|
return;
|
||||||
|
@ -181,6 +180,9 @@ export default defineComponent({
|
||||||
next();
|
next();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
created() {
|
||||||
|
this.autoRefresh = this.uiStore.executionSidebarAutoRefresh;
|
||||||
|
},
|
||||||
async mounted() {
|
async mounted() {
|
||||||
this.loading = true;
|
this.loading = true;
|
||||||
const workflowUpdated = this.$route.params.name !== this.workflowsStore.workflowId;
|
const workflowUpdated = this.$route.params.name !== this.workflowsStore.workflowId;
|
||||||
|
@ -198,16 +200,15 @@ export default defineComponent({
|
||||||
await this.setExecutions();
|
await this.setExecutions();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.autoRefresh = this.uiStore.executionSidebarAutoRefresh;
|
|
||||||
void this.startAutoRefreshInterval();
|
void this.startAutoRefreshInterval();
|
||||||
document.addEventListener('visibilitychange', this.onDocumentVisibilityChange);
|
document.addEventListener('visibilitychange', this.onDocumentVisibilityChange);
|
||||||
|
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
},
|
},
|
||||||
beforeUnmount() {
|
beforeUnmount() {
|
||||||
this.stopAutoRefreshInterval();
|
|
||||||
document.removeEventListener('visibilitychange', this.onDocumentVisibilityChange);
|
document.removeEventListener('visibilitychange', this.onDocumentVisibilityChange);
|
||||||
|
this.autoRefresh = false;
|
||||||
|
this.stopAutoRefreshInterval();
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
async initView(loadWorkflow: boolean): Promise<void> {
|
async initView(loadWorkflow: boolean): Promise<void> {
|
||||||
|
@ -292,7 +293,7 @@ export default defineComponent({
|
||||||
}
|
}
|
||||||
if (this.executions.length > 0) {
|
if (this.executions.length > 0) {
|
||||||
await this.$router
|
await this.$router
|
||||||
.push({
|
.replace({
|
||||||
name: VIEWS.EXECUTION_PREVIEW,
|
name: VIEWS.EXECUTION_PREVIEW,
|
||||||
params: { name: this.currentWorkflow, executionId: nextExecution.id },
|
params: { name: this.currentWorkflow, executionId: nextExecution.id },
|
||||||
})
|
})
|
||||||
|
@ -302,7 +303,7 @@ export default defineComponent({
|
||||||
} else {
|
} else {
|
||||||
// If there are no executions left, show empty state and clear active execution from the store
|
// If there are no executions left, show empty state and clear active execution from the store
|
||||||
this.workflowsStore.activeWorkflowExecution = null;
|
this.workflowsStore.activeWorkflowExecution = null;
|
||||||
await this.$router.push({
|
await this.$router.replace({
|
||||||
name: VIEWS.EXECUTION_HOME,
|
name: VIEWS.EXECUTION_HOME,
|
||||||
params: { name: this.currentWorkflow },
|
params: { name: this.currentWorkflow },
|
||||||
});
|
});
|
||||||
|
@ -363,10 +364,8 @@ export default defineComponent({
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
stopAutoRefreshInterval() {
|
stopAutoRefreshInterval() {
|
||||||
if (this.autoRefreshTimeout) {
|
clearTimeout(this.autoRefreshTimeout);
|
||||||
clearTimeout(this.autoRefreshTimeout);
|
this.autoRefreshTimeout = undefined;
|
||||||
this.autoRefreshTimeout = undefined;
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
onAutoRefreshToggle(value: boolean): void {
|
onAutoRefreshToggle(value: boolean): void {
|
||||||
this.autoRefresh = value;
|
this.autoRefresh = value;
|
||||||
|
@ -448,7 +447,7 @@ export default defineComponent({
|
||||||
params: { name: this.currentWorkflow, executionId: this.executions[0].id },
|
params: { name: this.currentWorkflow, executionId: this.executions[0].id },
|
||||||
})
|
})
|
||||||
.catch(() => {});
|
.catch(() => {});
|
||||||
} else if (this.executions.length === 0) {
|
} else if (this.executions.length === 0 && this.$route.name === VIEWS.EXECUTION_PREVIEW) {
|
||||||
this.$router
|
this.$router
|
||||||
.push({
|
.push({
|
||||||
name: VIEWS.EXECUTION_HOME,
|
name: VIEWS.EXECUTION_HOME,
|
||||||
|
|
|
@ -3,7 +3,7 @@ import { merge } from 'lodash-es';
|
||||||
import { createTestingPinia } from '@pinia/testing';
|
import { createTestingPinia } from '@pinia/testing';
|
||||||
import userEvent from '@testing-library/user-event';
|
import userEvent from '@testing-library/user-event';
|
||||||
import { faker } from '@faker-js/faker';
|
import { faker } from '@faker-js/faker';
|
||||||
import { STORES } from '@/constants';
|
import { STORES, VIEWS } from '@/constants';
|
||||||
import ExecutionsList from '@/components/ExecutionsList.vue';
|
import ExecutionsList from '@/components/ExecutionsList.vue';
|
||||||
import type { IWorkflowDb } from '@/Interface';
|
import type { IWorkflowDb } from '@/Interface';
|
||||||
import type { IExecutionsSummary } from 'n8n-workflow';
|
import type { IExecutionsSummary } from 'n8n-workflow';
|
||||||
|
@ -12,6 +12,12 @@ import { useWorkflowsStore } from '@/stores/workflows.store';
|
||||||
import type { RenderOptions } from '@/__tests__/render';
|
import type { RenderOptions } from '@/__tests__/render';
|
||||||
import { createComponentRenderer } from '@/__tests__/render';
|
import { createComponentRenderer } from '@/__tests__/render';
|
||||||
|
|
||||||
|
vi.mock('vue-router', () => ({
|
||||||
|
useRoute: vi.fn().mockReturnValue({
|
||||||
|
name: VIEWS.WORKFLOW_EXECUTIONS,
|
||||||
|
}),
|
||||||
|
}));
|
||||||
|
|
||||||
let pinia: ReturnType<typeof createTestingPinia>;
|
let pinia: ReturnType<typeof createTestingPinia>;
|
||||||
|
|
||||||
const generateUndefinedNullOrString = () => {
|
const generateUndefinedNullOrString = () => {
|
||||||
|
|
Loading…
Reference in a new issue