๐
Type Safety & Validation
End-to-end TypeScript inference with automatic runtime validation using Zod, Valibot, or ArkType.
End-to-end type safety ยท Runtime validation ยท Explicit error handling
Define your contract once โ get type safety everywhere:
import { defineContract } from "@temporal-contract/contract";
import { z } from "zod";
export const orderContract = defineContract({
taskQueue: "orders",
workflows: {
processOrder: {
input: z.object({
orderId: z.string(),
customerId: z.string(),
amount: z.number(),
}),
output: z.object({
status: z.enum(["success", "failed"]),
transactionId: z.string().optional(),
}),
activities: {
processPayment: {
input: z.object({ customerId: z.string(), amount: z.number() }),
output: z.object({ transactionId: z.string() }),
},
sendNotification: {
input: z.object({ customerId: z.string(), message: z.string() }),
output: z.void(),
},
},
},
},
});import { Future } from "@swan-io/boxed";
import { declareActivitiesHandler, ActivityError } from "@temporal-contract/worker/activity";
import { orderContract } from "./contract";
export const activities = declareActivitiesHandler({
contract: orderContract,
activities: {
processOrder: {
processPayment: ({ customerId, amount }) => {
return Future.fromPromise(paymentService.charge(customerId, amount))
.mapOk((tx) => ({ transactionId: tx.id }))
.mapError((e) => new ActivityError("PAYMENT_FAILED", e.message, e));
},
sendNotification: ({ customerId, message }) => {
return Future.fromPromise(notificationService.send(customerId, message)).mapError(
(e) => new ActivityError("NOTIFICATION_FAILED", e.message, e),
);
},
},
},
});import { declareWorkflow } from "@temporal-contract/worker/workflow";
import { orderContract } from "./contract";
export const processOrder = declareWorkflow({
workflowName: "processOrder",
contract: orderContract,
implementation: async ({ activities }, { orderId, customerId, amount }) => {
const { transactionId } = await activities.processPayment({ customerId, amount });
await activities.sendNotification({ customerId, message: `Order ${orderId} confirmed!` });
return { status: "success", transactionId };
},
});import { TypedClient } from "@temporal-contract/client";
import { Connection, Client } from "@temporalio/client";
import { orderContract } from "./contract";
const connection = await Connection.connect({ address: "localhost:7233" });
const temporalClient = new Client({ connection });
const client = TypedClient.create(orderContract, temporalClient);
const future = client.executeWorkflow("processOrder", {
workflowId: "order-123",
args: { orderId: "ORD-123", customerId: "CUST-456", amount: 99.99 },
});
const result = await future;
result.match({
Ok: (output) => console.log(output.status), // โ
'success' | 'failed'
Error: (error) => console.error("Failed:", error),
});