From 61ac0c77755210f570b887951fe6bbec1a323811 Mon Sep 17 00:00:00 2001 From: Michael Kret <88898367+michael-radency@users.noreply.github.com> Date: Mon, 19 Aug 2024 19:01:33 +0300 Subject: [PATCH] fix: Filter component - improve errors (#10456) --- .../src/components/Error/NodeErrorView.vue | 14 ++--- .../nodes-base/nodes/Filter/Filter.node.ts | 2 + .../nodes/Filter/V2/FilterV2.node.ts | 31 ++++++++--- packages/nodes-base/nodes/If/If.node.ts | 3 +- packages/nodes-base/nodes/If/V2/IfV2.node.ts | 31 ++++++++--- packages/nodes-base/nodes/If/V2/utils.ts | 15 ++++++ .../nodes-base/nodes/Switch/Switch.node.ts | 3 +- .../nodes/Switch/V3/SwitchV3.node.ts | 36 +++++++++---- packages/nodes-base/utils/descriptions.ts | 8 +++ .../src/NodeParameters/FilterParameter.ts | 53 +++++++++++++++++-- 10 files changed, 157 insertions(+), 39 deletions(-) create mode 100644 packages/nodes-base/nodes/If/V2/utils.ts diff --git a/packages/editor-ui/src/components/Error/NodeErrorView.vue b/packages/editor-ui/src/components/Error/NodeErrorView.vue index 2aff0307af5d7..7114323acf5c6 100644 --- a/packages/editor-ui/src/components/Error/NodeErrorView.vue +++ b/packages/editor-ui/src/components/Error/NodeErrorView.vue @@ -216,17 +216,13 @@ function getErrorDescription(): string { function addItemIndexSuffix(message: string): string { let itemIndexSuffix = ''; - const ITEM_INDEX_SUFFIX_TEXT = '[item '; - - if ( - hasManyInputItems.value && - !message.includes(ITEM_INDEX_SUFFIX_TEXT) && - props.error?.context?.itemIndex !== undefined - ) { - itemIndexSuffix = ` [item ${props.error.context.itemIndex}]`; + if (hasManyInputItems.value && props.error?.context?.itemIndex !== undefined) { + itemIndexSuffix = `item ${props.error.context.itemIndex}`; } - return message + itemIndexSuffix; + if (message.includes(itemIndexSuffix)) return message; + + return `${message} [${itemIndexSuffix}]`; } function getErrorMessage(): string { diff --git a/packages/nodes-base/nodes/Filter/Filter.node.ts b/packages/nodes-base/nodes/Filter/Filter.node.ts index 69a46d4a4f807..7eb36ca4f0565 100644 --- a/packages/nodes-base/nodes/Filter/Filter.node.ts +++ b/packages/nodes-base/nodes/Filter/Filter.node.ts @@ -13,11 +13,13 @@ export class Filter extends VersionedNodeType { iconColor: 'light-blue', group: ['transform'], description: 'Remove items matching a condition', + defaultVersion: 2.1, }; const nodeVersions: IVersionedNodeType['nodeVersions'] = { 1: new FilterV1(baseDescription), 2: new FilterV2(baseDescription), + 2.1: new FilterV2(baseDescription), }; super(nodeVersions, baseDescription); diff --git a/packages/nodes-base/nodes/Filter/V2/FilterV2.node.ts b/packages/nodes-base/nodes/Filter/V2/FilterV2.node.ts index 16e95ef634d7c..06df074cd539b 100644 --- a/packages/nodes-base/nodes/Filter/V2/FilterV2.node.ts +++ b/packages/nodes-base/nodes/Filter/V2/FilterV2.node.ts @@ -9,6 +9,8 @@ import { type INodeTypeDescription, } from 'n8n-workflow'; import { ENABLE_LESS_STRICT_TYPE_VALIDATION } from '../../../utils/constants'; +import { getTypeValidationParameter, getTypeValidationStrictness } from '../../If/V2/utils'; +import { looseTypeValidationProperty } from '../../../utils/descriptions'; export class FilterV2 implements INodeType { description: INodeTypeDescription; @@ -16,7 +18,7 @@ export class FilterV2 implements INodeType { constructor(baseDescription: INodeTypeBaseDescription) { this.description = { ...baseDescription, - version: 2, + version: [2, 2.1], defaults: { name: 'Filter', color: '#229eff', @@ -35,7 +37,16 @@ export class FilterV2 implements INodeType { typeOptions: { filter: { caseSensitive: '={{!$parameter.options.ignoreCase}}', - typeValidation: '={{$parameter.options.looseTypeValidation ? "loose" : "strict"}}', + typeValidation: getTypeValidationStrictness(2.1), + }, + }, + }, + { + ...looseTypeValidationProperty, + default: false, + displayOptions: { + show: { + '@version': [{ _cnd: { gte: 2.1 } }], }, }, }, @@ -54,11 +65,12 @@ export class FilterV2 implements INodeType { default: true, }, { - displayName: 'Less Strict Type Validation', - description: 'Whether to try casting value types based on the selected operator', - name: 'looseTypeValidation', - type: 'boolean', - default: true, + ...looseTypeValidationProperty, + displayOptions: { + show: { + '@version': [{ _cnd: { lt: 2.1 } }], + }, + }, }, ], }, @@ -82,7 +94,10 @@ export class FilterV2 implements INodeType { extractValue: true, }) as boolean; } catch (error) { - if (!options.looseTypeValidation && !error.description) { + if ( + !getTypeValidationParameter(2.1)(this, itemIndex, options.looseTypeValidation) && + !error.description + ) { set(error, 'description', ENABLE_LESS_STRICT_TYPE_VALIDATION); } set(error, 'context.itemIndex', itemIndex); diff --git a/packages/nodes-base/nodes/If/If.node.ts b/packages/nodes-base/nodes/If/If.node.ts index 7bc1d459ca8e4..4942fdebf278a 100644 --- a/packages/nodes-base/nodes/If/If.node.ts +++ b/packages/nodes-base/nodes/If/If.node.ts @@ -13,12 +13,13 @@ export class If extends VersionedNodeType { iconColor: 'green', group: ['transform'], description: 'Route items to different branches (true/false)', - defaultVersion: 2, + defaultVersion: 2.1, }; const nodeVersions: IVersionedNodeType['nodeVersions'] = { 1: new IfV1(baseDescription), 2: new IfV2(baseDescription), + 2.1: new IfV2(baseDescription), }; super(nodeVersions, baseDescription); diff --git a/packages/nodes-base/nodes/If/V2/IfV2.node.ts b/packages/nodes-base/nodes/If/V2/IfV2.node.ts index 31fb258e76ebe..691fc2e0b9016 100644 --- a/packages/nodes-base/nodes/If/V2/IfV2.node.ts +++ b/packages/nodes-base/nodes/If/V2/IfV2.node.ts @@ -9,6 +9,8 @@ import { type INodeTypeDescription, } from 'n8n-workflow'; import { ENABLE_LESS_STRICT_TYPE_VALIDATION } from '../../../utils/constants'; +import { looseTypeValidationProperty } from '../../../utils/descriptions'; +import { getTypeValidationParameter, getTypeValidationStrictness } from './utils'; export class IfV2 implements INodeType { description: INodeTypeDescription; @@ -16,7 +18,7 @@ export class IfV2 implements INodeType { constructor(baseDescription: INodeTypeBaseDescription) { this.description = { ...baseDescription, - version: 2, + version: [2, 2.1], defaults: { name: 'If', color: '#408000', @@ -35,7 +37,16 @@ export class IfV2 implements INodeType { typeOptions: { filter: { caseSensitive: '={{!$parameter.options.ignoreCase}}', - typeValidation: '={{$parameter.options.looseTypeValidation ? "loose" : "strict"}}', + typeValidation: getTypeValidationStrictness(2.1), + }, + }, + }, + { + ...looseTypeValidationProperty, + default: false, + displayOptions: { + show: { + '@version': [{ _cnd: { gte: 2.1 } }], }, }, }, @@ -54,11 +65,12 @@ export class IfV2 implements INodeType { default: true, }, { - displayName: 'Less Strict Type Validation', - description: 'Whether to try casting value types based on the selected operator', - name: 'looseTypeValidation', - type: 'boolean', - default: true, + ...looseTypeValidationProperty, + displayOptions: { + show: { + '@version': [{ _cnd: { lt: 2.1 } }], + }, + }, }, ], }, @@ -82,7 +94,10 @@ export class IfV2 implements INodeType { extractValue: true, }) as boolean; } catch (error) { - if (!options.looseTypeValidation && !error.description) { + if ( + !getTypeValidationParameter(2.1)(this, itemIndex, options.looseTypeValidation) && + !error.description + ) { set(error, 'description', ENABLE_LESS_STRICT_TYPE_VALIDATION); } set(error, 'context.itemIndex', itemIndex); diff --git a/packages/nodes-base/nodes/If/V2/utils.ts b/packages/nodes-base/nodes/If/V2/utils.ts new file mode 100644 index 0000000000000..21af3724218c6 --- /dev/null +++ b/packages/nodes-base/nodes/If/V2/utils.ts @@ -0,0 +1,15 @@ +import type { IExecuteFunctions } from 'n8n-workflow'; + +export const getTypeValidationStrictness = (version: number) => { + return `={{ ($nodeVersion < ${version} ? $parameter.options.looseTypeValidation : $parameter.looseTypeValidation) ? "loose" : "strict" }}`; +}; + +export const getTypeValidationParameter = (version: number) => { + return (context: IExecuteFunctions, itemIndex: number, option: boolean | undefined) => { + if (context.getNode().typeVersion < version) { + return option; + } else { + return context.getNodeParameter('looseTypeValidation', itemIndex, false) as boolean; + } + }; +}; diff --git a/packages/nodes-base/nodes/Switch/Switch.node.ts b/packages/nodes-base/nodes/Switch/Switch.node.ts index c54f5b55da624..fb7262f742fa7 100644 --- a/packages/nodes-base/nodes/Switch/Switch.node.ts +++ b/packages/nodes-base/nodes/Switch/Switch.node.ts @@ -14,13 +14,14 @@ export class Switch extends VersionedNodeType { iconColor: 'light-blue', group: ['transform'], description: 'Route items depending on defined expression or rules', - defaultVersion: 3, + defaultVersion: 3.1, }; const nodeVersions: IVersionedNodeType['nodeVersions'] = { 1: new SwitchV1(baseDescription), 2: new SwitchV2(baseDescription), 3: new SwitchV3(baseDescription), + 3.1: new SwitchV3(baseDescription), }; super(nodeVersions, baseDescription); diff --git a/packages/nodes-base/nodes/Switch/V3/SwitchV3.node.ts b/packages/nodes-base/nodes/Switch/V3/SwitchV3.node.ts index 2c54fb860efe3..c4ac82c990948 100644 --- a/packages/nodes-base/nodes/Switch/V3/SwitchV3.node.ts +++ b/packages/nodes-base/nodes/Switch/V3/SwitchV3.node.ts @@ -13,6 +13,8 @@ import { ApplicationError, NodeConnectionType, NodeOperationError } from 'n8n-wo import set from 'lodash/set'; import { capitalize } from '@utils/utilities'; import { ENABLE_LESS_STRICT_TYPE_VALIDATION } from '../../../utils/constants'; +import { looseTypeValidationProperty } from '../../../utils/descriptions'; +import { getTypeValidationParameter, getTypeValidationStrictness } from '../../If/V2/utils'; const configuredOutputs = (parameters: INodeParameters) => { const mode = parameters.mode as string; @@ -48,7 +50,7 @@ export class SwitchV3 implements INodeType { this.description = { ...baseDescription, subtitle: `=mode: {{(${capitalize})($parameter["mode"])}}`, - version: [3], + version: [3, 3.1], defaults: { name: 'Switch', color: '#506000', @@ -157,8 +159,7 @@ export class SwitchV3 implements INodeType { multipleValues: false, filter: { caseSensitive: '={{!$parameter.options.ignoreCase}}', - typeValidation: - '={{$parameter.options.looseTypeValidation ? "loose" : "strict"}}', + typeValidation: getTypeValidationStrictness(3.1), }, }, }, @@ -184,6 +185,15 @@ export class SwitchV3 implements INodeType { }, ], }, + { + ...looseTypeValidationProperty, + default: false, + displayOptions: { + show: { + '@version': [{ _cnd: { gte: 3.1 } }], + }, + }, + }, { displayName: 'Options', name: 'options', @@ -218,11 +228,12 @@ export class SwitchV3 implements INodeType { default: true, }, { - displayName: 'Less Strict Type Validation', - description: 'Whether to try casting value types based on the selected operator', - name: 'looseTypeValidation', - type: 'boolean', - default: true, + ...looseTypeValidationProperty, + displayOptions: { + show: { + '@version': [{ _cnd: { lt: 3.1 } }], + }, + }, }, { displayName: 'Rename Fallback Output', @@ -349,7 +360,14 @@ export class SwitchV3 implements INodeType { }, ) as boolean; } catch (error) { - if (!options.looseTypeValidation && !error.description) { + if ( + !getTypeValidationParameter(3.1)( + this, + itemIndex, + options.looseTypeValidation as boolean, + ) && + !error.description + ) { error.description = ENABLE_LESS_STRICT_TYPE_VALIDATION; } set(error, 'context.itemIndex', itemIndex); diff --git a/packages/nodes-base/utils/descriptions.ts b/packages/nodes-base/utils/descriptions.ts index 1465d3e31476c..fb6772a12bebf 100644 --- a/packages/nodes-base/utils/descriptions.ts +++ b/packages/nodes-base/utils/descriptions.ts @@ -33,6 +33,14 @@ export const returnAllOrLimit: INodeProperties[] = [ }, ]; +export const looseTypeValidationProperty: INodeProperties = { + displayName: 'Less Strict Type Validation', + description: 'Whether to try casting value types based on the selected operator', + name: 'looseTypeValidation', + type: 'boolean', + default: true, +}; + export const encodeDecodeOptions: INodePropertyOptions[] = [ { name: 'armscii8', diff --git a/packages/workflow/src/NodeParameters/FilterParameter.ts b/packages/workflow/src/NodeParameters/FilterParameter.ts index 1d8c1a9a25832..2c2fabf96ba4f 100644 --- a/packages/workflow/src/NodeParameters/FilterParameter.ts +++ b/packages/workflow/src/NodeParameters/FilterParameter.ts @@ -94,14 +94,61 @@ function parseFilterConditionValues( )} ${suffix}`; }; - const invalidTypeDescription = 'Try changing the type of comparison.'; + const getTypeDescription = (isStrict: boolean) => { + if (isStrict) + return 'Try changing the type of the comparison, or enabling less strict type validation.'; + return 'Try changing the type of the comparison.'; + }; + + const composeInvalidTypeDescription = ( + type: string, + fromType: string, + valuePosition: 'first' | 'second', + ) => { + fromType = fromType.toLocaleLowerCase(); + const expectedType = withIndefiniteArticle(type); + + let convertionFunction = ''; + if (type === 'string') { + convertionFunction = '.toString()'; + } else if (type === 'number') { + convertionFunction = '.toNumber()'; + } else if (type === 'boolean') { + convertionFunction = '.toBoolean()'; + } + + if (strict && convertionFunction) { + const suggestFunction = ` by adding ${convertionFunction}`; + return ` +

Try either:

+
    +
  1. Enabling less strict type validation
  2. +
  3. Converting the ${valuePosition} field to ${expectedType}${suggestFunction}
  4. +
+ `; + } + + return getTypeDescription(strict); + }; + + if (!leftValid && !rightValid && typeof condition.leftValue === typeof condition.rightValue) { + return { + ok: false, + error: new FilterError( + `Comparison type expects ${withIndefiniteArticle(operator.type)} but both fields are ${withIndefiniteArticle( + typeof condition.leftValue, + )}`, + getTypeDescription(strict), + ), + }; + } if (!leftValid) { return { ok: false, error: new FilterError( composeInvalidTypeMessage(operator.type, typeof condition.leftValue, leftValueString), - invalidTypeDescription, + composeInvalidTypeDescription(operator.type, typeof condition.leftValue, 'first'), ), }; } @@ -111,7 +158,7 @@ function parseFilterConditionValues( ok: false, error: new FilterError( composeInvalidTypeMessage(rightType, typeof condition.rightValue, rightValueString), - invalidTypeDescription, + composeInvalidTypeDescription(rightType, typeof condition.rightValue, 'second'), ), }; }