mirror of
https://github.com/n8n-io/n8n.git
synced 2025-03-05 20:50:17 -08:00
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:
parent
7483e147fc
commit
dea67ca6b7
|
@ -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>};
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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);
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue