mirror of
https://github.com/kennethnym/freya
synced 2026-06-23 18:05:11 +01:00
refactor: move conversation types to core (#149)
This commit is contained in:
@@ -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,
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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 {
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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", () => {
|
||||||
|
|||||||
@@ -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[]
|
||||||
|
|||||||
@@ -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"
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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:
|
||||||
|
|||||||
@@ -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.
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
@@ -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 {
|
||||||
|
|||||||
1
bun.lock
1
bun.lock
@@ -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": "*",
|
||||||
|
|||||||
@@ -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": "*",
|
||||||
|
|||||||
@@ -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", () => {
|
||||||
@@ -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
|
||||||
@@ -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"
|
||||||
|
|||||||
Reference in New Issue
Block a user