Fix manual trigger output schema (#8150)

- add schema for manual trigger
- split into sub functions
- handle case with no variables
This commit is contained in:
Thomas Trompette
2024-10-28 18:42:09 +01:00
committed by GitHub
parent 69c24968c1
commit 409def8431
13 changed files with 305 additions and 83 deletions

View File

@ -87,6 +87,7 @@ export const WorkflowEditTriggerManualForm = ({
...trigger, ...trigger,
settings: { settings: {
objectType: updatedObject, objectType: updatedObject,
outputSchema: {},
}, },
}); });
}} }}

View File

@ -6,13 +6,13 @@ import { useDropdown } from '@/ui/layout/dropdown/hooks/useDropdown';
import { SearchVariablesDropdownStepItem } from '@/workflow/search-variables/components/SearchVariablesDropdownStepItem'; import { SearchVariablesDropdownStepItem } from '@/workflow/search-variables/components/SearchVariablesDropdownStepItem';
import SearchVariablesDropdownStepSubItem from '@/workflow/search-variables/components/SearchVariablesDropdownStepSubItem'; import SearchVariablesDropdownStepSubItem from '@/workflow/search-variables/components/SearchVariablesDropdownStepSubItem';
import { SEARCH_VARIABLES_DROPDOWN_ID } from '@/workflow/search-variables/constants/SearchVariablesDropdownId'; import { SEARCH_VARIABLES_DROPDOWN_ID } from '@/workflow/search-variables/constants/SearchVariablesDropdownId';
import { useAvailableVariablesInWorkflowStep } from '@/workflow/search-variables/hooks/useAvailableVariablesInWorkflowStep';
import { StepOutputSchema } from '@/workflow/search-variables/types/StepOutputSchema'; import { StepOutputSchema } from '@/workflow/search-variables/types/StepOutputSchema';
import { useTheme } from '@emotion/react'; import { useTheme } from '@emotion/react';
import styled from '@emotion/styled'; import styled from '@emotion/styled';
import { Editor } from '@tiptap/react'; import { Editor } from '@tiptap/react';
import { useState } from 'react'; import { useState } from 'react';
import { IconVariable } from 'twenty-ui'; import { IconVariable } from 'twenty-ui';
import { useAvailableVariablesInWorkflowStep } from '@/workflow/hooks/useAvailableVariablesInWorkflowStep';
const StyledDropdownVariableButtonContainer = styled( const StyledDropdownVariableButtonContainer = styled(
StyledDropdownButtonContainer, StyledDropdownButtonContainer,

View File

@ -1,3 +1,4 @@
import { MenuItem } from '@/ui/navigation/menu-item/components/MenuItem';
import { MenuItemSelect } from '@/ui/navigation/menu-item/components/MenuItemSelect'; import { MenuItemSelect } from '@/ui/navigation/menu-item/components/MenuItemSelect';
import { StepOutputSchema } from '@/workflow/search-variables/types/StepOutputSchema'; import { StepOutputSchema } from '@/workflow/search-variables/types/StepOutputSchema';
@ -10,7 +11,7 @@ export const SearchVariablesDropdownStepItem = ({
steps, steps,
onSelect, onSelect,
}: SearchVariablesDropdownStepItemProps) => { }: SearchVariablesDropdownStepItemProps) => {
return ( return steps.length > 0 ? (
<> <>
{steps.map((item, _index) => ( {steps.map((item, _index) => (
<MenuItemSelect <MenuItemSelect
@ -24,5 +25,13 @@ export const SearchVariablesDropdownStepItem = ({
/> />
))} ))}
</> </>
) : (
<MenuItem
key="no-steps"
onClick={() => {}}
text="No variables available"
LeftIcon={undefined}
hasSubMenu={false}
/>
); );
}; };

View File

@ -1,11 +1,13 @@
import { capitalize } from '~/utils/string/capitalize';
import { useRecoilValue } from 'recoil';
import { workflowIdState } from '@/workflow/states/workflowIdState';
import { useWorkflowWithCurrentVersion } from '@/workflow/hooks/useWorkflowWithCurrentVersion'; import { useWorkflowWithCurrentVersion } from '@/workflow/hooks/useWorkflowWithCurrentVersion';
import { StepOutputSchema } from '@/workflow/search-variables/types/StepOutputSchema';
import { getTriggerStepName } from '@/workflow/search-variables/utils/getTriggerStepName';
import { workflowIdState } from '@/workflow/states/workflowIdState';
import { workflowSelectedNodeState } from '@/workflow/states/workflowSelectedNodeState'; import { workflowSelectedNodeState } from '@/workflow/states/workflowSelectedNodeState';
import { getStepDefinitionOrThrow } from '@/workflow/utils/getStepDefinitionOrThrow'; import { getStepDefinitionOrThrow } from '@/workflow/utils/getStepDefinitionOrThrow';
import isEmpty from 'lodash.isempty';
import { useRecoilValue } from 'recoil';
import { isDefined } from 'twenty-ui'; import { isDefined } from 'twenty-ui';
import { StepOutputSchema } from '@/workflow/search-variables/types/StepOutputSchema'; import { isEmptyObject } from '~/utils/isEmptyObject';
export const useAvailableVariablesInWorkflowStep = (): StepOutputSchema[] => { export const useAvailableVariablesInWorkflowStep = (): StepOutputSchema[] => {
const workflowId = useRecoilValue(workflowIdState); const workflowId = useRecoilValue(workflowIdState);
@ -41,20 +43,22 @@ export const useAvailableVariablesInWorkflowStep = (): StepOutputSchema[] => {
const result = []; const result = [];
if ( if (
workflow.currentVersion.trigger?.type === 'DATABASE_EVENT' && isDefined(workflow.currentVersion.trigger) &&
isDefined(workflow.currentVersion.trigger?.settings?.outputSchema) isDefined(workflow.currentVersion.trigger?.settings?.outputSchema) &&
!isEmptyObject(workflow.currentVersion.trigger?.settings?.outputSchema)
) { ) {
const [object, action] =
workflow.currentVersion.trigger.settings.eventName.split('.');
result.push({ result.push({
id: 'trigger', id: 'trigger',
name: `${capitalize(object)} is ${capitalize(action)}`, name: getTriggerStepName(workflow.currentVersion.trigger),
outputSchema: workflow.currentVersion.trigger.settings.outputSchema, outputSchema: workflow.currentVersion.trigger.settings.outputSchema,
}); });
} }
previousSteps.forEach((previousStep) => { previousSteps.forEach((previousStep) => {
if (isDefined(previousStep.settings.outputSchema)) { if (
isDefined(previousStep.settings.outputSchema) &&
!isEmpty(previousStep.settings.outputSchema)
) {
result.push({ result.push({
id: previousStep.id, id: previousStep.id,
name: previousStep.name, name: previousStep.name,

View File

@ -0,0 +1,28 @@
import {
WorkflowDatabaseEventTrigger,
WorkflowTrigger,
} from '@/workflow/types/Workflow';
import { capitalize } from '~/utils/string/capitalize';
export const getTriggerStepName = (trigger: WorkflowTrigger): string => {
switch (trigger.type) {
case 'DATABASE_EVENT':
return getDatabaseEventTriggerStepName(trigger);
case 'MANUAL':
if (!trigger.settings.objectType) {
return 'Manual trigger';
}
return 'Manual trigger for ' + capitalize(trigger.settings.objectType);
default:
return '';
}
};
const getDatabaseEventTriggerStepName = (
trigger: WorkflowDatabaseEventTrigger,
): string => {
const [object, action] = trigger.settings.eventName.split('.');
return `${capitalize(object)} is ${capitalize(action)}`;
};

View File

@ -69,6 +69,7 @@ export type WorkflowManualTrigger = BaseTrigger & {
type: 'MANUAL'; type: 'MANUAL';
settings: { settings: {
objectType?: string; objectType?: string;
outputSchema: object;
}; };
}; };

View File

@ -16,11 +16,13 @@ export const getManualTriggerDefaultSettings = ({
case 'EVERYWHERE': { case 'EVERYWHERE': {
return { return {
objectType: undefined, objectType: undefined,
outputSchema: {},
}; };
} }
case 'WHEN_RECORD_SELECTED': { case 'WHEN_RECORD_SELECTED': {
return { return {
objectType: activeObjectMetadataItems[0].nameSingular, objectType: activeObjectMetadataItems[0].nameSingular,
outputSchema: {},
}; };
} }
} }

View File

@ -1,16 +1,16 @@
import { Args, Mutation, Resolver } from '@nestjs/graphql';
import { UseFilters, UseGuards } from '@nestjs/common'; import { UseFilters, UseGuards } from '@nestjs/common';
import { Args, Mutation, Resolver } from '@nestjs/graphql';
import graphqlTypeJson from 'graphql-type-json'; import graphqlTypeJson from 'graphql-type-json';
import { WorkspaceAuthGuard } from 'src/engine/guards/workspace-auth.guard';
import { UserAuthGuard } from 'src/engine/guards/user-auth.guard';
import { WorkflowTriggerGraphqlApiExceptionFilter } from 'src/engine/core-modules/workflow/filters/workflow-trigger-graphql-api-exception.filter';
import { OutputSchema } from 'src/modules/workflow/workflow-executor/types/workflow-step-settings.type';
import { AuthWorkspace } from 'src/engine/decorators/auth/auth-workspace.decorator';
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
import { ComputeStepOutputSchemaInput } from 'src/engine/core-modules/workflow/dtos/compute-step-output-schema-input.dto'; import { ComputeStepOutputSchemaInput } from 'src/engine/core-modules/workflow/dtos/compute-step-output-schema-input.dto';
import { WorkflowTriggerGraphqlApiExceptionFilter } from 'src/engine/core-modules/workflow/filters/workflow-trigger-graphql-api-exception.filter';
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
import { AuthWorkspace } from 'src/engine/decorators/auth/auth-workspace.decorator';
import { UserAuthGuard } from 'src/engine/guards/user-auth.guard';
import { WorkspaceAuthGuard } from 'src/engine/guards/workspace-auth.guard';
import { WorkflowBuilderService } from 'src/modules/workflow/workflow-builder/workflow-builder.service'; import { WorkflowBuilderService } from 'src/modules/workflow/workflow-builder/workflow-builder.service';
import { OutputSchema } from 'src/modules/workflow/workflow-executor/types/workflow-step-settings.type';
@Resolver() @Resolver()
@UseGuards(WorkspaceAuthGuard, UserAuthGuard) @UseGuards(WorkspaceAuthGuard, UserAuthGuard)

View File

@ -19,7 +19,10 @@ import {
WorkflowStepExecutorExceptionCode, WorkflowStepExecutorExceptionCode,
} from 'src/modules/workflow/workflow-executor/exceptions/workflow-step-executor.exception'; } from 'src/modules/workflow/workflow-executor/exceptions/workflow-step-executor.exception';
import { WorkflowActionResult } from 'src/modules/workflow/workflow-executor/types/workflow-action-result.type'; import { WorkflowActionResult } from 'src/modules/workflow/workflow-executor/types/workflow-action-result.type';
import { WorkflowSendEmailStepInput } from 'src/modules/workflow/workflow-executor/types/workflow-step-settings.type'; import {
WorkflowSendEmailStepInput,
WorkflowSendEmailStepOutputSchema,
} from 'src/modules/workflow/workflow-executor/types/workflow-step-settings.type';
import { isDefined } from 'src/utils/is-defined'; import { isDefined } from 'src/utils/is-defined';
@Injectable() @Injectable()
@ -112,7 +115,9 @@ export class SendEmailWorkflowAction implements WorkflowAction {
this.logger.log(`Email sent successfully`); this.logger.log(`Email sent successfully`);
return { result: { success: true } }; return {
result: { success: true } satisfies WorkflowSendEmailStepOutputSchema,
};
} catch (error) { } catch (error) {
return { error }; return { error };
} }

View File

@ -0,0 +1,77 @@
import { v4 } from 'uuid';
import { ObjectRecordCreateEvent } from 'src/engine/core-modules/event-emitter/types/object-record-create.event';
import { ObjectRecordDeleteEvent } from 'src/engine/core-modules/event-emitter/types/object-record-delete.event';
import { ObjectRecordDestroyEvent } from 'src/engine/core-modules/event-emitter/types/object-record-destroy.event';
import { ObjectRecordUpdateEvent } from 'src/engine/core-modules/event-emitter/types/object-record-update.event';
import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity';
import { generateFakeObjectRecord } from 'src/modules/workflow/workflow-builder/utils/generate-fake-object-record';
export const generateFakeObjectRecordEvent = <Entity>(
objectMetadataEntity: ObjectMetadataEntity,
action: 'created' | 'updated' | 'deleted' | 'destroyed',
):
| ObjectRecordCreateEvent<Entity>
| ObjectRecordUpdateEvent<Entity>
| ObjectRecordDeleteEvent<Entity>
| ObjectRecordDestroyEvent<Entity> => {
const recordId = v4();
const userId = v4();
const workspaceMemberId = v4();
const after = generateFakeObjectRecord<Entity>(objectMetadataEntity);
if (action === 'created') {
return {
recordId,
userId,
workspaceMemberId,
objectMetadata: objectMetadataEntity,
properties: {
after,
},
} satisfies ObjectRecordCreateEvent<Entity>;
}
const before = generateFakeObjectRecord<Entity>(objectMetadataEntity);
if (action === 'updated') {
return {
recordId,
userId,
workspaceMemberId,
objectMetadata: objectMetadataEntity,
properties: {
before,
after,
diff: after,
},
} satisfies ObjectRecordUpdateEvent<Entity>;
}
if (action === 'deleted') {
return {
recordId,
userId,
workspaceMemberId,
objectMetadata: objectMetadataEntity,
properties: {
before,
},
} satisfies ObjectRecordDeleteEvent<Entity>;
}
if (action === 'destroyed') {
return {
recordId,
userId,
workspaceMemberId,
objectMetadata: objectMetadataEntity,
properties: {
before,
},
} satisfies ObjectRecordDestroyEvent<Entity>;
}
throw new Error(`Unknown action '${action}'`);
};

View File

@ -0,0 +1,11 @@
import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity';
import { generateFakeValue } from 'src/engine/utils/generate-fake-value';
export const generateFakeObjectRecord = <Entity>(
objectMetadataEntity: ObjectMetadataEntity,
): Entity =>
objectMetadataEntity.fields.reduce((acc, field) => {
acc[field.name] = generateFakeValue(field.type);
return acc;
}, {} as Entity);

View File

@ -1,24 +1,26 @@
import { InjectRepository } from '@nestjs/typeorm';
import { Injectable } from '@nestjs/common'; import { Injectable } from '@nestjs/common';
import { InjectRepository } from '@nestjs/typeorm';
import { join } from 'path'; import { join } from 'path';
import { Repository } from 'typeorm'; import { Repository } from 'typeorm';
import { generateFakeObjectRecordEvent } from 'src/engine/core-modules/event-emitter/utils/generate-fake-object-record-event';
import { INDEX_FILE_NAME } from 'src/engine/core-modules/serverless/drivers/constants/index-file-name';
import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity';
import { ServerlessFunctionService } from 'src/engine/metadata-modules/serverless-function/serverless-function.service'; import { ServerlessFunctionService } from 'src/engine/metadata-modules/serverless-function/serverless-function.service';
import { CodeIntrospectionService } from 'src/modules/code-introspection/code-introspection.service'; import { CodeIntrospectionService } from 'src/modules/code-introspection/code-introspection.service';
import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { generateFakeObjectRecord } from 'src/modules/workflow/workflow-builder/utils/generate-fake-object-record';
import {
WorkflowTrigger,
WorkflowTriggerType,
} from 'src/modules/workflow/workflow-trigger/types/workflow-trigger.type';
import { import {
WorkflowActionType, WorkflowActionType,
WorkflowStep, WorkflowStep,
} from 'src/modules/workflow/workflow-executor/types/workflow-action.type'; } from 'src/modules/workflow/workflow-executor/types/workflow-action.type';
import { WorkflowSendEmailStepOutputSchema } from 'src/modules/workflow/workflow-executor/types/workflow-step-settings.type';
import {
WorkflowTrigger,
WorkflowTriggerType,
} from 'src/modules/workflow/workflow-trigger/types/workflow-trigger.type';
import { isDefined } from 'src/utils/is-defined'; import { isDefined } from 'src/utils/is-defined';
import { generateFakeObjectRecordEvent } from 'src/engine/core-modules/event-emitter/utils/generate-fake-object-record-event';
import { INDEX_FILE_NAME } from 'src/engine/core-modules/serverless/drivers/constants/index-file-name';
@Injectable() @Injectable()
export class WorkflowBuilderService { export class WorkflowBuilderService {
@ -35,77 +37,155 @@ export class WorkflowBuilderService {
}: { }: {
step: WorkflowTrigger | WorkflowStep; step: WorkflowTrigger | WorkflowStep;
workspaceId: string; workspaceId: string;
}) { }): Promise<object> {
const stepType = step.type; const stepType = step.type;
switch (stepType) { switch (stepType) {
case WorkflowTriggerType.DATABASE_EVENT: { case WorkflowTriggerType.DATABASE_EVENT: {
const [nameSingular, action] = step.settings.eventName.split('.'); return await this.computeDatabaseEventTriggerOutputSchema({
eventName: step.settings.eventName,
workspaceId,
objectMetadataRepository: this.objectMetadataRepository,
});
}
case WorkflowTriggerType.MANUAL: {
const { objectType } = step.settings;
if (!['created', 'updated', 'deleted', 'destroyed'].includes(action)) { if (!objectType) {
return {}; return {};
} }
const objectMetadata = return await this.computeManualTriggerOutputSchema({
await this.objectMetadataRepository.findOneOrFail({ objectType,
where: { workspaceId,
nameSingular, objectMetadataRepository: this.objectMetadataRepository,
workspaceId, });
},
relations: ['fields'],
});
if (!isDefined(objectMetadata)) {
return {};
}
return generateFakeObjectRecordEvent(
objectMetadata,
action as 'created' | 'updated' | 'deleted' | 'destroyed',
);
} }
case WorkflowActionType.SEND_EMAIL: { case WorkflowActionType.SEND_EMAIL: {
return { success: true }; return this.computeSendEmailActionOutputSchema();
} }
case WorkflowActionType.CODE: { case WorkflowActionType.CODE: {
const { serverlessFunctionId, serverlessFunctionVersion } = const { serverlessFunctionId, serverlessFunctionVersion } =
step.settings.input; step.settings.input;
if (serverlessFunctionId === '') { return await this.computeCodeActionOutputSchema({
return {}; serverlessFunctionId,
} serverlessFunctionVersion,
workspaceId,
const sourceCode = ( serverlessFunctionService: this.serverlessFunctionService,
await this.serverlessFunctionService.getServerlessFunctionSourceCode( codeIntrospectionService: this.codeIntrospectionService,
workspaceId, });
serverlessFunctionId,
serverlessFunctionVersion,
)
)?.[join('src', INDEX_FILE_NAME)];
if (!isDefined(sourceCode)) {
return {};
}
const fakeFunctionInput =
this.codeIntrospectionService.generateInputData(sourceCode);
// handle the case when event parameter is destructured:
// (event: {param1: string; param2: number}) VS ({param1, param2}: {param1: string; param2: number})
const formattedInput = Object.values(fakeFunctionInput)[0];
const resultFromFakeInput =
await this.serverlessFunctionService.executeOneServerlessFunction(
serverlessFunctionId,
workspaceId,
formattedInput,
serverlessFunctionVersion,
);
return resultFromFakeInput.data ?? {};
} }
default: default:
throw new Error(`Unknown type ${stepType}`); return {};
} }
} }
private async computeDatabaseEventTriggerOutputSchema({
eventName,
workspaceId,
objectMetadataRepository,
}: {
eventName: string;
workspaceId: string;
objectMetadataRepository: Repository<ObjectMetadataEntity>;
}) {
const [nameSingular, action] = eventName.split('.');
if (!['created', 'updated', 'deleted', 'destroyed'].includes(action)) {
return {};
}
const objectMetadata = await objectMetadataRepository.findOneOrFail({
where: {
nameSingular,
workspaceId,
},
relations: ['fields'],
});
if (!isDefined(objectMetadata)) {
return {};
}
return generateFakeObjectRecordEvent(
objectMetadata,
action as 'created' | 'updated' | 'deleted' | 'destroyed',
);
}
private async computeManualTriggerOutputSchema<Entity>({
objectType,
workspaceId,
objectMetadataRepository,
}: {
objectType: string;
workspaceId: string;
objectMetadataRepository: Repository<ObjectMetadataEntity>;
}) {
const objectMetadata = await objectMetadataRepository.findOneOrFail({
where: {
nameSingular: objectType,
workspaceId,
},
relations: ['fields'],
});
if (!isDefined(objectMetadata)) {
return {};
}
return generateFakeObjectRecord<Entity>(objectMetadata);
}
private computeSendEmailActionOutputSchema(): WorkflowSendEmailStepOutputSchema {
return { success: true };
}
private async computeCodeActionOutputSchema({
serverlessFunctionId,
serverlessFunctionVersion,
workspaceId,
serverlessFunctionService,
codeIntrospectionService,
}: {
serverlessFunctionId: string;
serverlessFunctionVersion: string;
workspaceId: string;
serverlessFunctionService: ServerlessFunctionService;
codeIntrospectionService: CodeIntrospectionService;
}) {
if (serverlessFunctionId === '') {
return {};
}
const sourceCode = (
await serverlessFunctionService.getServerlessFunctionSourceCode(
workspaceId,
serverlessFunctionId,
serverlessFunctionVersion,
)
)?.[join('src', INDEX_FILE_NAME)];
if (!isDefined(sourceCode)) {
return {};
}
const fakeFunctionInput =
codeIntrospectionService.generateInputData(sourceCode);
// handle the case when event parameter is destructured:
// (event: {param1: string; param2: number}) VS ({param1, param2}: {param1: string; param2: number})
const formattedInput = Object.values(fakeFunctionInput)[0];
const resultFromFakeInput =
await serverlessFunctionService.executeOneServerlessFunction(
serverlessFunctionId,
workspaceId,
formattedInput,
serverlessFunctionVersion,
);
return resultFromFakeInput.data ?? {};
}
} }

View File

@ -30,6 +30,10 @@ export type WorkflowSendEmailStepInput = {
body?: string; body?: string;
}; };
export type WorkflowSendEmailStepOutputSchema = {
success: boolean;
};
export type WorkflowSendEmailStepSettings = BaseWorkflowStepSettings & { export type WorkflowSendEmailStepSettings = BaseWorkflowStepSettings & {
input: WorkflowSendEmailStepInput; input: WorkflowSendEmailStepInput;
}; };