2024-08-14 06:20:02 -07:00
|
|
|
import type { Schema } from '@/Interface';
|
2024-11-05 10:47:52 -08:00
|
|
|
import { ApplicationError, type INodeExecutionData } from 'n8n-workflow';
|
2024-08-14 06:20:02 -07:00
|
|
|
import { useWorkflowsStore } from '@/stores/workflows.store';
|
|
|
|
import { useNDVStore } from '@/stores/ndv.store';
|
|
|
|
import { useDataSchema } from '@/composables/useDataSchema';
|
|
|
|
import { executionDataToJson } from '@/utils/nodeTypesUtils';
|
2024-11-05 10:47:52 -08:00
|
|
|
import { generateCodeForPrompt } from '../../api/ai';
|
|
|
|
import { useRootStore } from '../../stores/root.store';
|
|
|
|
import { type AskAiRequest } from '../../types/assistant.types';
|
|
|
|
import { useSettingsStore } from '../../stores/settings.store';
|
|
|
|
import { format } from 'prettier';
|
|
|
|
import jsParser from 'prettier/plugins/babel';
|
|
|
|
import * as estree from 'prettier/plugins/estree';
|
2024-08-14 06:20:02 -07:00
|
|
|
|
|
|
|
export function getParentNodes() {
|
|
|
|
const activeNode = useNDVStore().activeNode;
|
|
|
|
const { getCurrentWorkflow, getNodeByName } = useWorkflowsStore();
|
|
|
|
const workflow = getCurrentWorkflow();
|
|
|
|
|
|
|
|
if (!activeNode || !workflow) return [];
|
|
|
|
|
|
|
|
return workflow
|
|
|
|
.getParentNodesByDepth(activeNode?.name)
|
|
|
|
.filter(({ name }, i, nodes) => {
|
|
|
|
return name !== activeNode.name && nodes.findIndex((node) => node.name === name) === i;
|
|
|
|
})
|
|
|
|
.map((n) => getNodeByName(n.name))
|
|
|
|
.filter((n) => n !== null);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getSchemas() {
|
|
|
|
const parentNodes = getParentNodes();
|
|
|
|
const parentNodesNames = parentNodes.map((node) => node?.name);
|
|
|
|
const { getSchemaForExecutionData, getInputDataWithPinned } = useDataSchema();
|
|
|
|
const parentNodesSchemas: Array<{ nodeName: string; schema: Schema }> = parentNodes
|
|
|
|
.map((node) => {
|
|
|
|
const inputData: INodeExecutionData[] = getInputDataWithPinned(node);
|
|
|
|
|
|
|
|
return {
|
|
|
|
nodeName: node?.name || '',
|
|
|
|
schema: getSchemaForExecutionData(executionDataToJson(inputData), true),
|
|
|
|
};
|
|
|
|
})
|
|
|
|
.filter((node) => node.schema?.value.length > 0);
|
|
|
|
|
|
|
|
const inputSchema = parentNodesSchemas.shift();
|
|
|
|
|
|
|
|
return {
|
|
|
|
parentNodesNames,
|
|
|
|
inputSchema,
|
|
|
|
parentNodesSchemas,
|
|
|
|
};
|
|
|
|
}
|
2024-11-05 10:47:52 -08:00
|
|
|
|
|
|
|
export async function generateCodeForAiTransform(prompt: string, path: string) {
|
|
|
|
const schemas = getSchemas();
|
|
|
|
|
|
|
|
const payload: AskAiRequest.RequestPayload = {
|
|
|
|
question: prompt,
|
|
|
|
context: {
|
|
|
|
schema: schemas.parentNodesSchemas,
|
|
|
|
inputSchema: schemas.inputSchema!,
|
|
|
|
ndvPushRef: useNDVStore().pushRef,
|
|
|
|
pushRef: useRootStore().pushRef,
|
|
|
|
},
|
|
|
|
forNode: 'transform',
|
|
|
|
};
|
|
|
|
|
|
|
|
let value;
|
|
|
|
if (useSettingsStore().isAskAiEnabled) {
|
|
|
|
const { restApiContext } = useRootStore();
|
|
|
|
const { code } = await generateCodeForPrompt(restApiContext, payload);
|
|
|
|
value = code;
|
|
|
|
} else {
|
|
|
|
throw new ApplicationError('AI code generation is not enabled');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value === undefined) return;
|
|
|
|
|
|
|
|
const formattedCode = await format(String(value), {
|
|
|
|
parser: 'babel',
|
|
|
|
plugins: [jsParser, estree],
|
|
|
|
});
|
|
|
|
|
|
|
|
const updateInformation = {
|
|
|
|
name: path,
|
|
|
|
value: formattedCode,
|
|
|
|
};
|
|
|
|
|
|
|
|
return updateInformation;
|
|
|
|
}
|