2024-12-06 02:30:44 -08:00
import { json as generateSchemaFromExample , type SchemaObject } from 'generate-schema' ;
import type { JSONSchema7 } from 'json-schema' ;
2024-08-22 07:59:12 -07:00
import type {
2024-12-06 02:30:44 -08:00
FieldValueOption ,
FieldType ,
IWorkflowNodeContext ,
INodeExecutionData ,
2024-08-22 07:59:12 -07:00
IExecuteFunctions ,
} from 'n8n-workflow' ;
2024-12-06 02:30:44 -08:00
import { jsonParse , NodeOperationError , validateFieldType } from 'n8n-workflow' ;
2023-10-06 08:04:33 -07:00
2024-12-06 02:30:44 -08:00
import {
JSON_EXAMPLE ,
INPUT_SOURCE ,
WORKFLOW_INPUTS ,
VALUES ,
TYPE_OPTIONS ,
INPUT_OPTIONS ,
FALLBACK_DEFAULT_VALUE ,
} from './constants' ;
const SUPPORTED_TYPES = TYPE_OPTIONS . map ( ( x ) = > x . value ) ;
function parseJsonSchema ( schema : JSONSchema7 ) : FieldValueOption [ ] | string {
if ( ! schema ? . properties ) {
return 'Invalid JSON schema. Missing key `properties` in schema' ;
}
if ( typeof schema . properties !== 'object' ) {
return 'Invalid JSON schema. Key `properties` is not an object' ;
}
const result : FieldValueOption [ ] = [ ] ;
for ( const [ name , v ] of Object . entries ( schema . properties ) ) {
if ( typeof v !== 'object' ) {
return ` Invalid JSON schema. Value for property ' ${ name } ' is not an object ` ;
}
const type = v ? . type ;
if ( type === 'null' ) {
result . push ( { name , type : 'any' } ) ;
} else if ( Array . isArray ( type ) ) {
// Schema allows an array of types, but we don't
return ` Invalid JSON schema. Array of types for property ' ${ name } ' is not supported by n8n. Either provide a single type or use type 'any' to allow any type ` ;
} else if ( typeof type !== 'string' ) {
return ` Invalid JSON schema. Unexpected non-string type ${ type } for property ' ${ name } ' ` ;
} else if ( ! SUPPORTED_TYPES . includes ( type as never ) ) {
return ` Invalid JSON schema. Unsupported type ${ type } for property ' ${ name } '. Supported types are ${ JSON . stringify ( SUPPORTED_TYPES , null , 1 ) } ` ;
2024-08-22 07:59:12 -07:00
} else {
2024-12-06 02:30:44 -08:00
result . push ( { name , type : type as FieldType } ) ;
2024-08-22 07:59:12 -07:00
}
2024-12-06 02:30:44 -08:00
}
return result ;
}
function parseJsonExample ( context : IWorkflowNodeContext ) : JSONSchema7 {
const jsonString = context . getNodeParameter ( JSON_EXAMPLE , 0 , '' ) as string ;
const json = jsonParse < SchemaObject > ( jsonString ) ;
2023-10-06 08:04:33 -07:00
2024-12-06 02:30:44 -08:00
return generateSchemaFromExample ( json ) as JSONSchema7 ;
}
export function getFieldEntries ( context : IWorkflowNodeContext ) : FieldValueOption [ ] {
const inputSource = context . getNodeParameter ( INPUT_SOURCE , 0 ) ;
let result : FieldValueOption [ ] | string = 'Internal Error: Invalid input source' ;
try {
if ( inputSource === WORKFLOW_INPUTS ) {
result = context . getNodeParameter (
` ${ WORKFLOW_INPUTS } . ${ VALUES } ` ,
0 ,
[ ] ,
) as FieldValueOption [ ] ;
} else if ( inputSource === JSON_EXAMPLE ) {
const schema = parseJsonExample ( context ) ;
result = parseJsonSchema ( schema ) ;
}
} catch ( e : unknown ) {
result =
e && typeof e === 'object' && 'message' in e && typeof e . message === 'string'
? e . message
: ` Unknown error occurred: ${ JSON . stringify ( e ) } ` ;
}
if ( Array . isArray ( result ) ) {
return result ;
}
throw new NodeOperationError ( context . getNode ( ) , result ) ;
}
export function getWorkflowInputData (
this : IExecuteFunctions ,
inputData : INodeExecutionData [ ] ,
newParams : FieldValueOption [ ] ,
) : INodeExecutionData [ ] {
const items : INodeExecutionData [ ] = [ ] ;
for ( const [ itemIndex , item ] of inputData . entries ( ) ) {
const attemptToConvertTypes = this . getNodeParameter (
` ${ INPUT_OPTIONS } .attemptToConvertTypes ` ,
itemIndex ,
false ,
) ;
const ignoreTypeErrors = this . getNodeParameter (
` ${ INPUT_OPTIONS } .ignoreTypeErrors ` ,
itemIndex ,
false ,
) ;
// Fields listed here will explicitly overwrite original fields
const newItem : INodeExecutionData = {
json : { } ,
index : itemIndex ,
// TODO: Ensure we handle sub-execution jumps correctly.
// metadata: {
// subExecution: {
// executionId: 'uhh',
// workflowId: 'maybe?',
// },
// },
pairedItem : { item : itemIndex } ,
} ;
2023-10-06 08:04:33 -07:00
try {
2024-12-06 02:30:44 -08:00
for ( const { name , type } of newParams ) {
if ( ! item . json . hasOwnProperty ( name ) ) {
newItem . json [ name ] = FALLBACK_DEFAULT_VALUE ;
continue ;
}
const result =
type === 'any'
? ( { valid : true , newValue : item.json [ name ] } as const )
: validateFieldType ( name , item . json [ name ] , type , {
strict : ! attemptToConvertTypes ,
parseStrings : true , // Default behavior is to accept anything as a string, this is a good opportunity for a stricter boundary
} ) ;
if ( ! result . valid ) {
if ( ignoreTypeErrors ) {
newItem . json [ name ] = item . json [ name ] ;
continue ;
}
throw new NodeOperationError ( this . getNode ( ) , result . errorMessage , {
itemIndex ,
} ) ;
} else {
// If the value is `null` or `undefined`, then `newValue` is not in the returned object
if ( result . hasOwnProperty ( 'newValue' ) ) {
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
newItem . json [ name ] = result . newValue ;
} else {
newItem . json [ name ] = item . json [ name ] ;
}
}
2023-10-06 08:04:33 -07:00
}
2024-12-06 02:30:44 -08:00
items . push ( newItem ) ;
} catch ( error ) {
if ( this . continueOnFail ( ) ) {
/** todo error case? */
} else {
throw new NodeOperationError ( this . getNode ( ) , error , {
itemIndex ,
} ) ;
}
2023-10-06 08:04:33 -07:00
}
}
2024-12-06 02:30:44 -08:00
return items ;
2023-10-06 08:04:33 -07:00
}