fix: Fix foreign credentials being shown for new nodes (#4622)

* feat: Extract usedCredentials into separate store entry and fix foreign credentials being shown for new nodes

* chore: adjust spacing
This commit is contained in:
Alex Grozav 2022-11-17 16:22:46 +02:00 committed by GitHub
parent 7483e147fc
commit dea67ca6b7
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 27 additions and 11 deletions

View file

@ -904,6 +904,13 @@ export interface INodeMetadata {
parametersLastUpdatedAt?: number; parametersLastUpdatedAt?: number;
} }
export interface IUsedCredential {
id: string;
name: string;
credentialType: string;
currentUserHasAccess: boolean;
}
export interface WorkflowsState { export interface WorkflowsState {
activeExecutions: IExecutionsCurrentSummaryExtended[]; activeExecutions: IExecutionsCurrentSummaryExtended[];
activeWorkflows: string[]; activeWorkflows: string[];
@ -915,6 +922,7 @@ export interface WorkflowsState {
finishedExecutionsCount: number; finishedExecutionsCount: number;
nodeMetadata: NodeMetadataMap; nodeMetadata: NodeMetadataMap;
subWorkflowExecutionError: Error | null; subWorkflowExecutionError: Error | null;
usedCredentials: Record<string, IUsedCredential>;
workflow: IWorkflowDb; workflow: IWorkflowDb;
workflowExecutionData: IExecutionResponse | null; workflowExecutionData: IExecutionResponse | null;
workflowExecutionPairedItemMappings: {[itemId: string]: Set<string>}; workflowExecutionPairedItemMappings: {[itemId: string]: Set<string>};

View file

@ -370,12 +370,12 @@ export default mixins(
}, },
hasForeignCredential(): boolean { hasForeignCredential(): boolean {
const credentials = (this.activeNode || {}).credentials; const credentials = (this.activeNode || {}).credentials;
const foreignCredentials = this.credentialsStore.foreignCredentialsById; const usedCredentials = this.workflowsStore.usedCredentials;
let hasForeignCredential = false; let hasForeignCredential = false;
if (credentials && this.settingsStore.isEnterpriseFeatureEnabled(EnterpriseEditionFeature.WorkflowSharing)) { if (credentials && this.settingsStore.isEnterpriseFeatureEnabled(EnterpriseEditionFeature.WorkflowSharing)) {
Object.values(credentials).forEach((credential) => { Object.values(credentials).forEach((credential) => {
if (credential.id && foreignCredentials[credential.id] && !foreignCredentials[credential.id].currentUserHasAccess) { if (credential.id && usedCredentials[credential.id] && !usedCredentials[credential.id].currentUserHasAccess) {
hasForeignCredential = true; hasForeignCredential = true;
} }
}); });

View file

@ -49,9 +49,6 @@ export const useCredentialsStore = defineStore(STORES.CREDENTIALS, {
getCredentialById() { getCredentialById() {
return (id: string): ICredentialsResponse => this.credentials[id]; return (id: string): ICredentialsResponse => this.credentials[id];
}, },
foreignCredentialsById(): ICredentialMap {
return Object.fromEntries(Object.entries(this.credentials).filter(([_, credential]) => credential.hasOwnProperty('currentUserHasAccess')));
},
getCredentialByIdAndType() { getCredentialByIdAndType() {
return (id: string, type: string): ICredentialsResponse | undefined => { return (id: string, type: string): ICredentialsResponse | undefined => {
const credential = this.credentials[id]; const credential = this.credentials[id];
@ -226,7 +223,7 @@ export const useCredentialsStore = defineStore(STORES.CREDENTIALS, {
return credential; return credential;
}, },
async deleteCredential({ id }: {id: string}): void { async deleteCredential({ id }: {id: string}) {
const rootStore = useRootStore(); const rootStore = useRootStore();
const deleted = await deleteCredential(rootStore.getRestApiContext, id); const deleted = await deleteCredential(rootStore.getRestApiContext, id);
if (deleted) { if (deleted) {
@ -263,10 +260,10 @@ export const useCredentialsStore = defineStore(STORES.CREDENTIALS, {
}, },
// Enterprise edition actions // Enterprise edition actions
setCredentialOwnedBy(payload: { credentialId: string, ownedBy: Partial<IUser> }): void { setCredentialOwnedBy(payload: { credentialId: string, ownedBy: Partial<IUser> }) {
Vue.set(this.credentials[payload.credentialId], 'ownedBy', payload.ownedBy); Vue.set(this.credentials[payload.credentialId], 'ownedBy', payload.ownedBy);
}, },
async setCredentialSharedWith(payload: { sharedWith: IUser[]; credentialId: string; }): void { async setCredentialSharedWith(payload: { sharedWith: IUser[]; credentialId: string; }) {
if(useSettingsStore().isEnterpriseFeatureEnabled(EnterpriseEditionFeature.Sharing)) { if(useSettingsStore().isEnterpriseFeatureEnabled(EnterpriseEditionFeature.Sharing)) {
await setCredentialSharedWith( await setCredentialSharedWith(
useRootStore().getRestApiContext, useRootStore().getRestApiContext,

View file

@ -7,6 +7,7 @@ import {
STORES, STORES,
} from "@/constants"; } from "@/constants";
import { import {
ICredentialMap,
IExecutionResponse, IExecutionResponse,
IExecutionsCurrentSummaryExtended, IExecutionsCurrentSummaryExtended,
IExecutionsSummary, IExecutionsSummary,
@ -16,6 +17,7 @@ import {
IPushDataExecutionFinished, IPushDataExecutionFinished,
IPushDataNodeExecuteAfter, IPushDataNodeExecuteAfter,
IUpdateInformation, IUpdateInformation,
IUsedCredential,
IWorkflowDb, IWorkflowDb,
IWorkflowsMap, IWorkflowsMap,
WorkflowsState, WorkflowsState,
@ -37,6 +39,7 @@ import {
IWorkflowSettings, IWorkflowSettings,
} from 'n8n-workflow'; } from 'n8n-workflow';
import Vue from "vue"; import Vue from "vue";
import {useRootStore} from "./n8nRootStore"; import {useRootStore} from "./n8nRootStore";
import { import {
getActiveWorkflows, getActiveWorkflows,
@ -73,6 +76,7 @@ const createEmptyWorkflow = (): IWorkflowDb => ({
export const useWorkflowsStore = defineStore(STORES.WORKFLOWS, { export const useWorkflowsStore = defineStore(STORES.WORKFLOWS, {
state: (): WorkflowsState => ({ state: (): WorkflowsState => ({
workflow: createEmptyWorkflow(), workflow: createEmptyWorkflow(),
usedCredentials: {},
activeWorkflows: [], activeWorkflows: [],
activeExecutions: [], activeExecutions: [],
currentWorkflowExecutions: [], currentWorkflowExecutions: [],
@ -266,6 +270,13 @@ export const useWorkflowsStore = defineStore(STORES.WORKFLOWS, {
this.workflow.id = id === 'new' ? PLACEHOLDER_EMPTY_WORKFLOW_ID : id; this.workflow.id = id === 'new' ? PLACEHOLDER_EMPTY_WORKFLOW_ID : id;
}, },
setUsedCredentials(data: IUsedCredential[]) {
this.usedCredentials = data.reduce<{ [name: string]: IUsedCredential }>((accu, credential) => {
accu[credential.id!] = credential;
return accu;
}, {});
},
setWorkflowName(data: { newName: string, setStateDirty: boolean }): void { setWorkflowName(data: { newName: string, setStateDirty: boolean }): void {
if (data.setStateDirty === true) { if (data.setStateDirty === true) {
const uiStore = useUIStore(); const uiStore = useUIStore();

View file

@ -870,7 +870,7 @@ export default mixins(
} }
if (data.usedCredentials) { if (data.usedCredentials) {
this.credentialsStore.addCredentials(data.usedCredentials); this.workflowsStore.setUsedCredentials(data.usedCredentials);
} }
const tags = (data.tags || []) as ITag[]; const tags = (data.tags || []) as ITag[];
@ -2409,10 +2409,10 @@ export default mixins(
} }
if (newNodeData.credentials && this.settingsStore.isEnterpriseFeatureEnabled(EnterpriseEditionFeature.WorkflowSharing)) { if (newNodeData.credentials && this.settingsStore.isEnterpriseFeatureEnabled(EnterpriseEditionFeature.WorkflowSharing)) {
const foreignCredentials = this.credentialsStore.foreignCredentialsById; const usedCredentials = this.workflowsStore.usedCredentials;
newNodeData.credentials = Object.fromEntries( newNodeData.credentials = Object.fromEntries(
Object.entries(newNodeData.credentials).filter(([_, credential]) => { Object.entries(newNodeData.credentials).filter(([_, credential]) => {
return credential.id && (!foreignCredentials[credential.id] || foreignCredentials[credential.id]?.currentUserHasAccess); return credential.id && (!usedCredentials[credential.id] || usedCredentials[credential.id]?.currentUserHasAccess);
}), }),
); );
} }