import path from "path"
import fs from "fs"
import { Effect, Schema } from "effect"
import * as Tool from "./tool"
import { Question } from "../question"
import { Session } from "@/session/session"
import { MessageV2 } from "../session/message-v2"
import { Provider } from "@/provider/provider"
import { InstanceState } from "@/effect/instance-state"
import { MessageID, PartID } from "../session/schema"
import EXIT_DESCRIPTION from "./plan-exit.txt"
import WRITE_DESCRIPTION from "./plan-write.txt"
import ENTER_DESCRIPTION from "./plan-enter.txt"
export const Parameters = Schema.Struct({})
export const PlanExitTool = Tool.define(
"plan_exit",
Effect.gen(function* () {
const session = yield* Session.Service
const question = yield* Question.Service
const provider = yield* Provider.Service
return {
description: EXIT_DESCRIPTION,
parameters: Parameters,
execute: (_params: {}, ctx: Tool.Context) =>
Effect.gen(function* () {
const instance = yield* InstanceState.context
const info = yield* session.get(ctx.sessionID)
const plan = path.relative(instance.worktree, Session.plan(info, instance))
const answers = yield* question.ask({
sessionID: ctx.sessionID,
questions: [
{
question: `Plan at ${plan} is complete. Would you like to switch to the build agent and start implementing?`,
header: "Build Agent",
custom: false,
options: [
{ label: "Yes", description: "Switch to build agent and start implementing the plan" },
{ label: "No", description: "Stay with plan agent to continue refining the plan" },
],
},
],
tool: ctx.callID ? { messageID: ctx.messageID, callID: ctx.callID } : undefined,
})
if (answers[0]?.[0] === "No") yield* new Question.RejectedError()
const messages = yield* session.messages({ sessionID: ctx.sessionID }).pipe(Effect.orDie)
const lastUser = messages.findLast((item) => item.info.role === "user" && item.info.model)
const model =
lastUser?.info.role === "user" && lastUser.info.model ? lastUser.info.model : yield* provider.defaultModel()
const msg: MessageV2.User = {
id: MessageID.ascending(),
sessionID: ctx.sessionID,
role: "user",
time: { created: Date.now() },
agent: "build",
model,
}
yield* session.updateMessage(msg)
yield* session.updatePart({
id: PartID.ascending(),
messageID: msg.id,
sessionID: ctx.sessionID,
type: "text",
text: `The plan at ${plan} has been approved, you can now edit files. Execute the plan`,
synthetic: true,
} satisfies MessageV2.TextPart)
return {
title: "Switching to build agent",
output: "User approved switching to build agent. Wait for further instructions.",
metadata: {},
}
}).pipe(Effect.orDie),
}
}),
)
const WriteParameters = Schema.Struct({
content: Schema.String.annotate({ description: "The full plan file content, including YAML frontmatter and markdown body." }),
})
export const PlanWriteTool = Tool.define(
"plan_write",
Effect.gen(function* () {
const session = yield* Session.Service
return {
description: WRITE_DESCRIPTION,
parameters: WriteParameters,
execute: (params: Schema.Schema.Type<typeof WriteParameters>, _ctx: Tool.Context) =>
Effect.gen(function* () {
const instance = yield* InstanceState.context
const info = yield* session.get(_ctx.sessionID)
const planPath = Session.plan(info, instance)
yield* Effect.tryPromise(() =>
fs.promises.mkdir(path.dirname(planPath), { recursive: true }),
)
yield* Effect.tryPromise(() =>
fs.promises.writeFile(planPath, params.content, "utf-8"),
)
const displayPath = path.relative(instance.worktree, planPath)
return {
title: "Plan Written",
output: `Plan written to ${displayPath}`,
metadata: { path: planPath },
}
}).pipe(Effect.orDie),
}
}),
)
export const PlanEnterTool = Tool.define(
"plan_enter",
Effect.gen(function* () {
const session = yield* Session.Service
const question = yield* Question.Service
const provider = yield* Provider.Service
return {
description: ENTER_DESCRIPTION,
parameters: Schema.Struct({}),
execute: (_params: {}, ctx: Tool.Context) =>
Effect.gen(function* () {
const instance = yield* InstanceState.context
const info = yield* session.get(ctx.sessionID)
const plan = path.relative(instance.worktree, Session.plan(info, instance))
const answers = yield* question.ask({
sessionID: ctx.sessionID,
questions: [
{
question: `Would you like to switch to the plan agent and create a plan saved to ${plan}?`,
header: "Plan Mode",
custom: false,
options: [
{ label: "Yes", description: "Switch to plan agent for research and planning" },
{ label: "No", description: "Stay with build agent to continue making changes" },
],
},
],
tool: ctx.callID ? { messageID: ctx.messageID, callID: ctx.callID } : undefined,
})
if (answers[0]?.[0] === "No") yield* new Question.RejectedError()
const messages = yield* session.messages({ sessionID: ctx.sessionID }).pipe(Effect.orDie)
const lastUser = messages.findLast((item) => item.info.role === "user" && item.info.model)
const model =
lastUser?.info.role === "user" && lastUser.info.model ? lastUser.info.model : yield* provider.defaultModel()
const userMsg: MessageV2.User = {
id: MessageID.ascending(),
sessionID: ctx.sessionID,
role: "user",
time: { created: Date.now() },
agent: "plan",
model,
}
yield* session.updateMessage(userMsg)
yield* session.updatePart({
id: PartID.ascending(),
messageID: userMsg.id,
sessionID: ctx.sessionID,
type: "text",
text: "User has requested to enter plan mode. Switch to plan mode and begin planning.",
synthetic: true,
} satisfies MessageV2.TextPart)
return {
title: "Switching to plan agent",
output: `User confirmed to switch to plan mode. A new message has been created to switch you to plan mode. The plan file will be at ${plan}. Begin planning.`,
metadata: {},
}
}).pipe(Effect.orDie),
}
}),
)