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:
@ -87,6 +87,7 @@ export const WorkflowEditTriggerManualForm = ({
|
|||||||
...trigger,
|
...trigger,
|
||||||
settings: {
|
settings: {
|
||||||
objectType: updatedObject,
|
objectType: updatedObject,
|
||||||
|
outputSchema: {},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}}
|
}}
|
||||||
|
|||||||
@ -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,
|
||||||
|
|||||||
@ -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}
|
||||||
|
/>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@ -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,
|
||||||
@ -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)}`;
|
||||||
|
};
|
||||||
@ -69,6 +69,7 @@ export type WorkflowManualTrigger = BaseTrigger & {
|
|||||||
type: 'MANUAL';
|
type: 'MANUAL';
|
||||||
settings: {
|
settings: {
|
||||||
objectType?: string;
|
objectType?: string;
|
||||||
|
outputSchema: object;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@ -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: {},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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)
|
||||||
|
|||||||
@ -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 };
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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}'`);
|
||||||
|
};
|
||||||
@ -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);
|
||||||
@ -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 ?? {};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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;
|
||||||
};
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user