2023-01-04 00:47:48 -08:00
|
|
|
<template>
|
|
|
|
<div>
|
|
|
|
<div :class="$style.header">
|
|
|
|
<div class="mb-2xl">
|
|
|
|
<n8n-heading size="2xlarge">
|
|
|
|
{{ $locale.baseText(`settings.log-streaming.heading`) }}
|
|
|
|
</n8n-heading>
|
|
|
|
<template v-if="environment !== 'production'">
|
2023-04-24 06:37:36 -07:00
|
|
|
<strong class="ml-m">Disable License ({{ environment }}) </strong>
|
2023-01-04 00:47:48 -08:00
|
|
|
<el-switch v-model="disableLicense" size="large" data-test-id="disable-license-toggle" />
|
|
|
|
</template>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<template v-if="isLicensed">
|
|
|
|
<div class="mb-l">
|
|
|
|
<n8n-info-tip theme="info" type="note">
|
|
|
|
<template>
|
|
|
|
<span v-html="$locale.baseText('settings.log-streaming.infoText')"></span>
|
|
|
|
</template>
|
|
|
|
</n8n-info-tip>
|
|
|
|
</div>
|
|
|
|
<template v-if="storeHasItems()">
|
|
|
|
<el-row
|
|
|
|
:gutter="10"
|
|
|
|
v-for="item in sortedItemKeysByLabel"
|
|
|
|
:key="item.key"
|
|
|
|
:class="$style.destinationItem"
|
|
|
|
>
|
|
|
|
<el-col v-if="logStreamingStore.items[item.key]?.destination">
|
|
|
|
<event-destination-card
|
|
|
|
:destination="logStreamingStore.items[item.key]?.destination"
|
|
|
|
:eventBus="eventBus"
|
|
|
|
:isInstanceOwner="isInstanceOwner"
|
|
|
|
@remove="onRemove(logStreamingStore.items[item.key]?.destination?.id)"
|
|
|
|
@edit="onEdit(logStreamingStore.items[item.key]?.destination?.id)"
|
|
|
|
/>
|
|
|
|
</el-col>
|
|
|
|
</el-row>
|
|
|
|
<div class="mt-m text-right">
|
|
|
|
<n8n-button v-if="isInstanceOwner" size="large" @click="addDestination">
|
|
|
|
{{ $locale.baseText(`settings.log-streaming.add`) }}
|
|
|
|
</n8n-button>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<template v-else>
|
2023-04-24 06:37:36 -07:00
|
|
|
<div data-test-id="action-box-licensed">
|
2023-01-04 00:47:48 -08:00
|
|
|
<n8n-action-box
|
|
|
|
:buttonText="$locale.baseText(`settings.log-streaming.add`)"
|
|
|
|
@click="addDestination"
|
|
|
|
>
|
|
|
|
<template #heading>
|
|
|
|
<span v-html="$locale.baseText(`settings.log-streaming.addFirstTitle`)" />
|
|
|
|
</template>
|
|
|
|
</n8n-action-box>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
</template>
|
|
|
|
<template v-else>
|
|
|
|
<div v-if="$locale.baseText('settings.log-streaming.infoText')" class="mb-l">
|
|
|
|
<n8n-info-tip theme="info" type="note">
|
|
|
|
<template>
|
|
|
|
<span v-html="$locale.baseText('settings.log-streaming.infoText')"></span>
|
|
|
|
</template>
|
|
|
|
</n8n-info-tip>
|
|
|
|
</div>
|
2023-04-24 06:37:36 -07:00
|
|
|
<div data-test-id="action-box-unlicensed">
|
2023-01-04 00:47:48 -08:00
|
|
|
<n8n-action-box
|
|
|
|
:description="$locale.baseText('settings.log-streaming.actionBox.description')"
|
|
|
|
:buttonText="$locale.baseText('settings.log-streaming.actionBox.button')"
|
2023-04-24 06:37:36 -07:00
|
|
|
@click="goToUpgrade"
|
2023-01-04 00:47:48 -08:00
|
|
|
>
|
|
|
|
<template #heading>
|
|
|
|
<span v-html="$locale.baseText('settings.log-streaming.actionBox.title')" />
|
|
|
|
</template>
|
|
|
|
</n8n-action-box>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts">
|
|
|
|
import { v4 as uuid } from 'uuid';
|
|
|
|
import { mapStores } from 'pinia';
|
|
|
|
import mixins from 'vue-typed-mixins';
|
2023-05-05 01:41:54 -07:00
|
|
|
import { useWorkflowsStore } from '../stores/workflows.store';
|
|
|
|
import { useUsersStore } from '../stores/users.store';
|
|
|
|
import { useCredentialsStore } from '../stores/credentials.store';
|
|
|
|
import { useLogStreamingStore } from '../stores/logStreaming.store';
|
|
|
|
import { useSettingsStore } from '../stores/settings.store';
|
|
|
|
import { useUIStore } from '../stores/ui.store';
|
2023-01-04 00:47:48 -08:00
|
|
|
import { LOG_STREAM_MODAL_KEY, EnterpriseEditionFeature } from '../constants';
|
2023-04-24 03:18:24 -07:00
|
|
|
import type { MessageEventBusDestinationOptions } from 'n8n-workflow';
|
|
|
|
import { deepCopy, defaultMessageEventBusDestinationOptions } from 'n8n-workflow';
|
2023-01-04 00:47:48 -08:00
|
|
|
import PageViewLayout from '@/components/layouts/PageViewLayout.vue';
|
|
|
|
import EventDestinationCard from '@/components/SettingsLogStreaming/EventDestinationCard.ee.vue';
|
2023-04-06 06:32:45 -07:00
|
|
|
import { createEventBus } from '@/event-bus';
|
2023-01-04 00:47:48 -08:00
|
|
|
|
2023-01-20 03:08:40 -08:00
|
|
|
export default mixins().extend({
|
2023-01-04 00:47:48 -08:00
|
|
|
name: 'SettingsLogStreamingView',
|
|
|
|
props: {},
|
|
|
|
components: {
|
|
|
|
PageViewLayout,
|
|
|
|
EventDestinationCard,
|
|
|
|
},
|
|
|
|
data() {
|
|
|
|
return {
|
2023-04-06 06:32:45 -07:00
|
|
|
eventBus: createEventBus(),
|
2023-01-04 00:47:48 -08:00
|
|
|
destinations: Array<MessageEventBusDestinationOptions>,
|
|
|
|
disableLicense: false,
|
|
|
|
allDestinations: [] as MessageEventBusDestinationOptions[],
|
|
|
|
isInstanceOwner: false,
|
|
|
|
};
|
|
|
|
},
|
|
|
|
async mounted() {
|
|
|
|
if (!this.isLicensed) return;
|
|
|
|
|
|
|
|
this.isInstanceOwner = this.usersStore.currentUser?.globalRole?.name === 'owner';
|
|
|
|
// Prepare credentialsStore so modals can pick up credentials
|
|
|
|
await this.credentialsStore.fetchCredentialTypes(false);
|
|
|
|
await this.credentialsStore.fetchAllCredentials();
|
|
|
|
this.uiStore.nodeViewInitialized = false;
|
|
|
|
|
|
|
|
// fetch Destination data from the backend
|
2023-01-20 03:08:40 -08:00
|
|
|
await this.getDestinationDataFromBackend();
|
2023-01-04 00:47:48 -08:00
|
|
|
|
|
|
|
// since we are not really integrated into the hooks, we listen to the store and refresh the destinations
|
|
|
|
this.logStreamingStore.$onAction(({ name, after }) => {
|
|
|
|
if (name === 'removeDestination' || name === 'updateDestination') {
|
|
|
|
after(async () => {
|
|
|
|
this.$forceUpdate();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
// refresh when a modal closes
|
2023-04-20 03:26:14 -07:00
|
|
|
this.eventBus.on('destinationWasSaved', this.onDestinationWasSaved);
|
2023-01-04 00:47:48 -08:00
|
|
|
// listen to remove emission
|
2023-04-20 03:26:14 -07:00
|
|
|
this.eventBus.on('remove', this.onRemove);
|
2023-01-04 00:47:48 -08:00
|
|
|
// listen to modal closing and remove nodes from store
|
2023-04-20 03:26:14 -07:00
|
|
|
this.eventBus.on('closing', this.onBusClosing);
|
|
|
|
},
|
|
|
|
destroyed() {
|
|
|
|
this.eventBus.off('destinationWasSaved', this.onDestinationWasSaved);
|
|
|
|
this.eventBus.off('remove', this.onRemove);
|
|
|
|
this.eventBus.off('closing', this.onBusClosing);
|
2023-01-04 00:47:48 -08:00
|
|
|
},
|
|
|
|
computed: {
|
|
|
|
...mapStores(
|
|
|
|
useSettingsStore,
|
|
|
|
useLogStreamingStore,
|
|
|
|
useWorkflowsStore,
|
|
|
|
useUIStore,
|
|
|
|
useUsersStore,
|
|
|
|
useCredentialsStore,
|
|
|
|
),
|
|
|
|
sortedItemKeysByLabel() {
|
|
|
|
const sortedKeys: Array<{ label: string; key: string }> = [];
|
|
|
|
for (const [key, value] of Object.entries(this.logStreamingStore.items)) {
|
|
|
|
sortedKeys.push({ key, label: value.destination?.label ?? 'Destination' });
|
|
|
|
}
|
|
|
|
return sortedKeys.sort((a, b) => a.label.localeCompare(b.label));
|
|
|
|
},
|
|
|
|
environment() {
|
|
|
|
return process.env.NODE_ENV;
|
|
|
|
},
|
|
|
|
isLicensed(): boolean {
|
|
|
|
if (this.disableLicense === true) return false;
|
|
|
|
return this.settingsStore.isEnterpriseFeatureEnabled(EnterpriseEditionFeature.LogStreaming);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
methods: {
|
2023-04-20 03:26:14 -07:00
|
|
|
onDestinationWasSaved() {
|
|
|
|
this.$forceUpdate();
|
|
|
|
},
|
|
|
|
onBusClosing() {
|
|
|
|
this.workflowsStore.removeAllNodes({ setStateDirty: false, removePinData: true });
|
|
|
|
this.uiStore.stateIsDirty = false;
|
|
|
|
},
|
2023-01-20 03:08:40 -08:00
|
|
|
async getDestinationDataFromBackend(): Promise<void> {
|
2023-01-04 00:47:48 -08:00
|
|
|
this.logStreamingStore.clearEventNames();
|
|
|
|
this.logStreamingStore.clearDestinationItemTrees();
|
|
|
|
this.allDestinations = [];
|
2023-01-20 03:08:40 -08:00
|
|
|
const eventNamesData = await this.logStreamingStore.fetchEventNames();
|
2023-01-04 00:47:48 -08:00
|
|
|
if (eventNamesData) {
|
|
|
|
for (const eventName of eventNamesData) {
|
|
|
|
this.logStreamingStore.addEventName(eventName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const destinationData: MessageEventBusDestinationOptions[] =
|
2023-01-20 03:08:40 -08:00
|
|
|
await this.logStreamingStore.fetchDestinations();
|
2023-01-04 00:47:48 -08:00
|
|
|
if (destinationData) {
|
|
|
|
for (const destination of destinationData) {
|
|
|
|
this.logStreamingStore.addDestination(destination);
|
|
|
|
this.allDestinations.push(destination);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.$forceUpdate();
|
|
|
|
},
|
2023-04-24 06:37:36 -07:00
|
|
|
goToUpgrade() {
|
|
|
|
this.uiStore.goToUpgrade('log-streaming', 'upgrade-log-streaming');
|
2023-01-04 00:47:48 -08:00
|
|
|
},
|
|
|
|
storeHasItems(): boolean {
|
|
|
|
return this.logStreamingStore.items && Object.keys(this.logStreamingStore.items).length > 0;
|
|
|
|
},
|
|
|
|
async addDestination() {
|
|
|
|
const newDestination = deepCopy(defaultMessageEventBusDestinationOptions);
|
|
|
|
newDestination.id = uuid();
|
|
|
|
this.logStreamingStore.addDestination(newDestination);
|
|
|
|
this.uiStore.openModalWithData({
|
|
|
|
name: LOG_STREAM_MODAL_KEY,
|
|
|
|
data: {
|
|
|
|
destination: newDestination,
|
|
|
|
isNew: true,
|
|
|
|
eventBus: this.eventBus,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
|
|
|
async onRemove(destinationId?: string) {
|
|
|
|
if (!destinationId) return;
|
2023-01-20 03:08:40 -08:00
|
|
|
await this.logStreamingStore.deleteDestination(destinationId);
|
2023-01-04 00:47:48 -08:00
|
|
|
const foundNode = this.workflowsStore.getNodeByName(destinationId);
|
|
|
|
if (foundNode) {
|
|
|
|
this.workflowsStore.removeNode(foundNode);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
async onEdit(destinationId?: string) {
|
|
|
|
if (!destinationId) return;
|
|
|
|
const editDestination = this.logStreamingStore.getDestination(destinationId);
|
|
|
|
if (editDestination) {
|
|
|
|
this.uiStore.openModalWithData({
|
|
|
|
name: LOG_STREAM_MODAL_KEY,
|
|
|
|
data: {
|
|
|
|
destination: editDestination,
|
|
|
|
isNew: false,
|
|
|
|
eventBus: this.eventBus,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" module>
|
|
|
|
.header {
|
|
|
|
display: flex;
|
|
|
|
flex-direction: column;
|
|
|
|
align-items: flex-start;
|
|
|
|
white-space: nowrap;
|
|
|
|
|
|
|
|
*:first-child {
|
|
|
|
flex-grow: 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.destinationItem {
|
|
|
|
margin-bottom: 0.5em;
|
|
|
|
}
|
|
|
|
</style>
|