mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-17 10:02:05 +00:00
feat(editor): Evaluations frontend (no-changelog) (#15550)
Co-authored-by: Yiorgis Gozadinos <yiorgis@n8n.io> Co-authored-by: JP van Oosten <jp@n8n.io> Co-authored-by: Giulio Andreini <g.andreini@gmail.com> Co-authored-by: Michael Kret <michael.k@radency.com>
This commit is contained in:
@@ -0,0 +1,162 @@
|
||||
<script setup lang="ts">
|
||||
import { useWorkflowsStore } from '@/stores/workflows.store';
|
||||
import { useUsageStore } from '@/stores/usage.store';
|
||||
import { useAsyncState } from '@vueuse/core';
|
||||
import { PLACEHOLDER_EMPTY_WORKFLOW_ID } from '@/constants';
|
||||
import { useCanvasOperations } from '@/composables/useCanvasOperations';
|
||||
import { useToast } from '@/composables/useToast';
|
||||
import { useI18n } from '@/composables/useI18n';
|
||||
import { useRouter } from 'vue-router';
|
||||
import { useNodeTypesStore } from '@/stores/nodeTypes.store';
|
||||
import { useEvaluationStore } from '@/stores/evaluation.store.ee';
|
||||
import { computed } from 'vue';
|
||||
import { N8nLink, N8nText } from '@n8n/design-system';
|
||||
import EvaluationsPaywall from '@/components/Evaluations.ee/Paywall/EvaluationsPaywall.vue';
|
||||
import SetupWizard from '@/components/Evaluations.ee/SetupWizard/SetupWizard.vue';
|
||||
|
||||
const props = defineProps<{
|
||||
name: string;
|
||||
}>();
|
||||
|
||||
const workflowsStore = useWorkflowsStore();
|
||||
const usageStore = useUsageStore();
|
||||
const evaluationStore = useEvaluationStore();
|
||||
const router = useRouter();
|
||||
const toast = useToast();
|
||||
const locale = useI18n();
|
||||
const nodeTypesStore = useNodeTypesStore();
|
||||
|
||||
const { initializeWorkspace } = useCanvasOperations({ router });
|
||||
|
||||
const evaluationsLicensed = computed(() => {
|
||||
return usageStore.workflowsWithEvaluationsLimit !== 0;
|
||||
});
|
||||
|
||||
const showWizard = computed(() => {
|
||||
const runs = Object.values(evaluationStore.testRunsById ?? {}).filter(
|
||||
({ workflowId }) => workflowId === props.name,
|
||||
);
|
||||
return runs.length === 0;
|
||||
});
|
||||
|
||||
// Method to run a test - will be used by the SetupWizard component
|
||||
async function runTest() {
|
||||
try {
|
||||
await evaluationStore.startTestRun(props.name);
|
||||
} catch (error) {
|
||||
toast.showError(error, locale.baseText('evaluation.listRuns.error.cantStartTestRun'));
|
||||
return;
|
||||
}
|
||||
try {
|
||||
await evaluationStore.fetchTestRuns(props.name);
|
||||
} catch (error) {
|
||||
toast.showError(error, locale.baseText('evaluation.listRuns.error.cantFetchTestRuns'));
|
||||
}
|
||||
}
|
||||
|
||||
const { isReady } = useAsyncState(async () => {
|
||||
try {
|
||||
await usageStore.getLicenseInfo();
|
||||
await evaluationStore.fetchTestRuns(props.name);
|
||||
} catch (error) {
|
||||
toast.showError(error, locale.baseText('evaluation.listRuns.error.cantFetchTestRuns'));
|
||||
}
|
||||
const workflowId = props.name;
|
||||
const isAlreadyInitialized = workflowsStore.workflow.id === workflowId;
|
||||
if (isAlreadyInitialized) return;
|
||||
|
||||
if (workflowId && workflowId !== 'new') {
|
||||
// Check if we are loading the Evaluation tab directly, without having loaded the workflow
|
||||
if (workflowsStore.workflow.id === PLACEHOLDER_EMPTY_WORKFLOW_ID) {
|
||||
try {
|
||||
const data = await workflowsStore.fetchWorkflow(workflowId);
|
||||
|
||||
// We need to check for the evaluation node with setMetrics operation, so we need to initialize the nodeTypesStore to have node properties initialized
|
||||
if (nodeTypesStore.allNodeTypes.length === 0) {
|
||||
await nodeTypesStore.getNodeTypes();
|
||||
}
|
||||
|
||||
initializeWorkspace(data);
|
||||
} catch (error) {
|
||||
toast.showError(error, locale.baseText('nodeView.showError.openWorkflow.title'));
|
||||
}
|
||||
}
|
||||
}
|
||||
}, undefined);
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div :class="$style.evaluationsView">
|
||||
<template v-if="isReady && showWizard">
|
||||
<div :class="$style.setupContent">
|
||||
<div>
|
||||
<N8nText size="large" color="text-dark" tag="h3" bold>
|
||||
{{ locale.baseText('evaluations.setupWizard.title') }}
|
||||
</N8nText>
|
||||
<N8nText tag="p" size="small" color="text-base" :class="$style.description">
|
||||
{{ locale.baseText('evaluations.setupWizard.description') }}
|
||||
<N8nLink size="small" href="https://docs.n8n.io/advanced-ai/evaluations/overview">{{
|
||||
locale.baseText('evaluations.setupWizard.moreInfo')
|
||||
}}</N8nLink>
|
||||
</N8nText>
|
||||
</div>
|
||||
|
||||
<div :class="$style.config">
|
||||
<iframe
|
||||
style="min-width: 500px"
|
||||
width="500"
|
||||
height="280"
|
||||
src="https://www.youtube.com/embed/5LlF196PKaE"
|
||||
title="n8n Evaluation quickstart"
|
||||
frameborder="0"
|
||||
allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share"
|
||||
referrerpolicy="strict-origin-when-cross-origin"
|
||||
allowfullscreen
|
||||
></iframe>
|
||||
<SetupWizard v-if="evaluationsLicensed" @run-test="runTest" />
|
||||
<EvaluationsPaywall v-else />
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
<router-view v-else-if="isReady" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style module lang="scss">
|
||||
.evaluationsView {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.setupContent {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: var(--spacing-l);
|
||||
max-width: 1024px;
|
||||
margin-top: var(--spacing-2xl);
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
.description {
|
||||
max-width: 600px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.config {
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
gap: var(--spacing-l);
|
||||
}
|
||||
|
||||
.setupDescription {
|
||||
margin-top: var(--spacing-2xs);
|
||||
|
||||
ul {
|
||||
li {
|
||||
margin-top: var(--spacing-2xs);
|
||||
}
|
||||
}
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user