mirror of
https://github.com/tiennm99/goclaw.git
synced 2026-06-10 06:10:46 +00:00
06b4eb912e
Add complete i18n support for channel config tab — field labels, help text, and select option translations for en/vi/zh. Enable DM streaming by default for Telegram and Slack channels.
199 lines
6.4 KiB
Go
199 lines
6.4 KiB
Go
package channels
|
|
|
|
import (
|
|
"context"
|
|
"log/slog"
|
|
"sync"
|
|
|
|
"github.com/nextlevelbuilder/goclaw/internal/bus"
|
|
"github.com/nextlevelbuilder/goclaw/internal/store"
|
|
)
|
|
|
|
// ChannelStream is the per-run streaming handle stored on RunContext.
|
|
// Each channel implementation returns a ChannelStream from CreateStream().
|
|
// RunContext owns the stream so concurrent runs in the same group chat
|
|
// each get their own stream — no sync.Map collision on chatID.
|
|
type ChannelStream interface {
|
|
// Update sends or edits the streaming message with the latest accumulated text.
|
|
Update(ctx context.Context, text string)
|
|
// Stop finalizes the stream (final edit/flush). Called on run.completed.
|
|
Stop(ctx context.Context) error
|
|
// MessageID returns the platform message ID of the streaming message (0 if none).
|
|
// Used to hand the message back to Send() via the channel's placeholder map.
|
|
MessageID() int
|
|
}
|
|
|
|
// RunContext tracks an active agent run for streaming/reaction event forwarding.
|
|
type RunContext struct {
|
|
ChannelName string
|
|
ChatID string
|
|
MessageID string // platform message ID (string to support Feishu "om_xxx", Telegram "12345", etc.)
|
|
Metadata map[string]string // outbound routing metadata (thread_id, local_key, group_id)
|
|
Streaming bool // whether run uses streaming (to avoid double-delivery of block replies)
|
|
BlockReplyEnabled bool // whether block.reply delivery is enabled for this run (resolved at RegisterRun time)
|
|
ToolStatusEnabled bool // whether tool name shows in streaming preview during tool execution
|
|
mu sync.Mutex
|
|
streamBuffer string // accumulated streaming text (chunks are deltas)
|
|
inToolPhase bool // true after tool.call, reset on next chunk (new LLM iteration)
|
|
stream ChannelStream // per-run stream handle (replaces per-chat sync.Map in channel impls)
|
|
thinkingBuffer string // accumulated thinking/reasoning text
|
|
hasThinking bool // true if any thinking events received this iteration
|
|
thinkingDone bool // true after first chunk arrives (reasoning→answer transition complete)
|
|
tagParseSkipped bool // true after first chunk with no <think> tags (skip re-parsing)
|
|
}
|
|
|
|
// Manager manages all registered channels, handling their lifecycle
|
|
// and routing outbound messages to the correct channel.
|
|
type Manager struct {
|
|
channels map[string]Channel
|
|
bus *bus.MessageBus
|
|
runs sync.Map // runID string → *RunContext
|
|
dispatchTask *asyncTask
|
|
mu sync.RWMutex
|
|
contactCollector *store.ContactCollector
|
|
}
|
|
|
|
type asyncTask struct {
|
|
cancel context.CancelFunc
|
|
}
|
|
|
|
// NewManager creates a new channel manager.
|
|
// Channels are registered externally via RegisterChannel.
|
|
func NewManager(msgBus *bus.MessageBus) *Manager {
|
|
return &Manager{
|
|
channels: make(map[string]Channel),
|
|
bus: msgBus,
|
|
}
|
|
}
|
|
|
|
// StartAll starts all registered channels and the outbound dispatch loop.
|
|
// The dispatcher is always started even when no channels exist yet,
|
|
// because channels may be loaded dynamically later via Reload().
|
|
func (m *Manager) StartAll(ctx context.Context) error {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
// Always start the outbound dispatcher — channels may be added later via Reload().
|
|
dispatchCtx, cancel := context.WithCancel(ctx)
|
|
m.dispatchTask = &asyncTask{cancel: cancel}
|
|
go m.dispatchOutbound(dispatchCtx)
|
|
|
|
if len(m.channels) == 0 {
|
|
slog.Warn("no channels enabled")
|
|
return nil
|
|
}
|
|
|
|
slog.Info("starting all channels")
|
|
|
|
for name, channel := range m.channels {
|
|
slog.Info("starting channel", "channel", name)
|
|
if err := channel.Start(ctx); err != nil {
|
|
slog.Error("failed to start channel", "channel", name, "error", err)
|
|
}
|
|
}
|
|
|
|
slog.Info("all channels started")
|
|
return nil
|
|
}
|
|
|
|
// StopAll gracefully stops all channels and the outbound dispatch loop.
|
|
func (m *Manager) StopAll(ctx context.Context) error {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
slog.Info("stopping all channels")
|
|
|
|
if m.dispatchTask != nil {
|
|
m.dispatchTask.cancel()
|
|
m.dispatchTask = nil
|
|
}
|
|
|
|
for name, channel := range m.channels {
|
|
slog.Info("stopping channel", "channel", name)
|
|
if err := channel.Stop(ctx); err != nil {
|
|
slog.Error("error stopping channel", "channel", name, "error", err)
|
|
}
|
|
}
|
|
|
|
slog.Info("all channels stopped")
|
|
return nil
|
|
}
|
|
|
|
// GetChannel returns a channel by name.
|
|
func (m *Manager) GetChannel(name string) (Channel, bool) {
|
|
m.mu.RLock()
|
|
defer m.mu.RUnlock()
|
|
channel, ok := m.channels[name]
|
|
return channel, ok
|
|
}
|
|
|
|
// GetStatus returns the running status of all channels.
|
|
func (m *Manager) GetStatus() map[string]any {
|
|
m.mu.RLock()
|
|
defer m.mu.RUnlock()
|
|
|
|
status := make(map[string]any)
|
|
for name, channel := range m.channels {
|
|
status[name] = map[string]any{
|
|
"enabled": true,
|
|
"running": channel.IsRunning(),
|
|
}
|
|
}
|
|
return status
|
|
}
|
|
|
|
// GetEnabledChannels returns the names of all enabled channels.
|
|
func (m *Manager) GetEnabledChannels() []string {
|
|
m.mu.RLock()
|
|
defer m.mu.RUnlock()
|
|
|
|
names := make([]string, 0, len(m.channels))
|
|
for name := range m.channels {
|
|
names = append(names, name)
|
|
}
|
|
return names
|
|
}
|
|
|
|
// RegisterChannel adds a channel to the manager.
|
|
func (m *Manager) RegisterChannel(name string, channel Channel) {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
// Propagate contact collector to channels that embed BaseChannel.
|
|
if m.contactCollector != nil {
|
|
if bc, ok := channel.(interface{ SetContactCollector(*store.ContactCollector) }); ok {
|
|
bc.SetContactCollector(m.contactCollector)
|
|
}
|
|
}
|
|
m.channels[name] = channel
|
|
}
|
|
|
|
// SetContactCollector sets the contact collector for all current and future channels.
|
|
func (m *Manager) SetContactCollector(cc *store.ContactCollector) {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
m.contactCollector = cc
|
|
for _, ch := range m.channels {
|
|
if bc, ok := ch.(interface{ SetContactCollector(*store.ContactCollector) }); ok {
|
|
bc.SetContactCollector(cc)
|
|
}
|
|
}
|
|
}
|
|
|
|
// ChannelTypeForName returns the platform type for a channel instance name.
|
|
// Reads directly from the Channel.Type() method — no separate map needed.
|
|
func (m *Manager) ChannelTypeForName(name string) string {
|
|
m.mu.RLock()
|
|
defer m.mu.RUnlock()
|
|
if ch, ok := m.channels[name]; ok {
|
|
return ch.Type()
|
|
}
|
|
return ""
|
|
}
|
|
|
|
// UnregisterChannel removes a channel from the manager.
|
|
func (m *Manager) UnregisterChannel(name string) {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
delete(m.channels, name)
|
|
}
|