From 2214a0cd3413bf09400f70be6347b3a26ca9c3fe Mon Sep 17 00:00:00 2001 From: Gosha Date: Fri, 6 Dec 2024 15:29:16 +0200 Subject: [PATCH] feat(api): update after pr comments --- .../build-payload-schema.usecase.ts | 41 +++++++------ ...build-available-variable-schema.usecase.ts | 7 +-- .../app/workflows-v2/util/path-to-object.ts | 7 ++- .../template-parser/liquid-parser.spec.ts | 60 +++++++++++-------- .../util/template-parser/liquid-parser.ts | 22 +++++-- .../util/template-parser/parser-types.ts | 10 ---- 6 files changed, 80 insertions(+), 67 deletions(-) delete mode 100644 apps/api/src/app/workflows-v2/util/template-parser/parser-types.ts diff --git a/apps/api/src/app/workflows-v2/usecases/build-payload-schema/build-payload-schema.usecase.ts b/apps/api/src/app/workflows-v2/usecases/build-payload-schema/build-payload-schema.usecase.ts index a8ae5713743..20d589540d4 100644 --- a/apps/api/src/app/workflows-v2/usecases/build-payload-schema/build-payload-schema.usecase.ts +++ b/apps/api/src/app/workflows-v2/usecases/build-payload-schema/build-payload-schema.usecase.ts @@ -28,34 +28,39 @@ export class BuildPayloadSchema { const variablesExample = pathsToObject(templateVars, { valuePrefix: '{{', valueSuffix: '}}', - }).payload as Record; + }).payload; return convertJsonToSchemaWithDefaults(variablesExample); } private async buildControlValues(command: BuildPayloadSchemaCommand) { - let aggregateControlValues = command.controlValues ? [command.controlValues] : []; - - if (!aggregateControlValues.length) { - aggregateControlValues = ( - await this.controlValuesRepository.find({ - _environmentId: command.environmentId, - _organizationId: command.organizationId, - _workflowId: command.workflowId, - level: ControlValuesLevelEnum.STEP_CONTROLS, - }) - ) - .map((item) => item.controls) - .filter((control): control is NonNullable => control != null); + let controlValues = command.controlValues ? [command.controlValues] : []; + + if (!controlValues.length) { + controlValues = ( + await this.controlValuesRepository.find( + { + _environmentId: command.environmentId, + _organizationId: command.organizationId, + _workflowId: command.workflowId, + level: ControlValuesLevelEnum.STEP_CONTROLS, + controls: { $ne: null }, + }, + { + controls: 1, + _id: 0, + } + ) + ).map((item) => item.controls); } - return aggregateControlValues; + return controlValues; } @Instrument() - private extractTemplateVariables(aggregateControlValues: Record[]): string[] { - const concatenatedControlValues = aggregateControlValues.map(flattenObjectValues).flat().join(' '); + private extractTemplateVariables(controlValues: Record[]): string[] { + const controlValuesString = controlValues.map(flattenObjectValues).flat().join(' '); - return extractLiquidTemplateVariables(concatenatedControlValues).validVariables; + return extractLiquidTemplateVariables(controlValuesString).validVariables.map((variable) => variable.name); } } diff --git a/apps/api/src/app/workflows-v2/usecases/build-variable-schema/build-available-variable-schema.usecase.ts b/apps/api/src/app/workflows-v2/usecases/build-variable-schema/build-available-variable-schema.usecase.ts index d10967dc0ed..5f08ff76558 100644 --- a/apps/api/src/app/workflows-v2/usecases/build-variable-schema/build-available-variable-schema.usecase.ts +++ b/apps/api/src/app/workflows-v2/usecases/build-variable-schema/build-available-variable-schema.usecase.ts @@ -56,12 +56,7 @@ export class BuildAvailableVariableSchemaUsecase { command: BuildAvailableVariableSchemaCommand ): Promise { if (workflow.payloadSchema) { - return ( - parsePayloadSchema(workflow.payloadSchema, { safe: true }) || { - type: 'object', - description: 'Payload for the current step', - } - ); + return parsePayloadSchema(workflow.payloadSchema, { safe: true }) || {}; } return this.buildPayloadSchema.execute( diff --git a/apps/api/src/app/workflows-v2/util/path-to-object.ts b/apps/api/src/app/workflows-v2/util/path-to-object.ts index 30bc37f0dc3..03568194b63 100644 --- a/apps/api/src/app/workflows-v2/util/path-to-object.ts +++ b/apps/api/src/app/workflows-v2/util/path-to-object.ts @@ -12,8 +12,11 @@ import _ from 'lodash'; * // Returns: { payload: { old: 'payload.old', new: 'payload.new' } } * // Note: 'payload' entry is ignored as it has no namespace */ -export function pathsToObject(keys: string[], { valuePrefix = '', valueSuffix = '' } = {}): Record { - const result: Record = {}; +export function pathsToObject( + keys: string[], + { valuePrefix = '', valueSuffix = '' } = {} +): Record> { + const result: Record> = {}; keys .filter((key) => key.includes('.')) diff --git a/apps/api/src/app/workflows-v2/util/template-parser/liquid-parser.spec.ts b/apps/api/src/app/workflows-v2/util/template-parser/liquid-parser.spec.ts index 2f0c73a4684..c794cca8190 100644 --- a/apps/api/src/app/workflows-v2/util/template-parser/liquid-parser.spec.ts +++ b/apps/api/src/app/workflows-v2/util/template-parser/liquid-parser.spec.ts @@ -4,58 +4,64 @@ import { extractLiquidTemplateVariables } from './liquid-parser'; describe('parseLiquidVariables', () => { it('should extract simple variable names', () => { const template = '{{name}} {{age}}'; - const { validVariables: variables } = extractLiquidTemplateVariables(template); + const { validVariables } = extractLiquidTemplateVariables(template); + const validVariablesNames = validVariables.map((variable) => variable.name); - expect(variables).to.have.members(['name', 'age']); + expect(validVariablesNames).to.have.members(['name', 'age']); }); it('should extract nested object paths', () => { const template = 'Hello {{user.profile.name}}, your address is {{user.address.street}}'; - const { validVariables: variables } = extractLiquidTemplateVariables(template); + const { validVariables } = extractLiquidTemplateVariables(template); + const validVariablesNames = validVariables.map((variable) => variable.name); - expect(variables).to.have.members(['user.profile.name', 'user.address.street']); + expect(validVariablesNames).to.have.members(['user.profile.name', 'user.address.street']); }); it('should handle multiple occurrences of the same variable', () => { const template = '{{user.name}} {{user.name}} {{user.name}}'; - const { validVariables: variables } = extractLiquidTemplateVariables(template); + const { validVariables } = extractLiquidTemplateVariables(template); + const validVariablesNames = validVariables.map((variable) => variable.name); - expect(variables).to.have.members(['user.name']); + expect(validVariablesNames).to.have.members(['user.name']); }); it('should handle mixed content with HTML and variables', () => { const template = '
Hello {{user.name}}
{{status}}'; - const { validVariables: variables } = extractLiquidTemplateVariables(template); + const { validVariables } = extractLiquidTemplateVariables(template); + const validVariablesNames = validVariables.map((variable) => variable.name); - expect(variables).to.have.members(['user.name', 'status']); + expect(validVariablesNames).to.have.members(['user.name', 'status']); }); it('should handle whitespace in template syntax', () => { const template = '{{ user.name }} {{ status }}'; - const { validVariables: variables } = extractLiquidTemplateVariables(template); + const { validVariables } = extractLiquidTemplateVariables(template); + const validVariablesNames = validVariables.map((variable) => variable.name); - expect(variables).to.have.members(['user.name', 'status']); + expect(validVariablesNames).to.have.members(['user.name', 'status']); }); it('should handle empty template string', () => { const template = ''; - const { validVariables: variables } = extractLiquidTemplateVariables(template); + const { validVariables } = extractLiquidTemplateVariables(template); - expect(variables.length).to.equal(0); + expect(validVariables).to.have.lengthOf(0); }); it('should handle template with no variables', () => { const template = 'Hello World!'; - const { validVariables: variables } = extractLiquidTemplateVariables(template); + const { validVariables } = extractLiquidTemplateVariables(template); - expect(variables).to.have.lengthOf(0); + expect(validVariables).to.have.lengthOf(0); }); it('should handle special characters in variable names', () => { const template = '{{special_var_1}} {{data-point}}'; - const { validVariables: variables } = extractLiquidTemplateVariables(template); + const { validVariables } = extractLiquidTemplateVariables(template); + const validVariablesNames = validVariables.map((variable) => variable.name); - expect(variables).to.have.members(['special_var_1', 'data-point']); + expect(validVariablesNames).to.have.members(['special_var_1', 'data-point']); }); describe('Error handling', () => { @@ -67,31 +73,35 @@ describe('parseLiquidVariables', () => { expect(variables).to.have.lengthOf(0); expect(errors).to.have.lengthOf(2); expect(errors[0].message).to.contain('expected "|" before filter'); - expect(errors[0].variable).to.equal('{{invalid..syntax}}'); - expect(errors[1].variable).to.equal('{{invalid2..syntax}}'); + expect(errors[0].name).to.equal('{{invalid..syntax}}'); + expect(errors[1].name).to.equal('{{invalid2..syntax}}'); }); it('should handle invalid liquid syntax gracefully, return valid variables', () => { const { validVariables, invalidVariables: errors } = extractLiquidTemplateVariables( '{{subscriber.name}} {{invalid..syntax}}' ); + const validVariablesNames = validVariables.map((variable) => variable.name); - expect(validVariables).to.have.members(['subscriber.name']); + expect(validVariablesNames).to.have.members(['subscriber.name']); expect(errors[0].message).to.contain('expected "|" before filter'); - expect(errors[0].variable).to.equal('{{invalid..syntax}}'); + expect(errors[0].name).to.equal('{{invalid..syntax}}'); }); it('should handle undefined input gracefully', () => { expect(() => extractLiquidTemplateVariables(undefined as any)).to.not.throw(); - expect(extractLiquidTemplateVariables(undefined as any)).to.deep.equal({ - validVariables: [], - invalidVariables: [], - }); + const { validVariables } = extractLiquidTemplateVariables(undefined as any); + const validVariablesNames = validVariables.map((variable) => variable.name); + + expect(validVariablesNames).to.have.lengthOf(0); }); it('should handle non-string input gracefully', () => { expect(() => extractLiquidTemplateVariables({} as any)).to.not.throw(); - expect(extractLiquidTemplateVariables({} as any)).to.deep.equal({ validVariables: [], invalidVariables: [] }); + const { validVariables } = extractLiquidTemplateVariables({} as any); + const validVariablesNames = validVariables.map((variable) => variable.name); + + expect(validVariablesNames).to.have.lengthOf(0); }); }); }); diff --git a/apps/api/src/app/workflows-v2/util/template-parser/liquid-parser.ts b/apps/api/src/app/workflows-v2/util/template-parser/liquid-parser.ts index a91bf08a63e..cec68e1c16b 100644 --- a/apps/api/src/app/workflows-v2/util/template-parser/liquid-parser.ts +++ b/apps/api/src/app/workflows-v2/util/template-parser/liquid-parser.ts @@ -1,5 +1,4 @@ import { Template, Liquid, RenderError, LiquidError } from 'liquidjs'; -import { TemplateParseResult, InvalidVariable } from './parser-types'; import { isValidTemplate, extractLiquidExpressions } from './parser-utils'; const LIQUID_CONFIG = { @@ -9,6 +8,17 @@ const LIQUID_CONFIG = { catchAllErrors: true, } as const; +export type Variable = { + context?: string; + message?: string; + name: string; +}; + +export type TemplateParseResult = { + validVariables: Variable[]; + invalidVariables: Variable[]; +}; + /** * Copy of LiquidErrors type from liquidjs since it's not exported. * Used to handle multiple render errors that can occur during template parsing. @@ -65,20 +75,20 @@ export function extractLiquidTemplateVariables(template: string): TemplateParseR } function processLiquidRawOutput(rawOutputs: string[]): TemplateParseResult { - const variables = new Set(); - const invalidVariables: InvalidVariable[] = []; + const validVariables = new Set(); + const invalidVariables: Variable[] = []; for (const rawOutput of rawOutputs) { try { const parsedVars = parseByLiquid(rawOutput); - parsedVars.forEach((variable) => variables.add(variable)); + parsedVars.forEach((variable) => validVariables.add(variable)); } catch (error: unknown) { if (isLiquidErrors(error)) { invalidVariables.push( ...error.errors.map((e: RenderError) => ({ context: e.context, message: e.message, - variable: rawOutput, + name: rawOutput, })) ); } @@ -86,7 +96,7 @@ function processLiquidRawOutput(rawOutputs: string[]): TemplateParseResult { } return { - validVariables: Array.from(variables), + validVariables: [...validVariables].map((name) => ({ name })), invalidVariables, }; } diff --git a/apps/api/src/app/workflows-v2/util/template-parser/parser-types.ts b/apps/api/src/app/workflows-v2/util/template-parser/parser-types.ts deleted file mode 100644 index 548e160ac1a..00000000000 --- a/apps/api/src/app/workflows-v2/util/template-parser/parser-types.ts +++ /dev/null @@ -1,10 +0,0 @@ -export type InvalidVariable = { - context: string; - message: string; - variable: string; -}; - -export type TemplateParseResult = { - validVariables: string[]; - invalidVariables: InvalidVariable[]; -};