mirror of
https://github.com/n8n-io/n8n.git
synced 2024-11-14 08:34:07 -08:00
dae01f3abe
* feat(ExecuteWorkflowTrigger node): Implement ExecuteWorkflowTrigger node (#4108) * feat(ExecuteWorkflowTrigger node): Implement ExecuteWorkflowTrigger node * feat(editor): Do not show duplicate button if canvas contains `maxNodes` amount of nodes * feat(ManualTrigger node): Implement ManualTrigger node (#4110) * feat(ManualTrigger node): Implement ManualTrigger node * 📝 Remove generics doc items from ManualTrigger node * feat(editor-ui): Trigger tab redesign (#4150) * 🚧 Begin with TriggerPanel implementation, add Other Trigger Nodes subcategory * 🚧 Extracted categorized categories/subcategory/nodes rendering into its own component — CategorizedItems, removed SubcategoryPanel, added translations * ✨ Implement MainPanel background scrim * ♻️ Move `categoriesWithNodes`, 'visibleNodeTypes` and 'categorizedItems` to store, implemented dynamic categories count based on `selectedType` * 🐛 Fix SlideTransition for all the NodeCreato panels * 💄 Fix cursos for CategoryItem and NodeItem * 🐛 Make sure ALL_NODE_FILTER is always set when MainPanel is mounted * 🎨 Address PR comments * label: Use Array type for CategorizedItems props * 🏷️ Add proper types for Vue props * 🎨 Use standard component registration for CategorizedItems inside TriggerHelperPanel * 🎨 Use kebab case for main-panel and icon component * 🏷️ Improve types * feat(editor-ui): Redesign search input inside node creator panel (#4204) * 🚧 Begin with TriggerPanel implementation, add Other Trigger Nodes subcategory * 🚧 Extracted categorized categories/subcategory/nodes rendering into its own component — CategorizedItems, removed SubcategoryPanel, added translations * ✨ Implement MainPanel background scrim * ♻️ Move `categoriesWithNodes`, 'visibleNodeTypes` and 'categorizedItems` to store, implemented dynamic categories count based on `selectedType` * 🐛 Fix SlideTransition for all the NodeCreato panels * 💄 Fix cursos for CategoryItem and NodeItem * 🐛 Make sure ALL_NODE_FILTER is always set when MainPanel is mounted * 🎨 Address PR comments * label: Use Array type for CategorizedItems props * 🏷️ Add proper types for Vue props * 🎨 Use standard component registration for CategorizedItems inside TriggerHelperPanel * ✨ Redesign search input and unify usage of categorized items * 🏷️ Use lowercase "Boolean" as `isSearchVisible` computed return type * 🔥 Remove useless emit * ✨ Implement no result view based on subcategory, minor fixes * 🎨 Remove unused properties * feat(node-email): Change EmailReadImap display name and name (#4239) * feat(editor-ui): Implement "Choose a Triger" action and related behaviour (#4226) * ✨ Implement "Choose a Triger" action and related behaviour * 🔇 Lint fix * ♻️ Remove PlaceholderTrigger node, add a button instead * 🎨 Merge onMouseEnter and onMouseLeave to a single function * 💡 Add comment * 🔥 Remove PlaceholderNode registration * 🎨 Rename TriggerPlaceholderButton to CanvasAddButton * ✨ Add method to unregister custom action and rework CanvasAddButton centering logic * 🎨 Run `setRecenteredCanvasAddButtonPosition` on `CanvasAddButton` mount * fix(editor): Fix selecting of node from node-creator panel by clicking * 🔀 Merge fixes * fix(editor): Show execute workflow trigger instead of workflow trigger in the trigger helper panel * feat(editor): Fix node creator panel slide transition (#4261) * fix(editor): Fix node creator panel slide-in/slide-out transitions * 🎨 Fix naming * 🎨 Use kebab-case for transition component name * feat(editor): Disable execution and show notice when user tries to run workflow without enabled triggers * fix(editor): Address first batch of new WF experience review (#4279) * fix(editor): Fix first batch of review items * bug(editor): Fix nodeview canvas add button centering * 🔇 Fix linter errors * bug(ManualTrigger Node): Fix manual trigger node execution * fix(editor): Do not show canvas add button in execution or demo mode and prevent clicking if creator is open * fix(editor): do not show pin data tooltip for manual trigger node * fix(editor): do not use nodeViewOffset on zoomToFit * 💄 Add margin for last node creator item and set font-weight to 700 for category title * ✨ Position welcome note next to the added trigger node * 🐛 Remve always true welcome note * feat(editor): Minor UI and UX tweaks (#4328) * 💄 Make top viewport buttons less prominent * ✨ Allow user to switch to all tabs if it contains filter results, move nodecreator state props to its own module * 🔇 Fix linting errors * 🔇 Fix linting errors * 🔇 Fix linting errors * chore(build): Ping Turbo version to 1.5.5 * 💄 Minor traigger panel and node view style changes * 💬 Update display name of execute workflow trigger * feat(core, editor): Update subworkflow execution logic (#4269) * ✨ Implement `findWorkflowStart` * ⚡ Extend `WorkflowOperationError` * ⚡ Add `WorkflowOperationError` to toast * 📘 Extend interface * ✨ Add `subworkflowExecutionError` to store * ✨ Create `SubworkflowOperationError` * ⚡ Render subworkflow error as node error * 🚚 Move subworkflow start validation to `cli` * ⚡ Reset subworkflow execution error state * 🔥 Remove unused import * ⚡ Adjust CLI commands * 🔥 Remove unneeded check * 🔥 Remove stray log * ⚡ Simplify syntax * ⚡ Sort in case both Start and EWT present * ♻️ Address Omar's feedback * 🔥 Remove unneeded lint exception * ✏️ Fix copy * 👕 Fix lint * fix: moved find start node function to catchable place Co-authored-by: Omar Ajoue <krynble@gmail.com> * 💄 Change ExecuteWorkflow node to primary * ✨ Allow user to navigate to all tab if it contains search results * 🐛 Fixed canvas control button while in demo, disable workflow activation for non-activavle nodes and revert zoomToFit bottom offset * :fix: Do not chow request text if there's results * 💬 Update noResults text Co-authored-by: Iván Ovejero <ivov.src@gmail.com> Co-authored-by: Omar Ajoue <krynble@gmail.com>
206 lines
5.1 KiB
TypeScript
206 lines
5.1 KiB
TypeScript
import { readFile as fsReadFile } from 'fs/promises';
|
|
|
|
import { IExecuteFunctions } from 'n8n-core';
|
|
import {
|
|
IExecuteWorkflowInfo,
|
|
INodeExecutionData,
|
|
INodeType,
|
|
INodeTypeDescription,
|
|
IWorkflowBase,
|
|
NodeOperationError,
|
|
} from 'n8n-workflow';
|
|
|
|
export class ExecuteWorkflow implements INodeType {
|
|
description: INodeTypeDescription = {
|
|
displayName: 'Execute Workflow',
|
|
name: 'executeWorkflow',
|
|
icon: 'fa:sign-in-alt',
|
|
group: ['transform'],
|
|
version: 1,
|
|
subtitle: '={{"Workflow: " + $parameter["workflowId"]}}',
|
|
description: 'Execute another workflow',
|
|
defaults: {
|
|
name: 'Execute Workflow',
|
|
color: '#ff6d5a',
|
|
},
|
|
inputs: ['main'],
|
|
outputs: ['main'],
|
|
properties: [
|
|
{
|
|
displayName: 'Source',
|
|
name: 'source',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'Database',
|
|
value: 'database',
|
|
description: 'Load the workflow from the database by ID',
|
|
},
|
|
{
|
|
name: 'Local File',
|
|
value: 'localFile',
|
|
description: 'Load the workflow from a locally saved file',
|
|
},
|
|
{
|
|
name: 'Parameter',
|
|
value: 'parameter',
|
|
description: 'Load the workflow from a parameter',
|
|
},
|
|
{
|
|
name: 'URL',
|
|
value: 'url',
|
|
description: 'Load the workflow from an URL',
|
|
},
|
|
],
|
|
default: 'database',
|
|
description: 'Where to get the workflow to execute from',
|
|
},
|
|
|
|
// ----------------------------------
|
|
// source:database
|
|
// ----------------------------------
|
|
{
|
|
displayName: 'Workflow ID',
|
|
name: 'workflowId',
|
|
type: 'string',
|
|
displayOptions: {
|
|
show: {
|
|
source: ['database'],
|
|
},
|
|
},
|
|
default: '',
|
|
required: true,
|
|
description: 'The workflow to execute',
|
|
},
|
|
|
|
// ----------------------------------
|
|
// source:localFile
|
|
// ----------------------------------
|
|
{
|
|
displayName: 'Workflow Path',
|
|
name: 'workflowPath',
|
|
type: 'string',
|
|
displayOptions: {
|
|
show: {
|
|
source: ['localFile'],
|
|
},
|
|
},
|
|
default: '',
|
|
placeholder: '/data/workflow.json',
|
|
required: true,
|
|
description: 'The path to local JSON workflow file to execute',
|
|
},
|
|
|
|
// ----------------------------------
|
|
// source:parameter
|
|
// ----------------------------------
|
|
{
|
|
displayName: 'Workflow JSON',
|
|
name: 'workflowJson',
|
|
type: 'string',
|
|
typeOptions: {
|
|
alwaysOpenEditWindow: true,
|
|
editor: 'json',
|
|
rows: 10,
|
|
},
|
|
displayOptions: {
|
|
show: {
|
|
source: ['parameter'],
|
|
},
|
|
},
|
|
default: '\n\n\n',
|
|
required: true,
|
|
description: 'The workflow JSON code to execute',
|
|
},
|
|
|
|
// ----------------------------------
|
|
// source:url
|
|
// ----------------------------------
|
|
{
|
|
displayName: 'Workflow URL',
|
|
name: 'workflowUrl',
|
|
type: 'string',
|
|
displayOptions: {
|
|
show: {
|
|
source: ['url'],
|
|
},
|
|
},
|
|
default: '',
|
|
placeholder: 'https://example.com/workflow.json',
|
|
required: true,
|
|
description: 'The URL from which to load the workflow from',
|
|
},
|
|
{
|
|
displayName:
|
|
'Any data you pass into this node will be output by the start node of the workflow to be executed. <a href="https://docs.n8n.io/integrations/builtin/core-nodes/n8n-nodes-base.executeworkflow/" target="_blank">More info</a>',
|
|
name: 'executeWorkflowNotice',
|
|
type: 'notice',
|
|
default: '',
|
|
},
|
|
],
|
|
};
|
|
|
|
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
|
const items = this.getInputData();
|
|
const source = this.getNodeParameter('source', 0) as string;
|
|
|
|
const workflowInfo: IExecuteWorkflowInfo = {};
|
|
|
|
try {
|
|
if (source === 'database') {
|
|
// Read workflow from database
|
|
workflowInfo.id = this.getNodeParameter('workflowId', 0) as string;
|
|
} else if (source === 'localFile') {
|
|
// Read workflow from filesystem
|
|
const workflowPath = this.getNodeParameter('workflowPath', 0) as string;
|
|
|
|
let workflowJson;
|
|
try {
|
|
workflowJson = (await fsReadFile(workflowPath, { encoding: 'utf8' })) as string;
|
|
} catch (error) {
|
|
if (error.code === 'ENOENT') {
|
|
throw new NodeOperationError(
|
|
this.getNode(),
|
|
`The file "${workflowPath}" could not be found.`,
|
|
);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
|
|
workflowInfo.code = JSON.parse(workflowJson) as IWorkflowBase;
|
|
} else if (source === 'parameter') {
|
|
// Read workflow from parameter
|
|
const workflowJson = this.getNodeParameter('workflowJson', 0) as string;
|
|
workflowInfo.code = JSON.parse(workflowJson) as IWorkflowBase;
|
|
} else if (source === 'url') {
|
|
// Read workflow from url
|
|
const workflowUrl = this.getNodeParameter('workflowUrl', 0) as string;
|
|
|
|
const requestOptions = {
|
|
headers: {
|
|
accept: 'application/json,text/*;q=0.99',
|
|
},
|
|
method: 'GET',
|
|
uri: workflowUrl,
|
|
json: true,
|
|
gzip: true,
|
|
};
|
|
|
|
const response = await this.helpers.request(requestOptions);
|
|
workflowInfo.code = response;
|
|
}
|
|
|
|
const receivedData = await this.executeWorkflow(workflowInfo, items);
|
|
|
|
return receivedData;
|
|
} catch (error) {
|
|
if (this.continueOnFail()) {
|
|
return this.prepareOutputData([{ json: { error: error.message } }]);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
}
|
|
}
|