import { describe, expect } from "bun:test" import { Effect, Layer, Stream } from "effect" import { AgentV2 } from "@opencode-ai/core/agent" import { eq } from "drizzle-orm" import { Database } from "@opencode-ai/core/database/database" import { EventV2 } from "@opencode-ai/core/event" import { EventTable } from "@opencode-ai/core/event/sql" import { Location } from "@opencode-ai/core/location" import { ModelV2 } from "@opencode-ai/core/model" import { ProjectV2 } from "@opencode-ai/core/project" import { ProviderV2 } from "@opencode-ai/core/provider" import { AbsolutePath } from "@opencode-ai/core/schema" import { SessionV2 } from "@opencode-ai/core/session" import { SessionV1 } from "@opencode-ai/core/v1/session" import { Prompt } from "@opencode-ai/core/session/prompt" import { SessionProjector } from "@opencode-ai/core/session/projector" import { SessionExecution } from "@opencode-ai/core/session/execution" import { SessionInput } from "@opencode-ai/core/session/input" import { SessionTable } from "@opencode-ai/core/session/sql" import { SessionStore } from "@opencode-ai/core/session/store" import { WorkspaceV2 } from "@opencode-ai/core/workspace" import { testEffect } from "./lib/effect" const database = Database.layerFromPath(":memory:") const events = EventV2.layer.pipe(Layer.provide(database)) const projects = Layer.succeed( ProjectV2.Service, ProjectV2.Service.of({ resolve: (directory) => Effect.succeed({ id: ProjectV2.ID.global, directory }), directories: () => Effect.succeed([]), commit: () => Effect.void, }), ) const projector = SessionProjector.layer.pipe(Layer.provide(events), Layer.provide(database)) const store = SessionStore.layer.pipe(Layer.provide(database)) const sessions = SessionV2.layer.pipe( Layer.provide(events), Layer.provide(database), Layer.provide(store), Layer.provide(projects), Layer.provide(SessionExecution.noopLayer), ) const it = testEffect( Layer.mergeAll(database, events, projects, projector, store, SessionExecution.noopLayer, sessions), ) const location = Location.Ref.make({ directory: AbsolutePath.make("/project") }) const id = SessionV2.ID.create() describe("SessionV2.create", () => { it.effect("derives stable namespaced external IDs", () => Effect.sync(() => { const input = { namespace: "opencord.agent-thread", key: "thread-1" } expect(SessionV2.ID.fromExternal(input)).toBe(SessionV2.ID.fromExternal(input)) expect(SessionV2.ID.fromExternal(input)).toMatch(/^ses_[a-f0-9]{64}$/) expect(SessionV2.ID.fromExternal({ ...input, namespace: "another-app" })).not.toBe( SessionV2.ID.fromExternal(input), ) expect(SessionV2.ID.fromExternal({ namespace: "a:b", key: "c" })).not.toBe( SessionV2.ID.fromExternal({ namespace: "a", key: "b:c" }), ) }), ) it.effect("creates a fresh projected session when the ID is omitted", () => Effect.gen(function* () { const session = yield* SessionV2.Service const first = yield* session.create({ location }) const second = yield* session.create({ location }) expect(second.id).not.toBe(first.id) expect(yield* session.list()).toHaveLength(2) }), ) it.effect("returns the original session when the ID is retried", () => Effect.gen(function* () { const session = yield* SessionV2.Service const input = { id, location } const first = yield* session.create(input) const retried = yield* session.create(input) expect(retried).toEqual(first) expect(yield* session.list()).toEqual([first]) }), ) it.effect("stores supplied immutable create attributes", () => Effect.gen(function* () { const session = yield* SessionV2.Service const workspaceID = WorkspaceV2.ID.make("wrk_test") const model = ModelV2.Ref.make({ id: ModelV2.ID.make("sonnet"), providerID: ProviderV2.ID.anthropic, variant: ModelV2.VariantID.make("fast"), }) expect( yield* session.create({ location: Location.Ref.make({ directory: location.directory, workspaceID }), agent: AgentV2.ID.make("build"), model, }), ).toMatchObject({ location: { directory: location.directory, workspaceID }, agent: "build", model }) }), ) it.effect("returns the existing Session when one ID is reused with different create arguments", () => Effect.gen(function* () { const session = yield* SessionV2.Service const created = yield* session.create({ id, location }) const changed = [ { id, location: Location.Ref.make({ directory: AbsolutePath.make("/other") }) }, { id, location, agent: AgentV2.ID.make("build") }, { id, location, model: ModelV2.Ref.make({ id: ModelV2.ID.make("sonnet"), providerID: ProviderV2.ID.anthropic }), }, ] for (const input of changed) { expect(yield* session.create(input)).toEqual(created) } expect(yield* session.list()).toHaveLength(1) }), ) it.effect("returns one recorded session to concurrent exact retries", () => Effect.gen(function* () { const session = yield* SessionV2.Service const input = { id, location } const created = yield* Effect.all([session.create(input), session.create(input)], { concurrency: "unbounded" }) expect(created[1]).toEqual(created[0]) expect(yield* session.list()).toEqual([created[0]]) }), ) it.effect("returns the current Session projection after updates", () => Effect.gen(function* () { const session = yield* SessionV2.Service const { db } = yield* Database.Service const input = { id, location } const created = yield* session.create(input) yield* db.update(SessionTable).set({ agent: "build" }).where(eq(SessionTable.id, id)).run().pipe(Effect.orDie) expect(yield* session.create(input)).toMatchObject({ id: created.id, agent: "build" }) }), ) it.effect("returns the current Session projection after projected updates", () => Effect.gen(function* () { const session = yield* SessionV2.Service const events = yield* EventV2.Service const input = { id, location } const created = yield* session.create(input) yield* events.publish(SessionV1.Event.Updated, { sessionID: id, info: SessionV1.SessionInfo.make({ id, slug: "updated", version: "test", projectID: created.projectID, directory: created.location.directory, title: "updated", agent: "build", time: { created: 0, updated: 1 }, }), }) expect(yield* session.create(input)).toMatchObject({ id, agent: "build" }) }), ) it.effect("persists creation through the existing legacy created event", () => Effect.gen(function* () { const session = yield* SessionV2.Service const { db } = yield* Database.Service const created = yield* session.create({ location }) expect( yield* db.select().from(EventTable).where(eq(EventTable.aggregate_id, created.id)).all().pipe(Effect.orDie), ).toMatchObject([{ type: EventV2.versionedType(SessionV1.Event.Created.type, 1) }]) }), ) it.effect("persists caller-ID creation through the existing created event", () => Effect.gen(function* () { const session = yield* SessionV2.Service const { db } = yield* Database.Service const created = yield* session.create({ id, location }) expect( yield* db.select().from(EventTable).where(eq(EventTable.aggregate_id, created.id)).get().pipe(Effect.orDie), ).toMatchObject({ data: { sessionID: id }, }) }), ) it.effect("omits legacy creation rows from the V2 Session event stream", () => Effect.gen(function* () { const session = yield* SessionV2.Service const events = yield* EventV2.Service const { db } = yield* Database.Service const created = yield* session.create({ location }) yield* session.prompt({ sessionID: created.id, prompt: new Prompt({ text: "Hello" }), resume: false }) yield* SessionInput.promoteSteers(db, events, created.id) expect( Array.from(yield* session.events({ sessionID: created.id }).pipe(Stream.take(1), Stream.runCollect)), ).toMatchObject([{ cursor: 1, event: { type: "session.next.prompted", data: { prompt: { text: "Hello" } } } }]) }), ) it.effect("does not mask unrelated created projector defects", () => Effect.gen(function* () { const session = yield* SessionV2.Service const event = yield* EventV2.Service const defect = new Error("unrelated projector defect") yield* event.project(SessionV1.Event.Created, () => Effect.die(defect)) expect(yield* session.create({ id, location }).pipe(Effect.catchDefect(Effect.succeed))).toBe(defect) }), ) it.effect("reports unfinished Session operations as unavailable", () => Effect.gen(function* () { const session = yield* SessionV2.Service const created = yield* session.create({ location }) const unavailable = ( effect: Effect.Effect, ) => effect.pipe( Effect.flip, Effect.map((error) => (error instanceof SessionV2.OperationUnavailableError ? error.operation : "not-found")), ) expect(yield* unavailable(session.move({ sessionID: created.id, location }))).toBe("move") expect(yield* unavailable(session.shell({ sessionID: created.id, command: "pwd" }))).toBe("shell") expect(yield* unavailable(session.skill({ sessionID: created.id, skill: "review" }))).toBe("skill") expect(yield* unavailable(session.switchAgent({ sessionID: created.id, agent: "build" }))).toBe("switchAgent") expect( yield* unavailable( session.switchModel({ sessionID: created.id, model: ModelV2.Ref.make({ id: ModelV2.ID.make("sonnet"), providerID: ProviderV2.ID.anthropic }), }), ), ).toBe("switchModel") }), ) })