mirror of
https://github.com/n8n-io/n8n.git
synced 2024-11-15 17:14:05 -08:00
271 lines
8.1 KiB
Vue
271 lines
8.1 KiB
Vue
<template>
|
|
<div v-if="dialogVisible" @keydown.stop>
|
|
<el-dialog :visible="dialogVisible" append-to-body width="55%" :title="title" :before-close="closeDialog">
|
|
|
|
<div class="credential-type-item">
|
|
<el-row v-if="!setCredentialType">
|
|
<el-col :span="6">
|
|
Credential type:
|
|
</el-col>
|
|
<el-col :span="18">
|
|
<el-select v-model="credentialType" filterable placeholder="Select Type" size="small">
|
|
<el-option
|
|
v-for="item in credentialTypes"
|
|
:key="item.name"
|
|
:label="item.displayName"
|
|
:value="item.name">
|
|
</el-option>
|
|
</el-select>
|
|
</el-col>
|
|
</el-row>
|
|
</div>
|
|
|
|
<credentials-input v-if="credentialType" @credentialsCreated="credentialsCreated" @credentialsUpdated="credentialsUpdated" :credentialTypeData="getCredentialTypeData(credentialType)" :credentialData="credentialData" :nodesInit="nodesInit"></credentials-input>
|
|
</el-dialog>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts">
|
|
import Vue from 'vue';
|
|
|
|
import { restApi } from '@/components/mixins/restApi';
|
|
import { showMessage } from '@/components/mixins/showMessage';
|
|
import CredentialsInput from '@/components/CredentialsInput.vue';
|
|
import {
|
|
ICredentialsCreatedEvent,
|
|
ICredentialsDecryptedResponse,
|
|
} from '@/Interface';
|
|
|
|
import {
|
|
ICredentialType,
|
|
INodeProperties,
|
|
} from 'n8n-workflow';
|
|
|
|
import mixins from 'vue-typed-mixins';
|
|
|
|
export default mixins(
|
|
restApi,
|
|
showMessage,
|
|
).extend({
|
|
name: 'CredentialsEdit',
|
|
props: [
|
|
'dialogVisible', // Boolean
|
|
'editCredentials',
|
|
'setCredentialType', // String
|
|
'nodesInit', // Array
|
|
],
|
|
components: {
|
|
CredentialsInput,
|
|
},
|
|
data () {
|
|
return {
|
|
credentialData: null as ICredentialsDecryptedResponse | null,
|
|
credentialType: null as string | null,
|
|
};
|
|
},
|
|
computed: {
|
|
credentialTypes (): ICredentialType[] {
|
|
const credentialTypes = this.$store.getters.allCredentialTypes;
|
|
if (credentialTypes === null) {
|
|
return [];
|
|
}
|
|
return credentialTypes;
|
|
},
|
|
title (): string {
|
|
if (this.editCredentials) {
|
|
const credentialType = this.$store.getters.credentialType(this.editCredentials.type);
|
|
return `Edit Credentials: "${credentialType.displayName}"`;
|
|
} else {
|
|
if (this.credentialType) {
|
|
const credentialType = this.$store.getters.credentialType(this.credentialType);
|
|
return `Create New Credentials: "${credentialType.displayName}"`;
|
|
} else {
|
|
return `Create New Credentials`;
|
|
}
|
|
}
|
|
},
|
|
},
|
|
watch: {
|
|
async dialogVisible (newValue, oldValue): Promise<void> {
|
|
if (newValue) {
|
|
if (this.editCredentials) {
|
|
// Credentials which should be edited are given
|
|
const credentialType = this.$store.getters.credentialType(this.editCredentials.type);
|
|
|
|
if (credentialType === null) {
|
|
this.$showMessage({
|
|
title: 'Credential type not known',
|
|
message: `Credentials of type "${this.editCredentials.type}" are not known.`,
|
|
type: 'error',
|
|
duration: 0,
|
|
});
|
|
this.closeDialog();
|
|
return;
|
|
}
|
|
|
|
if (this.editCredentials.id === undefined) {
|
|
this.$showMessage({
|
|
title: 'Credential ID missing',
|
|
message: 'The ID of the credentials which should be edited is missing!',
|
|
type: 'error',
|
|
});
|
|
this.closeDialog();
|
|
return;
|
|
}
|
|
|
|
let currentCredentials: ICredentialsDecryptedResponse | undefined;
|
|
try {
|
|
currentCredentials = await this.restApi().getCredentials(this.editCredentials.id as string, true) as ICredentialsDecryptedResponse | undefined;
|
|
} catch (error) {
|
|
this.$showError(error, 'Problem loading credentials', 'There was a problem loading the credentials:');
|
|
this.closeDialog();
|
|
return;
|
|
}
|
|
|
|
if (currentCredentials === undefined) {
|
|
this.$showMessage({
|
|
title: 'Credentials not found',
|
|
message: `Could not find the credentials with the id: ${this.editCredentials.id}`,
|
|
type: 'error',
|
|
duration: 0,
|
|
});
|
|
this.closeDialog();
|
|
return;
|
|
}
|
|
|
|
if (currentCredentials === undefined) {
|
|
this.$showMessage({
|
|
title: 'Problem loading credentials',
|
|
message: 'No credentials could be loaded!',
|
|
type: 'error',
|
|
});
|
|
return;
|
|
}
|
|
|
|
this.credentialData = currentCredentials;
|
|
} else {
|
|
if (this.credentialType || this.setCredentialType) {
|
|
const credentialType = this.$store.getters.credentialType(this.credentialType || this.setCredentialType);
|
|
if (credentialType === null) {
|
|
this.$showMessage({
|
|
title: 'Credential type not known',
|
|
message: `Credentials of type "${this.credentialType || this.setCredentialType}" are not known.`,
|
|
type: 'error',
|
|
duration: 0,
|
|
});
|
|
this.closeDialog();
|
|
return;
|
|
}
|
|
}
|
|
|
|
this.credentialData = null;
|
|
}
|
|
|
|
if (this.setCredentialType || (this.credentialData && this.credentialData.type)) {
|
|
this.credentialType = this.setCredentialType || (this.credentialData && this.credentialData.type);
|
|
}
|
|
} else {
|
|
// Make sure that it gets always reset else it uses by default
|
|
// again the last selection from when it was open the previous time.
|
|
this.credentialType = null;
|
|
}
|
|
},
|
|
},
|
|
methods: {
|
|
mergeCredentialProperties (mainProperties: INodeProperties[], addProperties: INodeProperties[]): void {
|
|
let existingIndex: number;
|
|
for (const property of addProperties) {
|
|
existingIndex = mainProperties.findIndex(element => element.name === property.name);
|
|
|
|
if (existingIndex === -1) {
|
|
// Property does not exist yet, so add
|
|
mainProperties.push(property);
|
|
} else {
|
|
// Property exists already, so overwrite
|
|
mainProperties[existingIndex] = property;
|
|
}
|
|
}
|
|
},
|
|
getCredentialProperties (name: string): INodeProperties[] {
|
|
const credentialsData = this.$store.getters.credentialType(name);
|
|
|
|
if (credentialsData === null) {
|
|
throw new Error(`Could not find credentials of type: ${name}`);
|
|
}
|
|
|
|
if (credentialsData.extends === undefined) {
|
|
return credentialsData.properties;
|
|
}
|
|
|
|
const combineProperties = [] as INodeProperties[];
|
|
for (const credentialsTypeName of credentialsData.extends) {
|
|
const mergeCredentialProperties = this.getCredentialProperties(credentialsTypeName);
|
|
this.mergeCredentialProperties(combineProperties, mergeCredentialProperties);
|
|
}
|
|
|
|
// The properties defined on the parent credentials take presidence
|
|
this.mergeCredentialProperties(combineProperties, credentialsData.properties);
|
|
|
|
return combineProperties;
|
|
},
|
|
getCredentialTypeData (name: string): ICredentialType | null {
|
|
let credentialData = this.$store.getters.credentialType(name);
|
|
|
|
if (credentialData === null || credentialData.extends === undefined) {
|
|
return credentialData;
|
|
}
|
|
|
|
// TODO: The credential-extend-resolve-logic is currently not needed in the backend
|
|
// as the whole credential data gets saved with the defaults. That logic should,
|
|
// however, probably also get improved in the future.
|
|
|
|
// Credentials extends another one. So get the properties of the one it
|
|
// extends and add them.
|
|
credentialData = JSON.parse(JSON.stringify(credentialData));
|
|
credentialData.properties = this.getCredentialProperties(credentialData.name);
|
|
|
|
return credentialData;
|
|
},
|
|
credentialsCreated (eventData: ICredentialsCreatedEvent): void {
|
|
this.$emit('credentialsCreated', eventData);
|
|
|
|
this.$showMessage({
|
|
title: 'Credentials created',
|
|
message: `The credential "${eventData.data.name}" got created!`,
|
|
type: 'success',
|
|
});
|
|
|
|
if (eventData.options.closeDialog === true) {
|
|
this.closeDialog();
|
|
}
|
|
},
|
|
credentialsUpdated (eventData: ICredentialsCreatedEvent): void {
|
|
this.$emit('credentialsUpdated', eventData);
|
|
|
|
this.$showMessage({
|
|
title: 'Credentials updated',
|
|
message: `The credential "${eventData.data.name}" got updated!`,
|
|
type: 'success',
|
|
});
|
|
|
|
if (eventData.options.closeDialog === true) {
|
|
this.closeDialog();
|
|
}
|
|
},
|
|
closeDialog (): void {
|
|
// Handle the close externally as the visible parameter is an external prop
|
|
// and is so not allowed to be changed here.
|
|
this.$emit('closeDialog');
|
|
},
|
|
},
|
|
});
|
|
</script>
|
|
|
|
<style lang="scss">
|
|
|
|
.credential-type-item {
|
|
padding-bottom: 1em;
|
|
}
|
|
|
|
</style>
|