-
Notifications
You must be signed in to change notification settings - Fork 20
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: implement email receivers in workflow triggers UI
- Loading branch information
1 parent
71e4f9c
commit cbd4702
Showing
11 changed files
with
464 additions
and
58 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
195 changes: 195 additions & 0 deletions
195
ui/admin/app/components/workflow-triggers/EmailReceiverForm.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,195 @@ | ||
import { zodResolver } from "@hookform/resolvers/zod"; | ||
import { useEffect } from "react"; | ||
import { useForm } from "react-hook-form"; | ||
import { useNavigate } from "react-router"; | ||
import { $path } from "safe-routes"; | ||
import { toast } from "sonner"; | ||
import useSWR, { mutate } from "swr"; | ||
import { z } from "zod"; | ||
|
||
import { EmailReceiver } from "~/lib/model/email-receivers"; | ||
import { EmailReceiverApiService } from "~/lib/service/api/emailRecieverApiService"; | ||
import { WorkflowService } from "~/lib/service/api/workflowService"; | ||
|
||
import { | ||
ControlledCustomInput, | ||
ControlledInput, | ||
} from "~/components/form/controlledInputs"; | ||
import { Button } from "~/components/ui/button"; | ||
import { Form } from "~/components/ui/form"; | ||
import { ScrollArea } from "~/components/ui/scroll-area"; | ||
import { | ||
Select, | ||
SelectContent, | ||
SelectItem, | ||
SelectTrigger, | ||
SelectValue, | ||
} from "~/components/ui/select"; | ||
import { useAsync } from "~/hooks/useAsync"; | ||
|
||
const formSchema = z.object({ | ||
name: z.string().min(1, "Name is required"), | ||
description: z.string(), | ||
alias: z.string(), | ||
workflow: z.string().min(1, "Workflow is required"), | ||
allowedSenders: z.array(z.string()), | ||
}); | ||
|
||
export type EmailRecieverFormValues = z.infer<typeof formSchema>; | ||
|
||
type EmailRecieverFormProps = { | ||
emailReceiver?: EmailReceiver; | ||
}; | ||
|
||
export function EmailReceiverForm({ emailReceiver }: EmailRecieverFormProps) { | ||
const navigate = useNavigate(); | ||
const getWorkflows = useSWR(WorkflowService.getWorkflows.key(), () => | ||
WorkflowService.getWorkflows() | ||
); | ||
|
||
const handleSubmitSuccess = () => { | ||
if (emailReceiver) { | ||
mutate( | ||
EmailReceiverApiService.getEmailReceiverById(emailReceiver.id) | ||
); | ||
} | ||
mutate(EmailReceiverApiService.getEmailReceivers.key()); | ||
navigate($path("/workflow-triggers")); | ||
}; | ||
|
||
const form = useForm<EmailRecieverFormValues>({ | ||
resolver: zodResolver(formSchema), | ||
mode: "onChange", | ||
defaultValues: { | ||
name: emailReceiver?.name || "", | ||
description: emailReceiver?.description || "", | ||
alias: emailReceiver?.alias || "", | ||
workflow: emailReceiver?.workflow || "", | ||
allowedSenders: emailReceiver?.allowedSenders || [], | ||
}, | ||
}); | ||
|
||
const createEmailReceiver = useAsync( | ||
EmailReceiverApiService.createEmailReceiver, | ||
{ | ||
onSuccess: handleSubmitSuccess, | ||
onError: () => { | ||
toast.error("Failed to create email receiver."); | ||
}, | ||
} | ||
); | ||
|
||
const updateEmailReceiver = useAsync( | ||
EmailReceiverApiService.updateEmailReceiver, | ||
{ | ||
onSuccess: handleSubmitSuccess, | ||
onError: () => { | ||
toast.error("Failed to update email receiver."); | ||
}, | ||
} | ||
); | ||
|
||
useEffect(() => { | ||
if (emailReceiver) { | ||
form.reset(emailReceiver); | ||
} | ||
}, [emailReceiver, form]); | ||
|
||
const handleSubmit = form.handleSubmit((values: EmailRecieverFormValues) => | ||
emailReceiver?.id | ||
? updateEmailReceiver.execute(emailReceiver.id, values) | ||
: createEmailReceiver.execute(values) | ||
); | ||
|
||
const workflows = getWorkflows.data; | ||
const isEdit = !!emailReceiver?.id; | ||
const loading = | ||
createEmailReceiver.isLoading || updateEmailReceiver.isLoading; | ||
|
||
return ( | ||
<ScrollArea className="h-full"> | ||
<Form {...form}> | ||
<form | ||
className="space-y-8 p-8 max-w-3xl mx-auto" | ||
onSubmit={handleSubmit} | ||
> | ||
<h2>{isEdit ? "Edit" : "Create"} Email Receiver</h2> | ||
|
||
<ControlledInput | ||
control={form.control} | ||
name="name" | ||
label="Name" | ||
/> | ||
|
||
<ControlledInput | ||
control={form.control} | ||
name="description" | ||
label="Description (Optional)" | ||
/> | ||
|
||
<ControlledInput | ||
control={form.control} | ||
name="alias" | ||
label="Alias (Optional)" | ||
/> | ||
|
||
<ControlledCustomInput | ||
control={form.control} | ||
name="workflow" | ||
label="Workflow" | ||
description="The workflow that will be called when an email is received." | ||
> | ||
{({ field: { ref: _, ...field }, className }) => ( | ||
<Select | ||
defaultValue={field.value} | ||
onValueChange={field.onChange} | ||
key={field.value} | ||
> | ||
<SelectTrigger className={className}> | ||
<SelectValue placeholder="Select a workflow" /> | ||
</SelectTrigger> | ||
|
||
<SelectContent> | ||
{getWorkflowOptions()} | ||
</SelectContent> | ||
</Select> | ||
)} | ||
</ControlledCustomInput> | ||
|
||
<Button | ||
className="w-full" | ||
type="submit" | ||
disabled={loading} | ||
loading={loading} | ||
> | ||
{isEdit ? "Update" : "Create"} Email Receiver | ||
</Button> | ||
</form> | ||
</Form> | ||
</ScrollArea> | ||
); | ||
|
||
function getWorkflowOptions() { | ||
const workflow = form.watch("workflow"); | ||
|
||
if (getWorkflows.isLoading) | ||
return ( | ||
<SelectItem value={workflow || "loading"} disabled> | ||
Loading workflows... | ||
</SelectItem> | ||
); | ||
|
||
if (!workflows?.length) | ||
return ( | ||
<SelectItem value={workflow || "empty"} disabled> | ||
No workflows found | ||
</SelectItem> | ||
); | ||
|
||
return workflows.map((workflow) => ( | ||
<SelectItem key={workflow.id} value={workflow.id}> | ||
{workflow.name} | ||
</SelectItem> | ||
)); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,71 @@ | ||
import { CronJob } from "~/lib/model/cronjobs"; | ||
import { EmailReceiver } from "~/lib/model/email-receivers"; | ||
import { Webhook } from "~/lib/model/webhooks"; | ||
|
||
type WorkFlowTriggerEntity = EmailReceiver | Webhook | CronJob; | ||
|
||
export type WorkflowTriggerType = "webhook" | "schedule" | "email"; | ||
|
||
export type WorkflowTrigger = { | ||
id: string; | ||
type: "webhook" | "schedule"; | ||
type: WorkflowTriggerType; | ||
name: string; | ||
workflow: string; | ||
}; | ||
|
||
const objectHasAllKeys = <T extends object = object>( | ||
obj: object, | ||
keys: (keyof T)[] | ||
): obj is T => keys.every((key) => key in obj); | ||
|
||
function isEmailReceiver( | ||
entity: WorkFlowTriggerEntity | ||
): entity is EmailReceiver { | ||
return objectHasAllKeys<EmailReceiver>(entity, [ | ||
"workflow", | ||
"emailAddress", | ||
]); | ||
} | ||
|
||
function isWebhook(entity: WorkFlowTriggerEntity): entity is Webhook { | ||
return objectHasAllKeys<Webhook>(entity, ["workflow", "validationHeader"]); | ||
} | ||
|
||
function isCronJob(entity: WorkFlowTriggerEntity): entity is CronJob { | ||
return objectHasAllKeys<CronJob>(entity, ["workflow", "schedule"]); | ||
} | ||
|
||
function convertToWorkflowTrigger( | ||
entity: WorkFlowTriggerEntity | ||
): WorkflowTrigger | null { | ||
switch (true) { | ||
case isEmailReceiver(entity): | ||
return { | ||
id: entity.id, | ||
type: "email", | ||
name: entity.name, | ||
workflow: entity.workflow, | ||
}; | ||
case isWebhook(entity): | ||
return { | ||
id: entity.id, | ||
type: "webhook", | ||
name: entity.name, | ||
workflow: entity.workflow, | ||
}; | ||
case isCronJob(entity): | ||
return { | ||
id: entity.id, | ||
type: "schedule", | ||
name: entity.id, | ||
workflow: entity.workflow, | ||
}; | ||
default: | ||
console.error("Unknown entity type", entity); | ||
return null; | ||
} | ||
} | ||
|
||
export function collateWorkflowTriggers(list: WorkFlowTriggerEntity[]) { | ||
return list.map(convertToWorkflowTrigger).filter((x) => !!x); | ||
} |
Oops, something went wrong.