feat(Simulate Node): New node (no-changelog) (#9109)

This commit is contained in:
Michael Kret 2024-05-08 14:02:36 +03:00 committed by GitHub
parent c4bf5b2b92
commit 6b6e8dfc33
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
15 changed files with 344 additions and 1 deletions

View file

@ -23,6 +23,7 @@ import type {
INodeTypes, INodeTypes,
IWorkflowExecuteAdditionalData, IWorkflowExecuteAdditionalData,
IExecuteData, IExecuteData,
IDataObject,
} from 'n8n-workflow'; } from 'n8n-workflow';
import { ICredentialsHelper, NodeHelpers, Workflow, ApplicationError } from 'n8n-workflow'; import { ICredentialsHelper, NodeHelpers, Workflow, ApplicationError } from 'n8n-workflow';
@ -57,6 +58,9 @@ const mockNodesData: INodeTypeData = {
}; };
const mockNodeTypes: INodeTypes = { const mockNodeTypes: INodeTypes = {
getKnownTypes(): IDataObject {
return {};
},
getByName(nodeType: string): INodeType | IVersionedNodeType { getByName(nodeType: string): INodeType | IVersionedNodeType {
return mockNodesData[nodeType]?.type; return mockNodesData[nodeType]?.type;
}, },

View file

@ -54,6 +54,10 @@ export class NodeTypes implements INodeTypes {
} }
} }
getKnownTypes() {
return this.loadNodesAndCredentials.knownNodes;
}
private getNode(type: string): LoadedClass<INodeType | IVersionedNodeType> { private getNode(type: string): LoadedClass<INodeType | IVersionedNodeType> {
const { loadedNodes, knownNodes } = this.loadNodesAndCredentials; const { loadedNodes, knownNodes } = this.loadNodesAndCredentials;
if (type in loadedNodes) { if (type in loadedNodes) {

View file

@ -23,6 +23,7 @@ import type {
INodeTypeData, INodeTypeData,
INodeTypes, INodeTypes,
ICredentialTestFunctions, ICredentialTestFunctions,
IDataObject,
} from 'n8n-workflow'; } from 'n8n-workflow';
import { import {
VersionedNodeType, VersionedNodeType,
@ -54,6 +55,9 @@ const mockNodesData: INodeTypeData = {
}; };
const mockNodeTypes: INodeTypes = { const mockNodeTypes: INodeTypes = {
getKnownTypes(): IDataObject {
return {};
},
getByName(nodeType: string): INodeType | IVersionedNodeType { getByName(nodeType: string): INodeType | IVersionedNodeType {
return mockNodesData[nodeType]?.type; return mockNodesData[nodeType]?.type;
}, },

View file

@ -2839,6 +2839,7 @@ const getCommonWorkflowFunctions = (
} }
return output; return output;
}, },
getKnownNodeTypes: () => workflow.nodeTypes.getKnownTypes(),
getRestApiUrl: () => additionalData.restApiUrl, getRestApiUrl: () => additionalData.restApiUrl,
getInstanceBaseUrl: () => additionalData.instanceBaseUrl, getInstanceBaseUrl: () => additionalData.instanceBaseUrl,
getInstanceId: () => Container.get(InstanceSettings).instanceId, getInstanceId: () => Container.get(InstanceSettings).instanceId,

View file

@ -98,7 +98,7 @@
<NodeIcon <NodeIcon
class="node-icon" class="node-icon"
:node-type="nodeType" :node-type="iconNodeType"
:size="40" :size="40"
:shrink="false" :shrink="false"
:color-default="iconColorDefault" :color-default="iconColorDefault"
@ -186,6 +186,8 @@ import {
MANUAL_TRIGGER_NODE_TYPE, MANUAL_TRIGGER_NODE_TYPE,
NODE_INSERT_SPACER_BETWEEN_INPUT_GROUPS, NODE_INSERT_SPACER_BETWEEN_INPUT_GROUPS,
NOT_DUPLICATABE_NODE_TYPES, NOT_DUPLICATABE_NODE_TYPES,
SIMULATE_NODE_TYPE,
SIMULATE_TRIGGER_NODE_TYPE,
WAIT_TIME_UNLIMITED, WAIT_TIME_UNLIMITED,
} from '@/constants'; } from '@/constants';
import { nodeBase } from '@/mixins/nodeBase'; import { nodeBase } from '@/mixins/nodeBase';
@ -586,6 +588,25 @@ export default defineComponent({
this.contextMenu.target.value.node.name === this.data?.name this.contextMenu.target.value.node.name === this.data?.name
); );
}, },
iconNodeType() {
if (
this.data?.type === SIMULATE_NODE_TYPE ||
this.data?.type === SIMULATE_TRIGGER_NODE_TYPE
) {
const icon = this.data.parameters?.icon as string;
const iconNodeType = this.workflow.expression.getSimpleParameterValue(
this.data,
icon,
'internal',
{},
);
if (iconNodeType && typeof iconNodeType === 'string') {
return this.nodeTypesStore.getNodeType(iconNodeType);
}
}
return this.nodeType;
},
}, },
watch: { watch: {
isActive(newValue, oldValue) { isActive(newValue, oldValue) {

View file

@ -184,6 +184,8 @@ export const COMPRESSION_NODE_TYPE = 'n8n-nodes-base.compression';
export const EDIT_IMAGE_NODE_TYPE = 'n8n-nodes-base.editImage'; export const EDIT_IMAGE_NODE_TYPE = 'n8n-nodes-base.editImage';
export const CHAIN_SUMMARIZATION_LANGCHAIN_NODE_TYPE = export const CHAIN_SUMMARIZATION_LANGCHAIN_NODE_TYPE =
'@n8n/n8n-nodes-langchain.chainSummarization'; '@n8n/n8n-nodes-langchain.chainSummarization';
export const SIMULATE_NODE_TYPE = 'n8n-nodes-base.simulate';
export const SIMULATE_TRIGGER_NODE_TYPE = 'n8n-nodes-base.simulateTrigger';
export const CREDENTIAL_ONLY_NODE_PREFIX = 'n8n-creds-base'; export const CREDENTIAL_ONLY_NODE_PREFIX = 'n8n-creds-base';
export const CREDENTIAL_ONLY_HTTP_NODE_VERSION = 4.1; export const CREDENTIAL_ONLY_HTTP_NODE_VERSION = 4.1;

View file

@ -0,0 +1,11 @@
{
"node": "n8n-nodes-base.simulate",
"nodeVersion": "1.0",
"codexVersion": "1.0",
"categories": ["Core Nodes"],
"resources": {},
"alias": ["placeholder", "stub", "dummy"],
"subcategories": {
"Core Nodes": ["Helpers"]
}
}

View file

@ -0,0 +1,131 @@
import { sleep, jsonParse, NodeOperationError } from 'n8n-workflow';
import type {
IExecuteFunctions,
INodeExecutionData,
INodeType,
INodeTypeDescription,
IDataObject,
} from 'n8n-workflow';
import { loadOptions } from './methods';
import {
executionDurationProperty,
iconSelector,
jsonOutputProperty,
subtitleProperty,
} from './descriptions';
export class Simulate implements INodeType {
description: INodeTypeDescription = {
displayName: 'Simulate',
hidden: true,
name: 'simulate',
group: ['organization'],
version: 1,
description: 'Simulate a node',
subtitle: '={{$parameter.subtitle || undefined}}',
icon: 'fa:arrow-right',
defaults: {
name: 'Simulate',
color: '#b0b0b0',
},
inputs: ['main'],
outputs: ['main'],
properties: [
iconSelector,
subtitleProperty,
{
displayName: 'Output',
name: 'output',
type: 'options',
default: 'all',
noDataExpression: true,
options: [
{
// eslint-disable-next-line n8n-nodes-base/node-param-display-name-miscased
name: 'Returns all input items',
value: 'all',
},
{
// eslint-disable-next-line n8n-nodes-base/node-param-display-name-miscased
name: 'Specify how many of input items to return',
value: 'specify',
},
{
// eslint-disable-next-line n8n-nodes-base/node-param-display-name-miscased
name: 'Specify output as JSON',
value: 'custom',
},
],
},
{
displayName: 'Number of Items',
name: 'numberOfItems',
type: 'number',
default: 1,
description:
'Number input of items to return, if greater then input length all items will be returned',
displayOptions: {
show: {
output: ['specify'],
},
},
typeOptions: {
minValue: 1,
},
},
{
...jsonOutputProperty,
displayOptions: {
show: {
output: ['custom'],
},
},
},
executionDurationProperty,
],
};
methods = { loadOptions };
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
const items = this.getInputData();
let returnItems: INodeExecutionData[] = [];
const output = this.getNodeParameter('output', 0) as string;
if (output === 'all') {
returnItems = items;
} else if (output === 'specify') {
const numberOfItems = this.getNodeParameter('numberOfItems', 0) as number;
returnItems = items.slice(0, numberOfItems);
} else if (output === 'custom') {
let jsonOutput = this.getNodeParameter('jsonOutput', 0);
if (typeof jsonOutput === 'string') {
try {
jsonOutput = jsonParse<IDataObject>(jsonOutput);
} catch (error) {
throw new NodeOperationError(this.getNode(), 'Invalid JSON');
}
}
if (!Array.isArray(jsonOutput)) {
jsonOutput = [jsonOutput];
}
for (const item of jsonOutput as IDataObject[]) {
returnItems.push({ json: item });
}
}
const executionDuration = this.getNodeParameter('executionDuration', 0) as number;
if (executionDuration > 0) {
await sleep(executionDuration);
}
return [returnItems];
}
}

View file

@ -0,0 +1,11 @@
{
"node": "n8n-nodes-base.simulateTrigger",
"nodeVersion": "1.0",
"codexVersion": "1.0",
"categories": ["Core Nodes"],
"resources": {},
"alias": ["placeholder", "stub", "dummy"],
"subcategories": {
"Core Nodes": ["Helpers"]
}
}

View file

@ -0,0 +1,79 @@
import { sleep, NodeOperationError, jsonParse } from 'n8n-workflow';
import type {
IDataObject,
ITriggerFunctions,
INodeExecutionData,
INodeType,
INodeTypeDescription,
ITriggerResponse,
} from 'n8n-workflow';
import {
executionDurationProperty,
iconSelector,
jsonOutputProperty,
subtitleProperty,
} from './descriptions';
import { loadOptions } from './methods';
export class SimulateTrigger implements INodeType {
description: INodeTypeDescription = {
hidden: true,
displayName: 'Simulate Trigger',
name: 'simulateTrigger',
subtitle: '={{$parameter.subtitle || undefined}}',
icon: 'fa:arrow-right',
group: ['trigger'],
version: 1,
description: 'Simulate a trigger node',
defaults: {
name: 'Simulate Trigger',
color: '#b0b0b0',
},
inputs: [],
outputs: ['main'],
properties: [
{ ...iconSelector, default: 'n8n-nodes-base.manualTrigger' },
subtitleProperty,
{ ...jsonOutputProperty, displayName: 'Output (JSON)' },
executionDurationProperty,
],
};
methods = { loadOptions };
async trigger(this: ITriggerFunctions): Promise<ITriggerResponse> {
const returnItems: INodeExecutionData[] = [];
let jsonOutput = this.getNodeParameter('jsonOutput', 0);
if (typeof jsonOutput === 'string') {
try {
jsonOutput = jsonParse<IDataObject>(jsonOutput);
} catch (error) {
throw new NodeOperationError(this.getNode(), 'Invalid JSON');
}
}
if (!Array.isArray(jsonOutput)) {
jsonOutput = [jsonOutput];
}
for (const item of jsonOutput as IDataObject[]) {
returnItems.push({ json: item });
}
const executionDuration = this.getNodeParameter('executionDuration', 0) as number;
if (executionDuration > 0) {
await sleep(executionDuration);
}
const manualTriggerFunction = async () => {
this.emit([returnItems]);
};
return {
manualTriggerFunction,
};
}
}

View file

@ -0,0 +1,44 @@
import type { INodeProperties } from 'n8n-workflow';
export const iconSelector: INodeProperties = {
// eslint-disable-next-line n8n-nodes-base/node-param-display-name-wrong-for-dynamic-options
displayName: 'Icon to Display on Canvas',
name: 'icon',
type: 'options',
// eslint-disable-next-line n8n-nodes-base/node-param-description-wrong-for-dynamic-options
description: 'Select a type of node to show corresponding icon',
default: 'n8n-nodes-base.noOp',
typeOptions: {
loadOptionsMethod: 'getNodeTypes',
},
};
export const subtitleProperty: INodeProperties = {
displayName: 'Subtitle',
name: 'subtitle',
type: 'string',
default: '',
placeholder: "e.g. 'record: read'",
};
export const jsonOutputProperty: INodeProperties = {
displayName: 'JSON',
name: 'jsonOutput',
type: 'json',
typeOptions: {
rows: 5,
},
default: '[\n {\n "my_field_1": "value",\n "my_field_2": 1\n }\n]',
validateType: 'array',
};
export const executionDurationProperty: INodeProperties = {
displayName: 'Execution Duration (MS)',
name: 'executionDuration',
type: 'number',
default: 150,
description: 'Execution duration in milliseconds',
typeOptions: {
minValue: 0,
},
};

View file

@ -0,0 +1 @@
export * as loadOptions from './loadOptions';

View file

@ -0,0 +1,26 @@
import type { ILoadOptionsFunctions, INodePropertyOptions } from 'n8n-workflow';
export async function getNodeTypes(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
const types = this.getKnownNodeTypes() as {
[key: string]: {
className: string;
};
};
const returnData: INodePropertyOptions[] = [];
let typeNames = Object.keys(types);
if (this.getNode().type === 'n8n-nodes-base.simulateTrigger') {
typeNames = typeNames.filter((type) => type.toLowerCase().includes('trigger'));
}
for (const type of typeNames) {
returnData.push({
name: types[type].className,
value: type,
});
}
return returnData;
}

View file

@ -717,6 +717,8 @@
"dist/nodes/Shopify/Shopify.node.js", "dist/nodes/Shopify/Shopify.node.js",
"dist/nodes/Shopify/ShopifyTrigger.node.js", "dist/nodes/Shopify/ShopifyTrigger.node.js",
"dist/nodes/Signl4/Signl4.node.js", "dist/nodes/Signl4/Signl4.node.js",
"dist/nodes/Simulate/Simulate.node.js",
"dist/nodes/Simulate/SimulateTrigger.node.js",
"dist/nodes/Slack/Slack.node.js", "dist/nodes/Slack/Slack.node.js",
"dist/nodes/Sms77/Sms77.node.js", "dist/nodes/Sms77/Sms77.node.js",
"dist/nodes/Snowflake/Snowflake.node.js", "dist/nodes/Snowflake/Snowflake.node.js",

View file

@ -815,6 +815,7 @@ export interface FunctionsBase {
getInstanceId(): string; getInstanceId(): string;
getChildNodes(nodeName: string): NodeTypeAndVersion[]; getChildNodes(nodeName: string): NodeTypeAndVersion[];
getParentNodes(nodeName: string): NodeTypeAndVersion[]; getParentNodes(nodeName: string): NodeTypeAndVersion[];
getKnownNodeTypes(): IDataObject;
getMode?: () => WorkflowExecuteMode; getMode?: () => WorkflowExecuteMode;
getActivationMode?: () => WorkflowActivateMode; getActivationMode?: () => WorkflowActivateMode;
@ -1839,6 +1840,7 @@ export type WebhookResponseMode = 'onReceived' | 'lastNode' | 'responseNode';
export interface INodeTypes { export interface INodeTypes {
getByName(nodeType: string): INodeType | IVersionedNodeType; getByName(nodeType: string): INodeType | IVersionedNodeType;
getByNameAndVersion(nodeType: string, version?: number): INodeType; getByNameAndVersion(nodeType: string, version?: number): INodeType;
getKnownTypes(): IDataObject;
} }
export type LoadingDetails = { export type LoadingDetails = {