fix(editor): Fix memory leak in Node Detail View by correctly unsubscribing from event buses (#6021)

This commit is contained in:
OlegIvaniv 2023-04-20 12:26:14 +02:00 committed by GitHub
parent 41660d9e28
commit 0970ec066d
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
13 changed files with 98 additions and 97 deletions

View file

@ -12,11 +12,12 @@ describe('Data transformation expressions', () => {
beforeEach(() => { beforeEach(() => {
wf.actions.visit(); wf.actions.visit();
cy.window() cy.window().then(
// @ts-ignore (win) => {
.then( // @ts-ignore
(win) => win.onBeforeUnloadNodeView && win.removeEventListener('beforeunload', win.onBeforeUnloadNodeView), win.preventNodeViewBeforeUnload = true;
); },
);
}); });
it('$json + native string methods', () => { it('$json + native string methods', () => {

View file

@ -17,11 +17,12 @@ describe('Data mapping', () => {
beforeEach(() => { beforeEach(() => {
workflowPage.actions.visit(); workflowPage.actions.visit();
cy.window() cy.window().then(
// @ts-ignore (win) => {
.then( // @ts-ignore
(win) => win.onBeforeUnloadNodeView && win.removeEventListener('beforeunload', win.onBeforeUnloadNodeView), win.preventNodeViewBeforeUnload = true;
); },
);
}); });
it('maps expressions from table header', () => { it('maps expressions from table header', () => {

View file

@ -99,11 +99,12 @@ describe('Webhook Trigger node', async () => {
beforeEach(() => { beforeEach(() => {
workflowPage.actions.visit(); workflowPage.actions.visit();
cy.window() cy.window().then(
// @ts-ignore (win) => {
.then( // @ts-ignore
(win) => win.onBeforeUnloadNodeView && win.removeEventListener('beforeunload', win.onBeforeUnloadNodeView), win.preventNodeViewBeforeUnload = true;
); },
);
}); });
it('should listen for a GET request', () => { it('should listen for a GET request', () => {

View file

@ -37,12 +37,10 @@ export default Vue.extend({
if (this.autofocus && this.$refs.input) { if (this.autofocus && this.$refs.input) {
this.focus(); this.focus();
} }
this.eventBus?.on('focus', this.focus);
if (this.eventBus) { },
this.eventBus.on('focus', () => { destroyed() {
this.focus(); this.eventBus?.off('focus', this.focus);
});
}
}, },
methods: { methods: {
focus() { focus() {

View file

@ -121,15 +121,8 @@ export default Vue.extend({
mounted() { mounted() {
window.addEventListener('keydown', this.onWindowKeydown); window.addEventListener('keydown', this.onWindowKeydown);
if (this.eventBus) { this.eventBus?.on('close', this.closeDialog);
this.eventBus.on('close', () => { this.eventBus?.on('closeAll', this.uiStore.closeAllModals);
this.closeDialog();
});
this.eventBus.on('closeAll', () => {
this.uiStore.closeAllModals();
});
}
const activeElement = document.activeElement as HTMLElement; const activeElement = document.activeElement as HTMLElement;
if (activeElement) { if (activeElement) {
@ -137,6 +130,8 @@ export default Vue.extend({
} }
}, },
beforeDestroy() { beforeDestroy() {
this.eventBus?.off('close', this.closeDialog);
this.eventBus?.off('closeAll', this.uiStore.closeAllModals);
window.removeEventListener('keydown', this.onWindowKeydown); window.removeEventListener('keydown', this.onWindowKeydown);
}, },
computed: { computed: {

View file

@ -53,12 +53,7 @@ export default Vue.extend({
}, },
mounted() { mounted() {
window.addEventListener('keydown', this.onWindowKeydown); window.addEventListener('keydown', this.onWindowKeydown);
this.eventBus?.on('close', this.close);
if (this.eventBus) {
this.eventBus.on('close', () => {
this.close();
});
}
const activeElement = document.activeElement as HTMLElement; const activeElement = document.activeElement as HTMLElement;
if (activeElement) { if (activeElement) {
@ -66,6 +61,7 @@ export default Vue.extend({
} }
}, },
beforeDestroy() { beforeDestroy() {
this.eventBus?.off('close', this.close);
window.removeEventListener('keydown', this.onWindowKeydown); window.removeEventListener('keydown', this.onWindowKeydown);
}, },
computed: { computed: {

View file

@ -211,15 +211,10 @@ export default mixins(
}; };
}, },
mounted() { mounted() {
dataPinningEventBus.on( dataPinningEventBus.on('data-pinning-discovery', this.setIsTooltipVisible);
'data-pinning-discovery',
({ isTooltipVisible }: { isTooltipVisible: boolean }) => {
this.pinDataDiscoveryTooltipVisible = isTooltipVisible;
},
);
}, },
destroyed() { destroyed() {
dataPinningEventBus.off('data-pinning-discovery'); dataPinningEventBus.off('data-pinning-discovery', this.setIsTooltipVisible);
}, },
computed: { computed: {
...mapStores(useNodeTypesStore, useNDVStore, useUIStore, useWorkflowsStore, useSettingsStore), ...mapStores(useNodeTypesStore, useNDVStore, useUIStore, useWorkflowsStore, useSettingsStore),
@ -480,6 +475,9 @@ export default mixins(
}, },
}, },
methods: { methods: {
setIsTooltipVisible({ isTooltipVisible }: { isTooltipVisible: boolean }) {
this.pinDataDiscoveryTooltipVisible = isTooltipVisible;
},
onKeyDown(e: KeyboardEvent) { onKeyDown(e: KeyboardEvent) {
if (e.key === 's' && this.isCtrlKeyPressed(e)) { if (e.key === 's' && this.isCtrlKeyPressed(e)) {
e.stopPropagation(); e.stopPropagation();

View file

@ -895,18 +895,20 @@ export default mixins(externalHooks, nodeHelpers).extend({
onStopExecution() { onStopExecution() {
this.$emit('stopExecution'); this.$emit('stopExecution');
}, },
openSettings() {
this.openPanel = 'settings';
},
}, },
mounted() { mounted() {
this.populateHiddenIssuesSet(); this.populateHiddenIssuesSet();
this.setNodeValues(); this.setNodeValues();
if (this.eventBus) { this.eventBus?.on('openSettings', this.openSettings);
this.eventBus.on('openSettings', () => {
this.openPanel = 'settings';
});
}
this.updateNodeParameterIssues(this.node as INodeUi, this.nodeType); this.updateNodeParameterIssues(this.node as INodeUi, this.nodeType);
}, },
destroyed() {
this.eventBus?.off('openSettings', this.openSettings);
},
}); });
</script> </script>

View file

@ -585,7 +585,6 @@ export default mixins(externalHooks, genericHelpers, nodeHelpers, pinData).exten
currentPage: 1, currentPage: 1,
pageSize: 10, pageSize: 10,
pageSizes: [10, 25, 50, 100], pageSizes: [10, 25, 50, 100],
eventBus: dataPinningEventBus,
pinDataDiscoveryTooltipVisible: false, pinDataDiscoveryTooltipVisible: false,
isControlledPinDataTooltip: false, isControlledPinDataTooltip: false,
@ -595,8 +594,8 @@ export default mixins(externalHooks, genericHelpers, nodeHelpers, pinData).exten
this.init(); this.init();
if (!this.isPaneTypeInput) { if (!this.isPaneTypeInput) {
this.eventBus.on('data-pinning-error', this.onDataPinningError); dataPinningEventBus.on('data-pinning-error', this.onDataPinningError);
this.eventBus.on('data-unpinning', this.onDataUnpinning); dataPinningEventBus.on('data-unpinning', this.onDataUnpinning);
this.showPinDataDiscoveryTooltip(this.jsonData); this.showPinDataDiscoveryTooltip(this.jsonData);
} }
@ -609,8 +608,8 @@ export default mixins(externalHooks, genericHelpers, nodeHelpers, pinData).exten
}, },
destroyed() { destroyed() {
this.hidePinDataDiscoveryTooltip(); this.hidePinDataDiscoveryTooltip();
this.eventBus.off('data-pinning-error', this.onDataPinningError); dataPinningEventBus.off('data-pinning-error', this.onDataPinningError);
this.eventBus.off('data-unpinning', this.onDataUnpinning); dataPinningEventBus.off('data-unpinning', this.onDataUnpinning);
}, },
computed: { computed: {
...mapStores(useNodeTypesStore, useNDVStore, useWorkflowsStore), ...mapStores(useNodeTypesStore, useNDVStore, useWorkflowsStore),
@ -908,7 +907,7 @@ export default mixins(externalHooks, genericHelpers, nodeHelpers, pinData).exten
setTimeout(() => { setTimeout(() => {
this.isControlledPinDataTooltip = true; this.isControlledPinDataTooltip = true;
this.pinDataDiscoveryTooltipVisible = true; this.pinDataDiscoveryTooltipVisible = true;
this.eventBus.emit('data-pinning-discovery', { isTooltipVisible: true }); dataPinningEventBus.emit('data-pinning-discovery', { isTooltipVisible: true });
}, 500); // Wait for NDV to open }, 500); // Wait for NDV to open
} }
}, },
@ -916,7 +915,7 @@ export default mixins(externalHooks, genericHelpers, nodeHelpers, pinData).exten
if (this.pinDataDiscoveryTooltipVisible) { if (this.pinDataDiscoveryTooltipVisible) {
this.isControlledPinDataTooltip = false; this.isControlledPinDataTooltip = false;
this.pinDataDiscoveryTooltipVisible = false; this.pinDataDiscoveryTooltipVisible = false;
this.eventBus.emit('data-pinning-discovery', { isTooltipVisible: false }); dataPinningEventBus.emit('data-pinning-discovery', { isTooltipVisible: false });
} }
}, },
pinDataDiscoveryComplete() { pinDataDiscoveryComplete() {

View file

@ -92,15 +92,10 @@ export default mixins(showMessage).extend({
deepCopy(defaultMessageEventBusDestinationOptions), deepCopy(defaultMessageEventBusDestinationOptions),
this.destination, this.destination,
); );
this.eventBus.on('destinationWasSaved', () => { this.eventBus?.on('destinationWasSaved', this.onDestinationWasSaved);
const updatedDestination = this.logStreamingStore.getDestination(this.destination.id); },
if (updatedDestination) { destroyed() {
this.nodeParameters = Object.assign( this.eventBus?.off('destinationWasSaved', this.onDestinationWasSaved);
deepCopy(defaultMessageEventBusDestinationOptions),
this.destination,
);
}
});
}, },
computed: { computed: {
...mapStores(useLogStreamingStore), ...mapStores(useLogStreamingStore),
@ -124,6 +119,15 @@ export default mixins(showMessage).extend({
}, },
}, },
methods: { methods: {
onDestinationWasSaved() {
const updatedDestination = this.logStreamingStore.getDestination(this.destination.id);
if (updatedDestination) {
this.nodeParameters = Object.assign(
deepCopy(defaultMessageEventBusDestinationOptions),
this.destination,
);
}
},
async onClick(event?: PointerEvent) { async onClick(event?: PointerEvent) {
if ( if (
event && event &&

View file

@ -117,15 +117,13 @@ export default mixins(showMessage).extend({
} }
} }
if (this.eventBus) { this.eventBus?.on('focus', this.onBusFocus);
this.eventBus.on('focus', () => {
this.focusOnInput();
this.focusOnTopOption();
});
}
this.tagsStore.fetchAll(); this.tagsStore.fetchAll();
}, },
destroyed() {
this.eventBus?.off('focus', this.onBusFocus);
},
computed: { computed: {
...mapStores(useTagsStore, useUIStore), ...mapStores(useTagsStore, useUIStore),
allTags(): ITag[] { allTags(): ITag[] {
@ -144,6 +142,10 @@ export default mixins(showMessage).extend({
}, },
}, },
methods: { methods: {
onBusFocus() {
this.focusOnInput();
this.focusOnTopOption();
},
filterOptions(filter = '') { filterOptions(filter = '') {
this.$data.filter = filter.trim(); this.$data.filter = filter.trim();
this.$nextTick(() => this.focusOnTopOption()); this.$nextTick(() => this.focusOnTopOption());

View file

@ -2516,6 +2516,20 @@ export default mixins(
} }
} }
}, },
onBeforeUnload(e) {
if (this.isDemo || window.preventNodeViewBeforeUnload) {
return;
} else if (this.uiStore.stateIsDirty) {
const confirmationMessage = this.$locale.baseText(
'nodeView.itLooksLikeYouHaveBeenEditingSomething',
);
(e || window.event).returnValue = confirmationMessage; //Gecko + IE
return confirmationMessage; //Gecko + Webkit, Safari, Chrome etc.
} else {
this.startLoading(this.$locale.baseText('nodeView.redirecting'));
return;
}
},
async newWorkflow(): Promise<void> { async newWorkflow(): Promise<void> {
this.startLoading(); this.startLoading();
await this.resetWorkspace(); await this.resetWorkspace();
@ -2597,23 +2611,7 @@ export default mixins(
document.addEventListener('keydown', this.keyDown); document.addEventListener('keydown', this.keyDown);
document.addEventListener('keyup', this.keyUp); document.addEventListener('keyup', this.keyUp);
// allow to be overriden in e2e tests window.addEventListener('beforeunload', this.onBeforeUnload);
// @ts-ignore
window.onBeforeUnloadNodeView = (e) => {
if (this.isDemo) {
return;
} else if (this.uiStore.stateIsDirty) {
const confirmationMessage = this.$locale.baseText(
'nodeView.itLooksLikeYouHaveBeenEditingSomething',
);
(e || window.event).returnValue = confirmationMessage; //Gecko + IE
return confirmationMessage; //Gecko + Webkit, Safari, Chrome etc.
} else {
this.startLoading(this.$locale.baseText('nodeView.redirecting'));
return;
}
};
window.addEventListener('beforeunload', window.onBeforeUnloadNodeView);
}, },
getOutputEndpointUUID(nodeName: string, index: number): string | null { getOutputEndpointUUID(nodeName: string, index: number): string | null {
const node = this.workflowsStore.getNodeByName(nodeName); const node = this.workflowsStore.getNodeByName(nodeName);
@ -3972,6 +3970,7 @@ export default mixins(
document.removeEventListener('keydown', this.keyDown); document.removeEventListener('keydown', this.keyDown);
document.removeEventListener('keyup', this.keyUp); document.removeEventListener('keyup', this.keyUp);
window.removeEventListener('message', this.onPostMessageReceived); window.removeEventListener('message', this.onPostMessageReceived);
window.removeEventListener('beforeunload', this.onBeforeUnload);
this.$root.$off('newWorkflow', this.newWorkflow); this.$root.$off('newWorkflow', this.newWorkflow);
this.$root.$off('importWorkflowData', this.onImportWorkflowDataEvent); this.$root.$off('importWorkflowData', this.onImportWorkflowDataEvent);

View file

@ -135,18 +135,16 @@ export default mixins().extend({
} }
}); });
// refresh when a modal closes // refresh when a modal closes
this.eventBus.on('destinationWasSaved', async () => { this.eventBus.on('destinationWasSaved', this.onDestinationWasSaved);
this.$forceUpdate();
});
// listen to remove emission // listen to remove emission
this.eventBus.on('remove', async (destinationId: string) => { this.eventBus.on('remove', this.onRemove);
await this.onRemove(destinationId);
});
// listen to modal closing and remove nodes from store // listen to modal closing and remove nodes from store
this.eventBus.on('closing', async (destinationId: string) => { this.eventBus.on('closing', this.onBusClosing);
this.workflowsStore.removeAllNodes({ setStateDirty: false, removePinData: true }); },
this.uiStore.stateIsDirty = false; destroyed() {
}); this.eventBus.off('destinationWasSaved', this.onDestinationWasSaved);
this.eventBus.off('remove', this.onRemove);
this.eventBus.off('closing', this.onBusClosing);
}, },
computed: { computed: {
...mapStores( ...mapStores(
@ -173,6 +171,13 @@ export default mixins().extend({
}, },
}, },
methods: { methods: {
onDestinationWasSaved() {
this.$forceUpdate();
},
onBusClosing() {
this.workflowsStore.removeAllNodes({ setStateDirty: false, removePinData: true });
this.uiStore.stateIsDirty = false;
},
async getDestinationDataFromBackend(): Promise<void> { async getDestinationDataFromBackend(): Promise<void> {
this.logStreamingStore.clearEventNames(); this.logStreamingStore.clearEventNames();
this.logStreamingStore.clearDestinationItemTrees(); this.logStreamingStore.clearDestinationItemTrees();