summaryrefslogtreecommitdiffhomepage
path: root/js/src/server/message.ts
diff options
context:
space:
mode:
authorDax Raad <[email protected]>2025-05-28 13:22:48 -0400
committerDax Raad <[email protected]>2025-05-28 13:22:48 -0400
commit3a4d3b249f4b529ead2f80dd0df19a9cdc0c10a6 (patch)
treebb77726cba6875a6daa8fe6230d9a2826fab83ed /js/src/server/message.ts
parent55a6fcdd3f5b3c55712e5cfc9dd4d994da38d4c8 (diff)
downloadopencode-3a4d3b249f4b529ead2f80dd0df19a9cdc0c10a6.tar.gz
opencode-3a4d3b249f4b529ead2f80dd0df19a9cdc0c10a6.zip
generate message type
Diffstat (limited to 'js/src/server/message.ts')
-rw-r--r--js/src/server/message.ts146
1 files changed, 146 insertions, 0 deletions
diff --git a/js/src/server/message.ts b/js/src/server/message.ts
new file mode 100644
index 000000000..77ef5b8f2
--- /dev/null
+++ b/js/src/server/message.ts
@@ -0,0 +1,146 @@
+import z from "zod";
+
+const ToolCall = z
+ .object({
+ state: z.literal("call"),
+ step: z.number().optional(),
+ toolCallId: z.string(),
+ toolName: z.string(),
+ args: z.record(z.string(), z.any()),
+ })
+ .openapi({
+ ref: "Session.Message.ToolInvocation.ToolCall",
+ });
+
+const ToolPartialCall = z
+ .object({
+ state: z.literal("partial-call"),
+ step: z.number().optional(),
+ toolCallId: z.string(),
+ toolName: z.string(),
+ args: z.record(z.string(), z.any()),
+ })
+ .openapi({
+ ref: "Session.Message.ToolInvocation.ToolPartialCall",
+ });
+
+const ToolResult = z
+ .object({
+ state: z.literal("result"),
+ step: z.number().optional(),
+ toolCallId: z.string(),
+ toolName: z.string(),
+ args: z.record(z.string(), z.any()),
+ result: z.string(),
+ })
+ .openapi({
+ ref: "Session.Message.ToolInvocation.ToolResult",
+ });
+
+const ToolInvocation = z
+ .union([ToolCall, ToolPartialCall, ToolResult])
+ .openapi({
+ ref: "Session.Message.ToolInvocation",
+ });
+export type ToolInvocation = z.infer<typeof ToolInvocation>;
+
+const TextPart = z
+ .object({
+ type: z.literal("text"),
+ text: z.string(),
+ })
+ .openapi({
+ ref: "Session.Message.Part.Text",
+ });
+
+const ReasoningPart = z
+ .object({
+ type: z.literal("reasoning"),
+ text: z.string(),
+ providerMetadata: z.record(z.any()).optional(),
+ })
+ .openapi({
+ ref: "Session.Message.Part.Reasoning",
+ });
+
+const ToolInvocationPart = z
+ .object({
+ type: z.literal("tool-invocation"),
+ toolInvocation: ToolInvocation,
+ })
+ .openapi({
+ ref: "Session.Message.Part.ToolInvocation",
+ });
+
+const SourceUrlPart = z
+ .object({
+ type: z.literal("source-url"),
+ sourceId: z.string(),
+ url: z.string(),
+ title: z.string().optional(),
+ providerMetadata: z.record(z.any()).optional(),
+ })
+ .openapi({
+ ref: "Session.Message.Part.SourceUrl",
+ });
+
+const FilePart = z
+ .object({
+ type: z.literal("file"),
+ mediaType: z.string(),
+ filename: z.string().optional(),
+ url: z.string(),
+ })
+ .openapi({
+ ref: "Session.Message.Part.File",
+ });
+
+const StepStartPart = z
+ .object({
+ type: z.literal("step-start"),
+ })
+ .openapi({
+ ref: "Session.Message.Part.StepStart",
+ });
+
+const DataPart = z
+ .object({
+ type: z.custom<`data-${string}`>(),
+ id: z.string().optional(),
+ data: z.unknown(),
+ })
+ .openapi({
+ ref: "Session.Message.Part.Data",
+ });
+
+const Part = z
+ .union([
+ TextPart,
+ ReasoningPart,
+ ToolInvocationPart,
+ SourceUrlPart,
+ FilePart,
+ StepStartPart,
+ DataPart,
+ ])
+ .openapi({
+ ref: "Session.Message.Part",
+ });
+
+export const SessionMessage = z
+ .object({
+ id: z.string(),
+ role: z.enum(["system", "user", "assistant"]),
+ parts: z.array(Part),
+ metadata: z.object({
+ time: z.object({
+ created: z.number(),
+ completed: z.number().optional(),
+ }),
+ sessionID: z.string(),
+ tool: z.record(z.string(), z.any()),
+ }),
+ })
+ .openapi({
+ ref: "Session.Message",
+ });