mirror of
https://github.com/coder/coder.git
synced 2026-06-02 20:48:20 +00:00
edee917d88
feat: add AI chat system with agent tools and chat UI Introduce the chatd subsystem and Agents UI for AI-powered chat within Coder workspaces. - Add chatd package with chat loop, message compaction, prompt management, and LLM provider integration (OpenAI, Anthropic) - Add agent tools: create workspace, list/read templates, read/write/ edit files, execute commands - Add chat API endpoints with streaming, message editing, and durable reconnection - Add database schema and migrations for chats, chat messages, chat providers, and chat model configs - Add RBAC policies and dbauthz enforcement for chat resources - Add Agents UI pages with conversation timeline, queued messages list, diff viewer, and model configuration panel - Add comprehensive test coverage including coderd integration tests, chatd unit tests, and Storybook stories - Gate feature behind experiments flag --------- Co-authored-by: Cian Johnston <cian@coder.com> Co-authored-by: Danielle Maywood <danielle@themaywoods.com> Co-authored-by: Jeremy Ruppel <jeremy@coder.com> Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
73 lines
1.6 KiB
Go
73 lines
1.6 KiB
Go
package chattool
|
|
|
|
import (
|
|
"context"
|
|
"io"
|
|
|
|
"charm.land/fantasy"
|
|
|
|
"github.com/coder/coder/v2/codersdk/workspacesdk"
|
|
)
|
|
|
|
type ReadFileOptions struct {
|
|
GetWorkspaceConn func(context.Context) (workspacesdk.AgentConn, error)
|
|
}
|
|
|
|
type ReadFileArgs struct {
|
|
Path string `json:"path"`
|
|
Offset *int64 `json:"offset,omitempty"`
|
|
Limit *int64 `json:"limit,omitempty"`
|
|
}
|
|
|
|
func ReadFile(options ReadFileOptions) fantasy.AgentTool {
|
|
return fantasy.NewAgentTool(
|
|
"read_file",
|
|
"Read a file from the workspace.",
|
|
func(ctx context.Context, args ReadFileArgs, _ fantasy.ToolCall) (fantasy.ToolResponse, error) {
|
|
if options.GetWorkspaceConn == nil {
|
|
return fantasy.NewTextErrorResponse("workspace connection resolver is not configured"), nil
|
|
}
|
|
conn, err := options.GetWorkspaceConn(ctx)
|
|
if err != nil {
|
|
return fantasy.NewTextErrorResponse(err.Error()), nil
|
|
}
|
|
return executeReadFileTool(ctx, conn, args)
|
|
},
|
|
)
|
|
}
|
|
|
|
func executeReadFileTool(
|
|
ctx context.Context,
|
|
conn workspacesdk.AgentConn,
|
|
args ReadFileArgs,
|
|
) (fantasy.ToolResponse, error) {
|
|
if args.Path == "" {
|
|
return fantasy.NewTextErrorResponse("path is required"), nil
|
|
}
|
|
|
|
offset := int64(0)
|
|
limit := int64(0)
|
|
if args.Offset != nil {
|
|
offset = *args.Offset
|
|
}
|
|
if args.Limit != nil {
|
|
limit = *args.Limit
|
|
}
|
|
|
|
reader, mimeType, err := conn.ReadFile(ctx, args.Path, offset, limit)
|
|
if err != nil {
|
|
return fantasy.NewTextErrorResponse(err.Error()), nil
|
|
}
|
|
defer reader.Close()
|
|
|
|
data, err := io.ReadAll(reader)
|
|
if err != nil {
|
|
return fantasy.NewTextErrorResponse(err.Error()), nil
|
|
}
|
|
|
|
return toolResponse(map[string]any{
|
|
"content": string(data),
|
|
"mime_type": mimeType,
|
|
}), nil
|
|
}
|