2022-06-08 11:53:12 -07:00
|
|
|
<template>
|
|
|
|
<SettingsView>
|
|
|
|
<div :class="$style.container">
|
|
|
|
<div :class="$style.header">
|
|
|
|
<n8n-heading size="2xlarge">
|
|
|
|
{{ $locale.baseText('settings.api') }}
|
|
|
|
<span :style="{ fontSize: 'var(--font-size-s)', color: 'var(--color-text-light)', }">
|
|
|
|
({{ $locale.baseText('beta') }})
|
|
|
|
</span>
|
|
|
|
</n8n-heading>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div v-if="apiKey">
|
|
|
|
<p class="mb-s">
|
|
|
|
<n8n-info-tip :bold="false">
|
2022-09-05 07:36:22 -07:00
|
|
|
<i18n path="settings.api.view.info" tag="span">
|
|
|
|
<template #apiAction>
|
|
|
|
<a
|
|
|
|
href="https://docs.n8n.io/api"
|
|
|
|
target="_blank"
|
|
|
|
v-text="$locale.baseText('settings.api.view.info.api')"
|
|
|
|
/>
|
|
|
|
</template>
|
|
|
|
<template #webhookAction>
|
|
|
|
<a
|
|
|
|
href="https://docs.n8n.io/integrations/core-nodes/n8n-nodes-base.webhook/"
|
|
|
|
target="_blank"
|
|
|
|
v-text="$locale.baseText('settings.api.view.info.webhook')"
|
|
|
|
/>
|
|
|
|
</template>
|
|
|
|
</i18n>
|
2022-06-08 11:53:12 -07:00
|
|
|
</n8n-info-tip>
|
|
|
|
</p>
|
|
|
|
<n8n-card class="mb-4xs" :class="$style.card">
|
|
|
|
<span :class="$style.delete">
|
|
|
|
<n8n-link @click="showDeleteModal" :bold="true">
|
|
|
|
{{ $locale.baseText('generic.delete') }}
|
|
|
|
</n8n-link>
|
|
|
|
</span>
|
2022-09-21 01:20:29 -07:00
|
|
|
<div class="ph-no-capture">
|
|
|
|
<CopyInput
|
|
|
|
:label="$locale.baseText('settings.api.view.myKey')"
|
|
|
|
:value="apiKey"
|
|
|
|
:copy-button-text="$locale.baseText('generic.clickToCopy')"
|
|
|
|
:toast-title="$locale.baseText('settings.api.view.copy.toast')"
|
|
|
|
@copy="onCopy"
|
|
|
|
/>
|
|
|
|
</div>
|
2022-06-08 11:53:12 -07:00
|
|
|
</n8n-card>
|
|
|
|
<div :class="$style.hint">
|
|
|
|
<n8n-text size="small">
|
|
|
|
{{ $locale.baseText('settings.api.view.tryapi') }}
|
|
|
|
</n8n-text>
|
|
|
|
<n8n-link :to="apiPlaygroundPath" :newWindow="true" size="small">
|
|
|
|
{{ $locale.baseText('settings.api.view.apiPlayground') }}
|
|
|
|
</n8n-link>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<n8n-action-box
|
|
|
|
v-else-if="mounted"
|
|
|
|
:buttonText="$locale.baseText(loading ? 'settings.api.create.button.loading' : 'settings.api.create.button')"
|
|
|
|
:description="$locale.baseText('settings.api.create.description')"
|
|
|
|
@click="createApiKey"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</SettingsView>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts">
|
|
|
|
import { showMessage } from '@/components/mixins/showMessage';
|
|
|
|
import { IUser } from '@/Interface';
|
|
|
|
import mixins from 'vue-typed-mixins';
|
|
|
|
|
|
|
|
import SettingsView from './SettingsView.vue';
|
|
|
|
import CopyInput from '../components/CopyInput.vue';
|
2022-11-04 06:04:31 -07:00
|
|
|
import { mapStores } from 'pinia';
|
|
|
|
import { useSettingsStore } from '@/stores/settings';
|
|
|
|
import { useRootStore } from '@/stores/n8nRootStore';
|
|
|
|
import { useUsersStore } from '@/stores/users';
|
2022-06-08 11:53:12 -07:00
|
|
|
|
|
|
|
export default mixins(
|
|
|
|
showMessage,
|
|
|
|
).extend({
|
|
|
|
name: 'SettingsPersonalView',
|
|
|
|
components: {
|
|
|
|
SettingsView,
|
|
|
|
CopyInput,
|
|
|
|
},
|
|
|
|
data() {
|
|
|
|
return {
|
|
|
|
loading: false,
|
|
|
|
mounted: false,
|
|
|
|
apiKey: '',
|
|
|
|
apiPlaygroundPath: '',
|
|
|
|
};
|
|
|
|
},
|
|
|
|
mounted() {
|
|
|
|
this.getApiKey();
|
2022-11-04 06:04:31 -07:00
|
|
|
const baseUrl = this.rootStore.baseUrl;
|
|
|
|
const apiPath = this.settingsStore.publicApiPath;
|
|
|
|
const latestVersion = this.settingsStore.publicApiLatestVersion;
|
2022-06-08 11:53:12 -07:00
|
|
|
this.apiPlaygroundPath = `${baseUrl}${apiPath}/v${latestVersion}/docs`;
|
|
|
|
},
|
|
|
|
computed: {
|
2022-11-04 06:04:31 -07:00
|
|
|
...mapStores(
|
|
|
|
useRootStore,
|
|
|
|
useSettingsStore,
|
|
|
|
useUsersStore,
|
|
|
|
),
|
|
|
|
currentUser(): IUser | null {
|
|
|
|
return this.usersStore.currentUser;
|
2022-06-08 11:53:12 -07:00
|
|
|
},
|
|
|
|
},
|
|
|
|
methods: {
|
|
|
|
async showDeleteModal() {
|
|
|
|
const confirmed = await this.confirmMessage(
|
|
|
|
this.$locale.baseText('settings.api.delete.description'),
|
|
|
|
this.$locale.baseText('settings.api.delete.title'),
|
|
|
|
null,
|
|
|
|
this.$locale.baseText('settings.api.delete.button'),
|
|
|
|
this.$locale.baseText('generic.cancel'),
|
|
|
|
);
|
|
|
|
if (confirmed) {
|
|
|
|
this.deleteApiKey();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
async getApiKey() {
|
|
|
|
try {
|
2022-11-04 06:04:31 -07:00
|
|
|
this.apiKey = await this.settingsStore.getApiKey() || '';
|
2022-06-08 11:53:12 -07:00
|
|
|
} catch (error) {
|
|
|
|
this.$showError(error, this.$locale.baseText('settings.api.view.error'));
|
|
|
|
} finally {
|
|
|
|
this.mounted = true;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
async createApiKey() {
|
|
|
|
this.loading = true;
|
|
|
|
|
|
|
|
try {
|
2022-11-04 06:04:31 -07:00
|
|
|
this.apiKey = await this.settingsStore.createApiKey() || '';
|
2022-06-08 11:53:12 -07:00
|
|
|
} catch (error) {
|
|
|
|
this.$showError(error, this.$locale.baseText('settings.api.create.error'));
|
|
|
|
} finally {
|
|
|
|
this.loading = false;
|
|
|
|
this.$telemetry.track('User clicked create API key button');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
async deleteApiKey() {
|
|
|
|
try {
|
2022-11-04 06:04:31 -07:00
|
|
|
await this.settingsStore.deleteApiKey();
|
2022-06-08 11:53:12 -07:00
|
|
|
this.$showMessage({ title: this.$locale.baseText("settings.api.delete.toast"), type: 'success' });
|
|
|
|
this.apiKey = '';
|
|
|
|
} catch (error) {
|
|
|
|
this.$showError(error, this.$locale.baseText('settings.api.delete.error'));
|
|
|
|
} finally {
|
|
|
|
this.$telemetry.track('User clicked delete API key button');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
onCopy() {
|
|
|
|
this.$telemetry.track('User clicked copy API key button');
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" module>
|
|
|
|
.container {
|
|
|
|
> * {
|
|
|
|
margin-bottom: var(--spacing-2xl);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.header {
|
|
|
|
display: flex;
|
|
|
|
align-items: center;
|
|
|
|
white-space: nowrap;
|
|
|
|
|
|
|
|
*:first-child {
|
|
|
|
flex-grow: 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.card {
|
|
|
|
position: relative;
|
|
|
|
}
|
|
|
|
|
|
|
|
.delete {
|
|
|
|
position: absolute;
|
|
|
|
display: inline-block;
|
|
|
|
top: var(--spacing-s);
|
|
|
|
right: var(--spacing-s);
|
|
|
|
}
|
|
|
|
|
|
|
|
.hint {
|
|
|
|
color: var(--color-text-light);
|
|
|
|
}
|
|
|
|
</style>
|
|
|
|
|