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>
173 lines
4.0 KiB
Go
173 lines
4.0 KiB
Go
package coderd
|
|
|
|
import (
|
|
"context"
|
|
"net/http"
|
|
"net/url"
|
|
|
|
"github.com/google/uuid"
|
|
"golang.org/x/xerrors"
|
|
|
|
"github.com/coder/coder/v2/coderd/chatd"
|
|
"github.com/coder/coder/v2/codersdk"
|
|
"github.com/coder/websocket"
|
|
)
|
|
|
|
// RelaySourceHeader marks replica-relayed stream requests.
|
|
const RelaySourceHeader = "X-Coder-Relay-Source-Replica"
|
|
|
|
const (
|
|
authorizationHeader = "Authorization"
|
|
cookieHeader = "Cookie"
|
|
)
|
|
|
|
// newRemotePartsProvider creates a RemotePartsProvider that dials a remote
|
|
// replica's stream endpoint to fetch message_part events. It filters to only
|
|
// forward message_part events since durable events come via pubsub.
|
|
func newRemotePartsProvider(
|
|
resolveReplicaAddress func(context.Context, uuid.UUID) (string, bool),
|
|
replicaHTTPClient *http.Client,
|
|
replicaID uuid.UUID,
|
|
) chatd.RemotePartsProvider {
|
|
return func(
|
|
ctx context.Context,
|
|
chatID uuid.UUID,
|
|
workerID uuid.UUID,
|
|
requestHeader http.Header,
|
|
) (
|
|
[]codersdk.ChatStreamEvent,
|
|
<-chan codersdk.ChatStreamEvent,
|
|
func(),
|
|
error,
|
|
) {
|
|
address, ok := resolveReplicaAddress(ctx, workerID)
|
|
if !ok {
|
|
return nil, nil, nil, xerrors.New("worker replica not found")
|
|
}
|
|
|
|
baseURL, err := url.Parse(address)
|
|
if err != nil {
|
|
return nil, nil, nil, xerrors.Errorf("parse relay address %q: %w", address, err)
|
|
}
|
|
relayCtx, relayCancel := context.WithCancel(ctx)
|
|
sdkClient := codersdk.New(baseURL)
|
|
sdkClient.HTTPClient = replicaHTTPClient
|
|
sdkClient.SessionTokenProvider = relayHeaderTokenProvider{
|
|
header: relayHeaders(requestHeader, replicaID),
|
|
}
|
|
sourceEvents, sourceStream, err := sdkClient.StreamChat(relayCtx, chatID)
|
|
if err != nil {
|
|
relayCancel()
|
|
return nil, nil, nil, xerrors.Errorf("dial relay stream: %w", err)
|
|
}
|
|
|
|
snapshot := make([]codersdk.ChatStreamEvent, 0, 100)
|
|
preloaded := make([]codersdk.ChatStreamEvent, 0, 100)
|
|
drainInitial:
|
|
for len(snapshot) < cap(snapshot) {
|
|
select {
|
|
case <-relayCtx.Done():
|
|
_ = sourceStream.Close()
|
|
relayCancel()
|
|
return nil, nil, nil, xerrors.Errorf("dial relay stream: %w", relayCtx.Err())
|
|
case event, ok := <-sourceEvents:
|
|
if !ok {
|
|
break drainInitial
|
|
}
|
|
if event.Type != codersdk.ChatStreamEventTypeMessagePart {
|
|
continue
|
|
}
|
|
snapshot = append(snapshot, event)
|
|
preloaded = append(preloaded, event)
|
|
default:
|
|
break drainInitial
|
|
}
|
|
}
|
|
|
|
events := make(chan codersdk.ChatStreamEvent, 128)
|
|
|
|
go func() {
|
|
defer close(events)
|
|
defer relayCancel()
|
|
defer func() {
|
|
_ = sourceStream.Close()
|
|
}()
|
|
|
|
for _, event := range preloaded {
|
|
select {
|
|
case events <- event:
|
|
case <-relayCtx.Done():
|
|
return
|
|
}
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case <-relayCtx.Done():
|
|
return
|
|
case event, ok := <-sourceEvents:
|
|
if !ok {
|
|
return
|
|
}
|
|
if event.Type != codersdk.ChatStreamEventTypeMessagePart {
|
|
continue
|
|
}
|
|
select {
|
|
case events <- event:
|
|
case <-relayCtx.Done():
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}()
|
|
|
|
cancel := func() {
|
|
relayCancel()
|
|
_ = sourceStream.Close()
|
|
}
|
|
return snapshot, events, cancel, nil
|
|
}
|
|
}
|
|
|
|
type relayHeaderTokenProvider struct {
|
|
header http.Header
|
|
}
|
|
|
|
func (p relayHeaderTokenProvider) AsRequestOption() codersdk.RequestOption {
|
|
return func(req *http.Request) {
|
|
for key, values := range p.header {
|
|
for _, value := range values {
|
|
req.Header.Add(key, value)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (p relayHeaderTokenProvider) SetDialOption(opts *websocket.DialOptions) {
|
|
if opts.HTTPHeader == nil {
|
|
opts.HTTPHeader = make(http.Header)
|
|
}
|
|
for key, values := range p.header {
|
|
for _, value := range values {
|
|
opts.HTTPHeader.Add(key, value)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (p relayHeaderTokenProvider) GetSessionToken() string {
|
|
return p.header.Get(codersdk.SessionTokenHeader)
|
|
}
|
|
|
|
func relayHeaders(source http.Header, replicaID uuid.UUID) http.Header {
|
|
header := make(http.Header)
|
|
if source != nil {
|
|
for _, key := range []string{codersdk.SessionTokenHeader, authorizationHeader, cookieHeader} {
|
|
for _, value := range source.Values(key) {
|
|
header.Add(key, value)
|
|
}
|
|
}
|
|
}
|
|
header.Set(RelaySourceHeader, replicaID.String())
|
|
return header
|
|
}
|