refactor: move conversation types to core (#149)

This commit is contained in:
2026-06-18 20:47:36 +01:00
committed by GitHub
parent 769fd5c77d
commit e6af1b7851
17 changed files with 143 additions and 75 deletions

View File

@@ -1,3 +1,4 @@
import { ConversationEntryKind } from "@freya/core"
import { describe, expect, test } from "bun:test" import { describe, expect, test } from "bun:test"
import type { AppendConversationEntryInput } from "../conversations/storage.ts" import type { AppendConversationEntryInput } from "../conversations/storage.ts"
@@ -6,7 +7,6 @@ import type {
ConversationStorageEntry, ConversationStorageEntry,
} from "./conversation-recording-query-agent.ts" } from "./conversation-recording-query-agent.ts"
import { ConversationEntryKind } from "../conversations/types.ts"
import { ConversationRecordingQueryAgent } from "./conversation-recording-query-agent.ts" import { ConversationRecordingQueryAgent } from "./conversation-recording-query-agent.ts"
import { import {
createQueryAgentEventListeners, createQueryAgentEventListeners,

View File

@@ -1,12 +1,13 @@
import type { ConversationEntryMetadata } from "@freya/core"
import { ConversationEntryKind } from "@freya/core"
import { randomUUID } from "node:crypto" import { randomUUID } from "node:crypto"
import type { import type {
AppendConversationEntryInput, AppendConversationEntryInput,
ConversationEntryRow, ConversationEntryRow,
} from "../conversations/storage.ts" } from "../conversations/storage.ts"
import type { ConversationEntryMetadata } from "../conversations/types.ts"
import { ConversationEntryKind } from "../conversations/types.ts"
import { import {
createQueryAgentEventListeners, createQueryAgentEventListeners,
QueryAgentEvent, QueryAgentEvent,
@@ -19,6 +20,7 @@ import {
type QueryAgentStreamEvent, type QueryAgentStreamEvent,
} from "./query-agent.ts" } from "./query-agent.ts"
/** Storage operations used to persist and replay query-agent conversation entries. */
export interface ConversationStorage { export interface ConversationStorage {
getOrCreateConversation(): Promise<{ id: string }> getOrCreateConversation(): Promise<{ id: string }>
appendEntry( appendEntry(
@@ -28,11 +30,13 @@ export interface ConversationStorage {
listEntries(conversationId: string): Promise<ConversationStorageEntry[]> listEntries(conversationId: string): Promise<ConversationStorageEntry[]>
} }
/** Minimal persisted entry shape needed by recording and replay agents. */
export type ConversationStorageEntry = Pick< export type ConversationStorageEntry = Pick<
ConversationEntryRow, ConversationEntryRow,
"id" | "sequence" | "kind" | "payload" | "metadata" | "createdAt" "id" | "sequence" | "kind" | "payload" | "metadata" | "createdAt"
> >
/** Configuration for wrapping a QueryAgent with conversation recording. */
export interface ConversationRecordingQueryAgentConfig { export interface ConversationRecordingQueryAgentConfig {
agent: QueryAgent agent: QueryAgent
storage: ConversationStorage storage: ConversationStorage

View File

@@ -1,9 +1,9 @@
import { ConversationEntryKind } from "@freya/core"
import { beforeEach, describe, expect, mock, test } from "bun:test" import { beforeEach, describe, expect, mock, test } from "bun:test"
import type { QueryAgentToolbox } from "./query-agent-toolbox.ts" import type { QueryAgentToolbox } from "./query-agent-toolbox.ts"
import type { QueryAgentStreamEvent } from "./query-agent.ts" import type { QueryAgentStreamEvent } from "./query-agent.ts"
import { ConversationEntryKind } from "../conversations/types.ts"
import { QueryAgentEvent } from "./query-agent.ts" import { QueryAgentEvent } from "./query-agent.ts"
interface FakePiSession { interface FakePiSession {

View File

@@ -33,13 +33,25 @@ import {
import { createSessionManager } from "./session-manager.ts" import { createSessionManager } from "./session-manager.ts"
import { createFreyaAgentTools, FREYA_AGENT_TOOL_NAMES } from "./tools.ts" import { createFreyaAgentTools, FREYA_AGENT_TOOL_NAMES } from "./tools.ts"
/** Active Pi SDK session instance returned by createAgentSession. */
type PiSession = Awaited<ReturnType<typeof createAgentSession>>["session"] type PiSession = Awaited<ReturnType<typeof createAgentSession>>["session"]
/** Pi event emitted when a message finishes. */
type PiMessageEndEvent = Extract<AgentSessionEvent, { type: "message_end" }> type PiMessageEndEvent = Extract<AgentSessionEvent, { type: "message_end" }>
/** Message payload carried by Pi's message-end event. */
type PiAgentMessage = PiMessageEndEvent["message"] type PiAgentMessage = PiMessageEndEvent["message"]
/** Pi event emitted when an agent run finishes. */
type PiAgentEndEvent = Extract<AgentSessionEvent, { type: "agent_end" }> type PiAgentEndEvent = Extract<AgentSessionEvent, { type: "agent_end" }>
/** Session manager created for Pi conversation replay. */
type PiSessionManager = ReturnType<typeof createSessionManager> type PiSessionManager = ReturnType<typeof createSessionManager>
/** Message shape accepted by the replay session manager. */
type PiSessionMessage = Parameters<PiSessionManager["appendMessage"]>[0] type PiSessionMessage = Parameters<PiSessionManager["appendMessage"]>[0]
/** Configuration for the Pi-backed query agent. */
export interface PiQueryAgentConfig { export interface PiQueryAgentConfig {
toolbox: QueryAgentToolbox toolbox: QueryAgentToolbox
apiKey?: string apiKey?: string

View File

@@ -1,8 +1,8 @@
import { ConversationEntryKind } from "@freya/core"
import { describe, expect, test } from "bun:test" import { describe, expect, test } from "bun:test"
import type { ConversationStorageEntry } from "./conversation-recording-query-agent.ts" import type { ConversationStorageEntry } from "./conversation-recording-query-agent.ts"
import { ConversationEntryKind } from "../conversations/types.ts"
import { createSessionManager } from "./session-manager.ts" import { createSessionManager } from "./session-manager.ts"
describe("createSessionManager", () => { describe("createSessionManager", () => {

View File

@@ -1,18 +1,21 @@
import { SessionManager } from "@earendil-works/pi-coding-agent" import { SessionManager } from "@earendil-works/pi-coding-agent"
import { tmpdir } from "node:os"
import type { ConversationStorageEntry } from "./conversation-recording-query-agent.ts"
import { import {
AssistantMessagePayload, AssistantMessagePayload,
ContextSummaryPayload, ContextSummaryPayload,
ConversationEntryKind, ConversationEntryKind,
UserMessagePayload, UserMessagePayload,
} from "../conversations/types.ts" } from "@freya/core"
import { tmpdir } from "node:os"
import type { ConversationStorageEntry } from "./conversation-recording-query-agent.ts"
/** Message shape accepted by Pi's SessionManager.appendMessage API. */
type PiMessage = Parameters<SessionManager["appendMessage"]>[0] type PiMessage = Parameters<SessionManager["appendMessage"]>[0]
/** Assistant message variant required when replaying stored assistant entries. */
type PiAssistantMessage = Extract<PiMessage, { role: "assistant" }> type PiAssistantMessage = Extract<PiMessage, { role: "assistant" }>
/** Inputs required to rebuild a Pi session manager from stored conversation entries. */
export interface CreateSessionManagerInput { export interface CreateSessionManagerInput {
cwd?: string cwd?: string
entries: ConversationStorageEntry[] entries: ConversationStorageEntry[]

View File

@@ -1,3 +1,4 @@
import { ConversationEntryKind, ConversationEntryVisibility } from "@freya/core"
import { beforeEach, describe, expect, mock, test } from "bun:test" import { beforeEach, describe, expect, mock, test } from "bun:test"
import { Hono } from "hono" import { Hono } from "hono"
@@ -11,7 +12,6 @@ import type {
import { mockAuthSessionMiddleware } from "../auth/session-middleware.ts" import { mockAuthSessionMiddleware } from "../auth/session-middleware.ts"
import { ConversationNotFoundError } from "./errors.ts" import { ConversationNotFoundError } from "./errors.ts"
import { registerConversationsHttpHandlers } from "./http.ts" import { registerConversationsHttpHandlers } from "./http.ts"
import { ConversationEntryKind, ConversationEntryVisibility } from "./types.ts"
const MockUserId = "k7Gx2mPqRvNwYs9TdLfA4bHcJeUo1iZn" const MockUserId = "k7Gx2mPqRvNwYs9TdLfA4bHcJeUo1iZn"
const ConversationId = "11111111-1111-4111-8111-111111111111" const ConversationId = "11111111-1111-4111-8111-111111111111"

View File

@@ -1,5 +1,6 @@
import type { Context, Hono } from "hono" import type { Context, Hono } from "hono"
import { ConversationEntryVisibility } from "@freya/core"
import { type } from "arktype" import { type } from "arktype"
import { createMiddleware } from "hono/factory" import { createMiddleware } from "hono/factory"
@@ -9,20 +10,22 @@ import type { ConversationRow } from "./storage.ts"
import { ConversationNotFoundError } from "./errors.ts" import { ConversationNotFoundError } from "./errors.ts"
import { conversations } from "./storage.ts" import { conversations } from "./storage.ts"
import { ConversationEntryVisibility } from "./types.ts"
/** Hono environment populated by the conversations route middleware. */
type Env = { type Env = {
Variables: { Variables: {
db: Database db: Database
} }
} }
/** Serialized conversation summary returned by the list endpoint. */
interface ConversationSummaryResponse { interface ConversationSummaryResponse {
id: string id: string
createdAt: string createdAt: string
updatedAt: string updatedAt: string
} }
/** Dependencies required to register conversation HTTP handlers. */
interface ConversationsHttpHandlersDeps { interface ConversationsHttpHandlersDeps {
db: Database db: Database
authSessionMiddleware: AuthSessionMiddleware authSessionMiddleware: AuthSessionMiddleware

View File

@@ -1,17 +1,18 @@
import {
AssistantMessagePayload,
AttachmentPayload,
ConversationEntryKind,
ConversationEntryVisibility,
ContextSummaryPayload,
ConversationEntryMetadata,
GenericObjectPayload,
UserMessagePayload,
type ConversationEntryPayload,
} from "@freya/core"
import { type } from "arktype"
import { and, asc, desc, eq } from "drizzle-orm" import { and, asc, desc, eq } from "drizzle-orm"
import type { Database } from "../db/index.ts" import type { Database } from "../db/index.ts"
import type {
AssistantMessagePayload,
AttachmentPayload,
ContextSummaryPayload,
ConversationEntryKind as ConversationEntryKindType,
ConversationEntryMetadata,
ConversationEntryPayload,
ConversationEntryVisibility as ConversationEntryVisibilityType,
GenericObjectPayload,
UserMessagePayload,
} from "./types.ts"
import { import {
conversationEntries, conversationEntries,
@@ -20,23 +21,20 @@ import {
user, user,
} from "../db/schema.ts" } from "../db/schema.ts"
import { ConversationNotFoundError } from "./errors.ts" import { ConversationNotFoundError } from "./errors.ts"
import {
ConversationEntryMetadata as ConversationEntryMetadataSchema,
AssistantMessagePayload as AssistantMessagePayloadSchema,
AttachmentPayload as AttachmentPayloadSchema,
ConversationEntryKind,
ConversationEntryKindInput,
ConversationEntryVisibility,
ConversationEntryVisibilityInput,
ContextSummaryPayload as ContextSummaryPayloadSchema,
GenericObjectPayload as GenericObjectPayloadSchema,
UserMessagePayload as UserMessagePayloadSchema,
} from "./types.ts"
const conversationEntryKind = type.enumerated(...Object.values(ConversationEntryKind))
const conversationEntryVisibility = type.enumerated(...Object.values(ConversationEntryVisibility))
/** Database row shape for a conversation owned by a user. */
export type ConversationRow = typeof conversationsTable.$inferSelect export type ConversationRow = typeof conversationsTable.$inferSelect
/** Database row shape for an entry in a conversation timeline. */
export type ConversationEntryRow = typeof conversationEntries.$inferSelect export type ConversationEntryRow = typeof conversationEntries.$inferSelect
/** Database row shape for an uploaded file referenced by conversations. */
export type FileRow = typeof files.$inferSelect export type FileRow = typeof files.$inferSelect
/** Input required to create a stored file record. */
export interface CreateFileInput { export interface CreateFileInput {
storageKey: string storageKey: string
originalName?: string originalName?: string
@@ -45,23 +43,27 @@ export interface CreateFileInput {
metadata?: Record<string, unknown> metadata?: Record<string, unknown>
} }
/** Input for creating a file and appending its attachment entry together. */
export interface AppendAttachmentEntryInput { export interface AppendAttachmentEntryInput {
file: CreateFileInput file: CreateFileInput
payload: AttachmentPayload payload: AttachmentPayload
visibility?: ConversationEntryVisibilityType visibility?: ConversationEntryVisibility
metadata?: ConversationEntryMetadata metadata?: ConversationEntryMetadata
} }
/** Result returned after a file-backed attachment entry is appended. */
export interface AppendAttachmentEntryResult { export interface AppendAttachmentEntryResult {
file: FileRow file: FileRow
entry: ConversationEntryRow entry: ConversationEntryRow
} }
/** Common fields accepted when appending any conversation entry. */
interface AppendConversationEntryBase { interface AppendConversationEntryBase {
visibility?: ConversationEntryVisibilityType visibility?: ConversationEntryVisibility
metadata?: ConversationEntryMetadata metadata?: ConversationEntryMetadata
} }
/** Discriminated input for appending any supported entry kind to a conversation. */
export type AppendConversationEntryInput = export type AppendConversationEntryInput =
| (AppendConversationEntryBase & { | (AppendConversationEntryBase & {
kind: typeof ConversationEntryKind.UserMessage kind: typeof ConversationEntryKind.UserMessage
@@ -92,8 +94,9 @@ export type AppendConversationEntryInput =
fileId?: never fileId?: never
}) })
/** Filters accepted when listing conversation entries. */
export interface ListConversationEntriesParams { export interface ListConversationEntriesParams {
visibility?: ConversationEntryVisibilityType visibility?: ConversationEntryVisibility
} }
export function conversations(db: Database, userId: string) { export function conversations(db: Database, userId: string) {
@@ -140,12 +143,12 @@ export function conversations(db: Database, userId: string) {
conversationId: string, conversationId: string,
input: AppendConversationEntryInput, input: AppendConversationEntryInput,
): Promise<ConversationEntryRow> { ): Promise<ConversationEntryRow> {
const kind = ConversationEntryKindInput.assert(input.kind) const kind = conversationEntryKind.assert(input.kind)
const visibility = ConversationEntryVisibilityInput.assert( const visibility = conversationEntryVisibility.assert(
input.visibility ?? defaultVisibilityForKind(kind), input.visibility ?? defaultVisibilityForKind(kind),
) )
const payload = payloadForKind(kind, input.payload) const payload = payloadForKind(kind, input.payload)
const metadata = ConversationEntryMetadataSchema.assert(input.metadata ?? {}) const metadata = ConversationEntryMetadata.assert(input.metadata ?? {})
let fileId: string | null = null let fileId: string | null = null
if (input.kind === ConversationEntryKind.Attachment) { if (input.kind === ConversationEntryKind.Attachment) {
@@ -183,11 +186,11 @@ export function conversations(db: Database, userId: string) {
conversationId: string, conversationId: string,
input: AppendAttachmentEntryInput, input: AppendAttachmentEntryInput,
): Promise<AppendAttachmentEntryResult> { ): Promise<AppendAttachmentEntryResult> {
const payload = AttachmentPayloadSchema.assert(input.payload) const payload = AttachmentPayload.assert(input.payload)
const visibility = ConversationEntryVisibilityInput.assert( const visibility = conversationEntryVisibility.assert(
input.visibility ?? defaultVisibilityForKind(ConversationEntryKind.Attachment), input.visibility ?? defaultVisibilityForKind(ConversationEntryKind.Attachment),
) )
const metadata = ConversationEntryMetadataSchema.assert(input.metadata ?? {}) const metadata = ConversationEntryMetadata.assert(input.metadata ?? {})
return db.transaction(async (tx) => { return db.transaction(async (tx) => {
if (!(await findConversationForUpdate(tx, userId, conversationId))) { if (!(await findConversationForUpdate(tx, userId, conversationId))) {
@@ -250,22 +253,22 @@ export function conversations(db: Database, userId: string) {
} }
function payloadForKind( function payloadForKind(
kind: ConversationEntryKindType, kind: ConversationEntryKind,
payload: AppendConversationEntryInput["payload"], payload: AppendConversationEntryInput["payload"],
): ConversationEntryPayload { ): ConversationEntryPayload {
switch (kind) { switch (kind) {
case ConversationEntryKind.UserMessage: case ConversationEntryKind.UserMessage:
return UserMessagePayloadSchema.assert(payload) return UserMessagePayload.assert(payload)
case ConversationEntryKind.AssistantMessage: case ConversationEntryKind.AssistantMessage:
return AssistantMessagePayloadSchema.assert(payload) return AssistantMessagePayload.assert(payload)
case ConversationEntryKind.Attachment: case ConversationEntryKind.Attachment:
return AttachmentPayloadSchema.assert(payload) return AttachmentPayload.assert(payload)
case ConversationEntryKind.ContextSummary: case ConversationEntryKind.ContextSummary:
return ContextSummaryPayloadSchema.assert(payload) return ContextSummaryPayload.assert(payload)
case ConversationEntryKind.ToolCall: case ConversationEntryKind.ToolCall:
case ConversationEntryKind.ToolResult: case ConversationEntryKind.ToolResult:
case ConversationEntryKind.SystemNote: case ConversationEntryKind.SystemNote:
return GenericObjectPayloadSchema.assert(payload) return GenericObjectPayload.assert(payload)
} }
} }
@@ -371,9 +374,7 @@ function requireRow<T>(rows: T[], message = "Expected database row"): T {
return row return row
} }
function defaultVisibilityForKind( function defaultVisibilityForKind(kind: ConversationEntryKind): ConversationEntryVisibility {
kind: ConversationEntryKindType,
): ConversationEntryVisibilityType {
switch (kind) { switch (kind) {
case ConversationEntryKind.UserMessage: case ConversationEntryKind.UserMessage:
case ConversationEntryKind.AssistantMessage: case ConversationEntryKind.AssistantMessage:

View File

@@ -1,3 +1,10 @@
import {
ConversationEntryVisibility,
type ConversationEntryKind,
type ConversationEntryMetadata,
type ConversationEntryPayload,
type ConversationEntryVisibility as ConversationEntryVisibilityType,
} from "@freya/core"
import { sql } from "drizzle-orm" import { sql } from "drizzle-orm"
import { import {
boolean, boolean,
@@ -13,14 +20,6 @@ import {
uuid, uuid,
} from "drizzle-orm/pg-core" } from "drizzle-orm/pg-core"
import {
ConversationEntryVisibility,
type ConversationEntryKind,
type ConversationEntryMetadata,
type ConversationEntryPayload,
type ConversationEntryVisibility as ConversationEntryVisibilityType,
} from "../conversations/types.ts"
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// Better Auth core tables // Better Auth core tables
// Re-exported from CLI-generated schema. // Re-exported from CLI-generated schema.

View File

@@ -1,5 +1,6 @@
import type { ActionDefinition, ContextEntry, FeedItem, FeedSource } from "@freya/core" import type { ActionDefinition, ContextEntry, FeedItem, FeedSource } from "@freya/core"
import { ConversationEntryKind } from "@freya/core"
import { LocationSource } from "@freya/source-location" import { LocationSource } from "@freya/source-location"
import { WeatherSource } from "@freya/source-weatherkit" import { WeatherSource } from "@freya/source-weatherkit"
import { beforeEach, describe, expect, mock, spyOn, test } from "bun:test" import { beforeEach, describe, expect, mock, spyOn, test } from "bun:test"
@@ -9,7 +10,6 @@ import type { AppendConversationEntryInput } from "../conversations/storage.ts"
import type { Database } from "../db/index.ts" import type { Database } from "../db/index.ts"
import type { FeedSourceProvider } from "./feed-source-provider.ts" import type { FeedSourceProvider } from "./feed-source-provider.ts"
import { ConversationEntryKind } from "../conversations/types.ts"
import { CredentialEncryptor } from "../lib/crypto.ts" import { CredentialEncryptor } from "../lib/crypto.ts"
import { import {
CredentialStorageUnavailableError, CredentialStorageUnavailableError,

View File

@@ -1,5 +1,6 @@
import type { ActionDefinition, ContextEntry, FeedItem, FeedSource } from "@freya/core" import type { ActionDefinition, ContextEntry, FeedItem, FeedSource } from "@freya/core"
import { ConversationEntryKind } from "@freya/core"
import { LocationSource } from "@freya/source-location" import { LocationSource } from "@freya/source-location"
import { describe, expect, spyOn, test } from "bun:test" import { describe, expect, spyOn, test } from "bun:test"
@@ -9,7 +10,6 @@ import type {
} from "../agent/conversation-recording-query-agent.ts" } from "../agent/conversation-recording-query-agent.ts"
import type { AppendConversationEntryInput } from "../conversations/storage.ts" import type { AppendConversationEntryInput } from "../conversations/storage.ts"
import { ConversationEntryKind } from "../conversations/types.ts"
import { UserSession } from "./user-session.ts" import { UserSession } from "./user-session.ts"
function createStubSource(id: string, items: FeedItem[] = []): FeedSource { function createStubSource(id: string, items: FeedItem[] = []): FeedSource {

View File

@@ -172,6 +172,7 @@
"version": "0.0.0", "version": "0.0.0",
"dependencies": { "dependencies": {
"@standard-schema/spec": "^1.1.0", "@standard-schema/spec": "^1.1.0",
"arktype": "^2.1.29",
}, },
"peerDependencies": { "peerDependencies": {
"@json-render/core": "*", "@json-render/core": "*",

View File

@@ -8,7 +8,8 @@
"test": "bun test ." "test": "bun test ."
}, },
"dependencies": { "dependencies": {
"@standard-schema/spec": "^1.1.0" "@standard-schema/spec": "^1.1.0",
"arktype": "^2.1.29"
}, },
"peerDependencies": { "peerDependencies": {
"@json-render/core": "*", "@json-render/core": "*",

View File

@@ -7,7 +7,7 @@ import {
ConversationEntryMetadata, ConversationEntryMetadata,
GenericObjectPayload, GenericObjectPayload,
UserMessagePayload, UserMessagePayload,
} from "./types.ts" } from "./conversation"
describe("conversation entry schemas", () => { describe("conversation entry schemas", () => {
test("parses valid user message payloads", () => { test("parses valid user message payloads", () => {

View File

@@ -1,5 +1,6 @@
import { type } from "arktype" import { type } from "arktype"
/** Entry kinds supported by the persisted conversation timeline. */
export const ConversationEntryKind = { export const ConversationEntryKind = {
UserMessage: "user_message", UserMessage: "user_message",
AssistantMessage: "assistant_message", AssistantMessage: "assistant_message",
@@ -10,17 +11,21 @@ export const ConversationEntryKind = {
SystemNote: "system_note", SystemNote: "system_note",
} as const } as const
/** Discriminator for the payload shape and handling of a conversation entry. */
export type ConversationEntryKind = export type ConversationEntryKind =
(typeof ConversationEntryKind)[keyof typeof ConversationEntryKind] (typeof ConversationEntryKind)[keyof typeof ConversationEntryKind]
/** Visibility scopes supported by stored conversation entries. */
export const ConversationEntryVisibility = { export const ConversationEntryVisibility = {
UserVisible: "user_visible", UserVisible: "user_visible",
Internal: "internal", Internal: "internal",
} as const } as const
/** Indicates whether a conversation entry should be exposed to the user. */
export type ConversationEntryVisibility = export type ConversationEntryVisibility =
(typeof ConversationEntryVisibility)[keyof typeof ConversationEntryVisibility] (typeof ConversationEntryVisibility)[keyof typeof ConversationEntryVisibility]
/** Attachment media categories accepted by conversation entries. */
export const AttachmentType = { export const AttachmentType = {
Image: "image", Image: "image",
Audio: "audio", Audio: "audio",
@@ -29,57 +34,64 @@ export const AttachmentType = {
Other: "other", Other: "other",
} as const } as const
/** File or media category associated with an attachment payload. */
export type AttachmentType = (typeof AttachmentType)[keyof typeof AttachmentType] export type AttachmentType = (typeof AttachmentType)[keyof typeof AttachmentType]
export const ConversationEntryKindInput = type.enumerated(...Object.values(ConversationEntryKind)) /** Plain text content part for a message. */
export const ConversationEntryVisibilityInput = type.enumerated( export const TextMessagePart = type({
...Object.values(ConversationEntryVisibility),
)
export const AttachmentTypeInput = type.enumerated(...Object.values(AttachmentType))
const TextMessagePart = type({
"+": "reject", "+": "reject",
type: "'text'", type: "'text'",
text: "string", text: "string",
}) })
const JsonMessagePart = type({ /** Structured JSON content part for a message. */
export const JsonMessagePart = type({
"+": "reject", "+": "reject",
type: "'json'", type: "'json'",
value: "unknown", value: "unknown",
}) })
/** Content part variants supported by user and assistant messages. */
export const MessagePart = type.or(TextMessagePart, JsonMessagePart) export const MessagePart = type.or(TextMessagePart, JsonMessagePart)
/** A structured content part inside a user or assistant message payload. */
export type MessagePart = typeof MessagePart.infer export type MessagePart = typeof MessagePart.infer
/** User-authored message entry payload. */
export const UserMessagePayload = type({ export const UserMessagePayload = type({
"+": "reject", "+": "reject",
role: "'user'", role: "'user'",
parts: MessagePart.array().atLeastLength(1), parts: MessagePart.array().atLeastLength(1),
}) })
/** Payload stored for a conversation entry containing a user message. */
export type UserMessagePayload = typeof UserMessagePayload.infer export type UserMessagePayload = typeof UserMessagePayload.infer
/** Assistant-authored message entry payload. */
export const AssistantMessagePayload = type({ export const AssistantMessagePayload = type({
"+": "reject", "+": "reject",
role: "'assistant'", role: "'assistant'",
parts: MessagePart.array().atLeastLength(1), parts: MessagePart.array().atLeastLength(1),
}) })
/** Payload stored for a conversation entry containing an assistant message. */
export type AssistantMessagePayload = typeof AssistantMessagePayload.infer export type AssistantMessagePayload = typeof AssistantMessagePayload.infer
/** Attachment entry payload. */
export const AttachmentPayload = type({ export const AttachmentPayload = type({
"+": "reject", "+": "reject",
role: type.enumerated("user", "assistant"), role: type.enumerated("user", "assistant"),
name: "string", name: "string",
mimeType: "string", mimeType: "string",
attachmentType: AttachmentTypeInput, attachmentType: type.enumerated(...Object.values(AttachmentType)),
"caption?": "string", "caption?": "string",
}) })
/** Payload stored for a conversation entry that references an uploaded file. */
export type AttachmentPayload = typeof AttachmentPayload.infer export type AttachmentPayload = typeof AttachmentPayload.infer
const ContextSummary = type({ /** Durable facts extracted from compacted conversation history. */
export const ContextSummary = type({
"+": "reject", "+": "reject",
"userIntent?": "string", "userIntent?": "string",
durableFacts: type.string.array(), durableFacts: type.string.array(),
@@ -89,6 +101,10 @@ const ContextSummary = type({
importantDetails: type.string.array(), importantDetails: type.string.array(),
}) })
/** Durable facts and follow-ups retained from compacted conversation history. */
export type ContextSummary = typeof ContextSummary.infer
/** Context-summary conversation entry payload. */
export const ContextSummaryPayload = type({ export const ContextSummaryPayload = type({
"+": "reject", "+": "reject",
covers: type({ covers: type({
@@ -101,8 +117,10 @@ export const ContextSummaryPayload = type({
"sourceEntryIds?": type.string.array(), "sourceEntryIds?": type.string.array(),
}) })
/** Payload describing a compaction summary and the sequence range it covers. */
export type ContextSummaryPayload = typeof ContextSummaryPayload.infer export type ContextSummaryPayload = typeof ContextSummaryPayload.infer
/** Model invocation metadata recorded on generated entries. */
export const ModelRunMetadata = type({ export const ModelRunMetadata = type({
"+": "reject", "+": "reject",
route: "string", route: "string",
@@ -116,18 +134,25 @@ export const ModelRunMetadata = type({
"providerRequestId?": "string", "providerRequestId?": "string",
}) })
/** Metadata describing the model run that produced a conversation entry. */
export type ModelRunMetadata = typeof ModelRunMetadata.infer export type ModelRunMetadata = typeof ModelRunMetadata.infer
/** Arbitrary metadata stored alongside conversation entries. */
export const ConversationEntryMetadata = type({ export const ConversationEntryMetadata = type({
"modelRun?": ModelRunMetadata, "modelRun?": ModelRunMetadata,
"[string]": "unknown", "[string]": "unknown",
}) })
/** Metadata bag attached to a conversation entry. */
export type ConversationEntryMetadata = typeof ConversationEntryMetadata.infer export type ConversationEntryMetadata = typeof ConversationEntryMetadata.infer
/** Generic object payload used by operational entries. */
export const GenericObjectPayload = type("Record<string, unknown>") export const GenericObjectPayload = type("Record<string, unknown>")
/** Fallback payload shape for tool calls, tool results, and system notes. */
export type GenericObjectPayload = typeof GenericObjectPayload.infer export type GenericObjectPayload = typeof GenericObjectPayload.infer
/** Union of payload shapes that can be stored on a conversation entry. */
export type ConversationEntryPayload = export type ConversationEntryPayload =
| UserMessagePayload | UserMessagePayload
| AssistantMessagePayload | AssistantMessagePayload

View File

@@ -6,6 +6,25 @@ export { Context, contextKey, serializeKey } from "./context"
export type { ActionDefinition } from "./action" export type { ActionDefinition } from "./action"
export { UnknownActionError } from "./action" export { UnknownActionError } from "./action"
// Conversation
export type { ConversationEntryPayload } from "./conversation"
export {
AssistantMessagePayload,
AttachmentPayload,
AttachmentType,
ContextSummary,
ContextSummaryPayload,
ConversationEntryKind,
ConversationEntryMetadata,
ConversationEntryVisibility,
GenericObjectPayload,
JsonMessagePart,
MessagePart,
ModelRunMetadata,
TextMessagePart,
UserMessagePayload,
} from "./conversation"
// Feed // Feed
export type { FeedItem, FeedItemRenderer, FeedItemSignals, RenderedFeedItem, Slot } from "./feed" export type { FeedItem, FeedItemRenderer, FeedItemSignals, RenderedFeedItem, Slot } from "./feed"
export { TimeRelevance } from "./feed" export { TimeRelevance } from "./feed"