Files
chenhg5-cc-connect/core/api.go
2026-06-23 07:20:17 +08:00

834 lines
23 KiB
Go

package core
import (
"encoding/json"
"errors"
"fmt"
"io"
"log/slog"
"net"
"net/http"
"os"
"path/filepath"
"strings"
"sync"
"time"
)
// DefaultMaxAttachmentSize is the default per-attachment size limit (50 MiB)
// applied by the /send API and `cc-connect send` when max_attachment_size_mb
// is unset. Exported so cmd/cc-connect can resolve the same default.
const DefaultMaxAttachmentSize int64 = 50 << 20
// APIServer exposes a local Unix socket API for external tools (e.g. cron jobs)
// to send messages to active sessions.
type APIServer struct {
socketPath string
listener net.Listener
server *http.Server
mux *http.ServeMux
engines map[string]*Engine // project name → engine
cron *CronScheduler
timer *TimerScheduler
relay *RelayManager
// maxAttachmentBytes caps the raw size of a single attachment accepted by
// /send; the request body limit in handleSend is derived from it (base64
// expansion + envelope). Defaults to DefaultMaxAttachmentSize.
maxAttachmentBytes int64
mu sync.RWMutex
}
// SendRequest is the JSON body for POST /send.
//
// Audios and Videos are kept separate from Files so the engine can
// dispatch them to AudioSender / VideoSender (native voice / video
// bubble) instead of FileSender (generic file download). The fields
// reuse FileAttachment as the wire format because audio/video clips
// are byte blobs with a name + mime — the dedicated typing happens at
// the dispatch layer in engine.go. See cc-connect internal task
// t-20260615-cqjbk1.
type SendRequest struct {
Project string `json:"project"`
SessionKey string `json:"session_key"`
Message string `json:"message"`
WorkDir string `json:"work_dir,omitempty"`
CWD string `json:"cwd,omitempty"`
TTSText string `json:"tts_text,omitempty"`
Images []ImageAttachment `json:"images,omitempty"`
Files []FileAttachment `json:"files,omitempty"`
Audios []FileAttachment `json:"audios,omitempty"`
Videos []FileAttachment `json:"videos,omitempty"`
AtUsers []string `json:"at_users,omitempty"`
AtAll bool `json:"at_all,omitempty"`
}
// NewAPIServer creates an API server on a Unix socket.
func NewAPIServer(dataDir string) (*APIServer, error) {
sockDir := filepath.Join(dataDir, "run")
if err := os.MkdirAll(sockDir, 0o755); err != nil {
return nil, fmt.Errorf("create run dir: %w", err)
}
sockPath := filepath.Join(sockDir, "api.sock")
// Remove stale socket
os.Remove(sockPath)
listener, err := net.Listen("unix", sockPath)
if err != nil {
return nil, fmt.Errorf("listen unix socket: %w", err)
}
if err := os.Chmod(sockPath, 0o600); err != nil {
_ = listener.Close()
return nil, fmt.Errorf("chmod socket: %w", err)
}
s := &APIServer{
socketPath: sockPath,
listener: listener,
mux: http.NewServeMux(),
engines: make(map[string]*Engine),
maxAttachmentBytes: DefaultMaxAttachmentSize,
}
s.mux.HandleFunc("/send", s.handleSend)
s.mux.HandleFunc("/sessions", s.handleSessions)
s.mux.HandleFunc("/cron/add", s.handleCronAdd)
s.mux.HandleFunc("/cron/list", s.handleCronList)
s.mux.HandleFunc("/cron/info", s.handleCronInfo)
s.mux.HandleFunc("/cron/edit", s.handleCronEdit)
s.mux.HandleFunc("/cron/del", s.handleCronDel)
s.mux.HandleFunc("/timer/add", s.handleTimerAdd)
s.mux.HandleFunc("/timer/list", s.handleTimerList)
s.mux.HandleFunc("/timer/info", s.handleTimerInfo)
s.mux.HandleFunc("/timer/del", s.handleTimerDel)
s.mux.HandleFunc("/cron/exec", s.handleCronExec)
s.mux.HandleFunc("/cron/run", s.handleCronExec)
s.mux.HandleFunc("/relay/send", s.handleRelaySend)
s.mux.HandleFunc("/relay/bind", s.handleRelayBind)
s.mux.HandleFunc("/relay/binding", s.handleRelayBinding)
return s, nil
}
func (s *APIServer) SocketPath() string {
return s.socketPath
}
func (s *APIServer) RegisterEngine(name string, e *Engine) {
s.mu.Lock()
defer s.mu.Unlock()
s.engines[name] = e
if s.relay != nil {
s.relay.RegisterEngine(name, e)
}
}
func (s *APIServer) SetRelayManager(rm *RelayManager) {
s.relay = rm
}
func (s *APIServer) RelayManager() *RelayManager {
return s.relay
}
func (s *APIServer) SetCronScheduler(cs *CronScheduler) {
s.cron = cs
}
func (s *APIServer) SetTimerScheduler(ts *TimerScheduler) {
s.timer = ts
}
// SetMaxAttachmentSize overrides the per-attachment size limit (bytes) used by
// /send. Non-positive values are ignored so the default is retained. Safe to
// call at any time, including from the config reload path: it is guarded by
// s.mu because handleSend reads the limit concurrently.
func (s *APIServer) SetMaxAttachmentSize(bytes int64) {
s.mu.Lock()
defer s.mu.Unlock()
if bytes > 0 {
s.maxAttachmentBytes = bytes
}
}
// sendBodyEnvelope is the slack added on top of the base64-expanded attachment
// limit when sizing the /send request body: it covers the JSON envelope (field
// names, message text, metadata) and a few sub-limit attachments.
const sendBodyEnvelope int64 = 8 << 20 // 8 MiB
// sendBodyLimit returns the maximum accepted /send request body size in bytes.
// It is derived from the per-attachment limit to accommodate base64 expansion
// (~4/3) plus envelope slack, falling back to DefaultMaxAttachmentSize when no
// limit has been set (e.g. APIServer zero value in tests). Callers in hot paths
// (handleSend) run concurrently with SetMaxAttachmentSize, so the read is
// guarded by s.mu.
func (s *APIServer) sendBodyLimit() int64 {
s.mu.RLock()
defer s.mu.RUnlock()
limit := s.maxAttachmentBytes
if limit <= 0 {
limit = DefaultMaxAttachmentSize
}
return limit*4/3 + sendBodyEnvelope
}
func (s *APIServer) Start() {
s.server = &http.Server{Handler: s.mux}
go func() {
if err := s.server.Serve(s.listener); err != nil && err != http.ErrServerClosed {
slog.Error("api server error", "error", err)
}
}()
slog.Info("api server started", "socket", s.socketPath)
}
func (s *APIServer) Stop() {
if s.server != nil {
if err := s.server.Close(); err != nil && err != http.ErrServerClosed {
slog.Debug("api server close failed", "error", err)
}
}
if err := os.Remove(s.socketPath); err != nil && !os.IsNotExist(err) {
slog.Debug("api server remove socket failed", "error", err)
}
}
func apiJSON(w http.ResponseWriter, status int, v any) {
w.Header().Set("Content-Type", "application/json")
w.WriteHeader(status)
if err := json.NewEncoder(w).Encode(v); err != nil {
slog.Error("api server: write JSON failed", "error", err)
}
}
func (s *APIServer) handleSend(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "POST only", http.StatusMethodNotAllowed)
return
}
// Attachments travel base64-encoded inside the JSON body (~4/3 expansion)
// plus the request envelope, so size the reader to fit one max-size
// attachment with overhead to spare. The previous hard-coded 52 MB cap was
// smaller than a single 50 MB attachment after base64 encoding and would
// reject valid sends; deriving it from maxAttachmentBytes keeps the body
// limit in step with the configured attachment limit.
var req SendRequest
if err := json.NewDecoder(io.LimitReader(r.Body, s.sendBodyLimit())).Decode(&req); err != nil {
http.Error(w, "invalid JSON: "+err.Error(), http.StatusBadRequest)
return
}
if req.Message == "" && strings.TrimSpace(req.TTSText) == "" && len(req.Images) == 0 && len(req.Files) == 0 && len(req.Audios) == 0 && len(req.Videos) == 0 {
http.Error(w, "message, tts_text, or attachment is required", http.StatusBadRequest)
return
}
s.mu.RLock()
var engine *Engine
var ok bool
if req.Project != "" {
engine, ok = s.engines[req.Project]
} else if len(s.engines) == 1 {
// No project specified and only one engine: use it by default.
// Do NOT silently fall back when a non-empty project name is unknown —
// that misroutes the message to the wrong engine. Mirrors the resolve
// pattern in webhook.go and handleCronAdd.
for _, e := range s.engines {
engine = e
ok = true
}
}
s.mu.RUnlock()
if !ok {
if req.Project == "" {
http.Error(w, "project is required (multiple projects configured)", http.StatusBadRequest)
return
}
http.Error(w, fmt.Sprintf("project %q not found", req.Project), http.StatusNotFound)
return
}
workDir := req.WorkDir
if workDir == "" {
workDir = req.CWD
}
if req.Message != "" || len(req.Images) > 0 || len(req.Files) > 0 {
if err := engine.SendToSessionWithOptions(req.SessionKey, req.Message, req.Images, req.Files, SendOptions{WorkDir: workDir, AtUsers: req.AtUsers, AtAll: req.AtAll}); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
}
if len(req.Audios) > 0 {
if err := engine.SendAudiosToSession(req.SessionKey, req.Audios); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
}
if len(req.Videos) > 0 {
if err := engine.SendVideosToSession(req.SessionKey, req.Videos); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
}
if strings.TrimSpace(req.TTSText) != "" {
if err := engine.SendTTSToSession(req.SessionKey, req.TTSText); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
}
apiJSON(w, http.StatusOK, map[string]string{"status": "ok"})
}
func (s *APIServer) handleSessions(w http.ResponseWriter, r *http.Request) {
s.mu.RLock()
defer s.mu.RUnlock()
type sessionInfo struct {
Project string `json:"project"`
SessionKey string `json:"session_key"`
Platform string `json:"platform"`
}
var result []sessionInfo
for name, e := range s.engines {
e.interactiveMu.Lock()
for key, state := range e.interactiveStates {
if state.platform != nil {
result = append(result, sessionInfo{
Project: name,
SessionKey: key,
Platform: state.platform.Name(),
})
}
}
e.interactiveMu.Unlock()
}
apiJSON(w, http.StatusOK, result)
}
// ── Cron API ───────────────────────────────────────────────────
// CronAddRequest is the JSON body for POST /cron/add.
type CronAddRequest struct {
Project string `json:"project"`
SessionKey string `json:"session_key"`
CronExpr string `json:"cron_expr"`
Prompt string `json:"prompt"`
Exec string `json:"exec"`
WorkDir string `json:"work_dir"`
Description string `json:"description"`
Silent *bool `json:"silent,omitempty"`
SessionMode string `json:"session_mode,omitempty"`
Mode string `json:"mode,omitempty"`
TimeoutMins *int `json:"timeout_mins,omitempty"`
}
func (s *APIServer) handleCronAdd(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "POST only", http.StatusMethodNotAllowed)
return
}
if s.cron == nil {
http.Error(w, "cron scheduler not available", http.StatusServiceUnavailable)
return
}
var req CronAddRequest
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
http.Error(w, "invalid JSON: "+err.Error(), http.StatusBadRequest)
return
}
if req.CronExpr == "" {
http.Error(w, "cron_expr is required", http.StatusBadRequest)
return
}
if req.Prompt == "" && req.Exec == "" {
http.Error(w, "either prompt or exec is required", http.StatusBadRequest)
return
}
if req.Prompt != "" && req.Exec != "" {
http.Error(w, "prompt and exec are mutually exclusive", http.StatusBadRequest)
return
}
// Resolve project: use provided, or pick single engine
project := req.Project
if project == "" {
s.mu.RLock()
if len(s.engines) == 1 {
for name := range s.engines {
project = name
}
}
s.mu.RUnlock()
}
if project == "" {
http.Error(w, "project is required (multiple projects configured)", http.StatusBadRequest)
return
}
// Resolve session_key: use provided, or auto-detect from active sessions
sessionKey := req.SessionKey
if sessionKey == "" {
s.mu.RLock()
engine := s.engines[project]
s.mu.RUnlock()
if engine != nil {
keys := engine.ActiveSessionKeys()
if len(keys) == 1 {
sessionKey = keys[0]
slog.Debug("auto-detected session_key for cron job", "session_key", sessionKey)
}
}
}
if sessionKey == "" {
http.Error(w, "session_key is required: set CC_SESSION_KEY env, pass --session-key, or ensure exactly one active session exists", http.StatusBadRequest)
return
}
job := &CronJob{
ID: GenerateCronID(),
Project: project,
SessionKey: sessionKey,
CronExpr: req.CronExpr,
Prompt: req.Prompt,
Exec: req.Exec,
WorkDir: req.WorkDir,
Description: req.Description,
Enabled: true,
Silent: req.Silent,
SessionMode: NormalizeCronSessionMode(req.SessionMode),
Mode: req.Mode,
TimeoutMins: req.TimeoutMins,
}
job.CreatedAt = time.Now()
if err := s.cron.AddJob(job); err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
apiJSON(w, http.StatusOK, job)
}
func (s *APIServer) handleCronList(w http.ResponseWriter, r *http.Request) {
if s.cron == nil {
http.Error(w, "cron scheduler not available", http.StatusServiceUnavailable)
return
}
project := r.URL.Query().Get("project")
var jobs []*CronJob
if project != "" {
jobs = s.cron.Store().ListByProject(project)
} else {
jobs = s.cron.Store().List()
}
apiJSON(w, http.StatusOK, jobs)
}
func (s *APIServer) handleCronDel(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "POST only", http.StatusMethodNotAllowed)
return
}
if s.cron == nil {
http.Error(w, "cron scheduler not available", http.StatusServiceUnavailable)
return
}
var req struct {
ID string `json:"id"`
}
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
http.Error(w, "invalid JSON: "+err.Error(), http.StatusBadRequest)
return
}
if req.ID == "" {
http.Error(w, "id is required", http.StatusBadRequest)
return
}
if s.cron.RemoveJob(req.ID) {
apiJSON(w, http.StatusOK, map[string]string{"status": "ok"})
} else {
http.Error(w, fmt.Sprintf("job %q not found", req.ID), http.StatusNotFound)
}
}
func (s *APIServer) handleCronExec(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "POST only", http.StatusMethodNotAllowed)
return
}
if s.cron == nil {
http.Error(w, "cron scheduler not available", http.StatusServiceUnavailable)
return
}
var req struct {
ID string `json:"id"`
}
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
http.Error(w, "invalid JSON: "+err.Error(), http.StatusBadRequest)
return
}
if req.ID == "" {
http.Error(w, "id is required", http.StatusBadRequest)
return
}
if err := s.cron.RunJobNow(req.ID); err != nil {
if errors.Is(err, ErrCronJobNotFound) {
http.Error(w, err.Error(), http.StatusNotFound)
return
}
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
apiJSON(w, http.StatusAccepted, map[string]string{
"id": req.ID,
"status": "triggered",
})
}
func (s *APIServer) handleCronInfo(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodGet {
http.Error(w, "GET only", http.StatusMethodNotAllowed)
return
}
if s.cron == nil {
http.Error(w, "cron scheduler not available", http.StatusServiceUnavailable)
return
}
id := r.URL.Query().Get("id")
if id == "" {
http.Error(w, "id is required", http.StatusBadRequest)
return
}
job := s.cron.store.Get(id)
if job == nil {
http.Error(w, fmt.Sprintf("job %q not found", id), http.StatusNotFound)
return
}
apiJSON(w, http.StatusOK, job)
}
func (s *APIServer) handleCronEdit(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "POST only", http.StatusMethodNotAllowed)
return
}
if s.cron == nil {
http.Error(w, "cron scheduler not available", http.StatusServiceUnavailable)
return
}
var req struct {
ID string `json:"id"`
Field string `json:"field"`
Value any `json:"value"`
}
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
http.Error(w, "invalid JSON: "+err.Error(), http.StatusBadRequest)
return
}
if req.ID == "" {
http.Error(w, "id is required", http.StatusBadRequest)
return
}
if req.Field == "" {
http.Error(w, "field is required", http.StatusBadRequest)
return
}
if req.Value == nil {
http.Error(w, "value is required", http.StatusBadRequest)
return
}
if err := s.cron.UpdateJob(req.ID, req.Field, req.Value); err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
// Return updated job
job := s.cron.Store().Get(req.ID)
apiJSON(w, http.StatusOK, job)
}
// ── Timer API ─────────────────────────────────────────────────
// TimerAddRequest is the JSON body for POST /timer/add.
type TimerAddRequest struct {
Project string `json:"project"`
SessionKey string `json:"session_key"`
Delay string `json:"delay"` // relative ("2h") or absolute ISO time
Prompt string `json:"prompt"`
Exec string `json:"exec"`
WorkDir string `json:"work_dir"`
Description string `json:"description"`
Silent *bool `json:"silent,omitempty"`
Mute bool `json:"mute,omitempty"`
SessionMode string `json:"session_mode,omitempty"`
Mode string `json:"mode,omitempty"`
TimeoutMins *int `json:"timeout_mins,omitempty"`
}
func (s *APIServer) handleTimerAdd(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "POST only", http.StatusMethodNotAllowed)
return
}
if s.timer == nil {
http.Error(w, "timer scheduler not available", http.StatusServiceUnavailable)
return
}
var req TimerAddRequest
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
http.Error(w, "invalid JSON: "+err.Error(), http.StatusBadRequest)
return
}
if req.Delay == "" {
http.Error(w, "delay is required (e.g. 2h, 30m, or ISO time)", http.StatusBadRequest)
return
}
if req.Prompt == "" && req.Exec == "" {
http.Error(w, "either prompt or exec is required", http.StatusBadRequest)
return
}
if req.Prompt != "" && req.Exec != "" {
http.Error(w, "prompt and exec are mutually exclusive", http.StatusBadRequest)
return
}
fireAt, err := ParseDelayOrTime(req.Delay)
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
project := req.Project
if project == "" {
s.mu.RLock()
if len(s.engines) == 1 {
for name := range s.engines {
project = name
}
}
s.mu.RUnlock()
}
if project == "" {
http.Error(w, "project is required (multiple projects configured)", http.StatusBadRequest)
return
}
sessionKey := req.SessionKey
if sessionKey == "" {
s.mu.RLock()
engine := s.engines[project]
s.mu.RUnlock()
if engine != nil {
keys := engine.ActiveSessionKeys()
if len(keys) == 1 {
sessionKey = keys[0]
}
}
}
if sessionKey == "" {
http.Error(w, "session_key is required", http.StatusBadRequest)
return
}
job := &TimerJob{
ID: GenerateTimerID(),
Project: project,
SessionKey: sessionKey,
ScheduledAt: fireAt,
Prompt: req.Prompt,
Exec: req.Exec,
WorkDir: req.WorkDir,
Description: req.Description,
Silent: req.Silent,
Mute: req.Mute,
SessionMode: req.SessionMode,
Mode: req.Mode,
TimeoutMins: req.TimeoutMins,
CreatedAt: time.Now(),
}
if err := s.timer.AddJob(job); err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
apiJSON(w, http.StatusOK, job)
}
func (s *APIServer) handleTimerList(w http.ResponseWriter, r *http.Request) {
if s.timer == nil {
http.Error(w, "timer scheduler not available", http.StatusServiceUnavailable)
return
}
project := r.URL.Query().Get("project")
var jobs []*TimerJob
if project != "" {
jobs = s.timer.Store().ListByProject(project)
} else {
jobs = s.timer.Store().List()
}
// Filter to pending only
var pending []*TimerJob
for _, j := range jobs {
if !j.Fired {
pending = append(pending, j)
}
}
apiJSON(w, http.StatusOK, pending)
}
func (s *APIServer) handleTimerInfo(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodGet {
http.Error(w, "GET only", http.StatusMethodNotAllowed)
return
}
if s.timer == nil {
http.Error(w, "timer scheduler not available", http.StatusServiceUnavailable)
return
}
id := r.URL.Query().Get("id")
if id == "" {
http.Error(w, "id is required", http.StatusBadRequest)
return
}
job := s.timer.Store().Get(id)
if job == nil {
http.Error(w, "timer not found", http.StatusNotFound)
return
}
apiJSON(w, http.StatusOK, job)
}
func (s *APIServer) handleTimerDel(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "POST only", http.StatusMethodNotAllowed)
return
}
if s.timer == nil {
http.Error(w, "timer scheduler not available", http.StatusServiceUnavailable)
return
}
var req struct {
ID string `json:"id"`
}
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
http.Error(w, "invalid JSON: "+err.Error(), http.StatusBadRequest)
return
}
if req.ID == "" {
http.Error(w, "id is required", http.StatusBadRequest)
return
}
if !s.timer.RemoveJob(req.ID) {
http.Error(w, "timer not found", http.StatusNotFound)
return
}
apiJSON(w, http.StatusOK, map[string]string{"status": "ok", "id": req.ID})
}
// ── Relay API ──────────────────────────────────────────────────
func (s *APIServer) handleRelaySend(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "POST only", http.StatusMethodNotAllowed)
return
}
if s.relay == nil {
http.Error(w, "relay not available", http.StatusServiceUnavailable)
return
}
var req RelayRequest
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
http.Error(w, "invalid JSON: "+err.Error(), http.StatusBadRequest)
return
}
if req.To == "" || req.Message == "" || req.SessionKey == "" {
http.Error(w, "to, session_key, and message are required", http.StatusBadRequest)
return
}
resp, err := s.relay.Send(r.Context(), req)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
apiJSON(w, http.StatusOK, resp)
}
func (s *APIServer) handleRelayBind(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "POST only", http.StatusMethodNotAllowed)
return
}
if s.relay == nil {
http.Error(w, "relay not available", http.StatusServiceUnavailable)
return
}
var req struct {
Platform string `json:"platform"`
ChatID string `json:"chat_id"`
Bots map[string]string `json:"bots"`
}
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
http.Error(w, "invalid JSON: "+err.Error(), http.StatusBadRequest)
return
}
if req.ChatID == "" || len(req.Bots) < 2 {
http.Error(w, "chat_id and at least 2 bots are required", http.StatusBadRequest)
return
}
s.relay.Bind(req.Platform, req.ChatID, req.Bots)
apiJSON(w, http.StatusOK, map[string]string{"status": "ok"})
}
func (s *APIServer) handleRelayBinding(w http.ResponseWriter, r *http.Request) {
if s.relay == nil {
http.Error(w, "relay not available", http.StatusServiceUnavailable)
return
}
chatID := r.URL.Query().Get("chat_id")
if chatID == "" {
http.Error(w, "chat_id is required", http.StatusBadRequest)
return
}
binding := s.relay.GetBinding(chatID)
if binding == nil {
http.Error(w, "no binding found", http.StatusNotFound)
return
}
apiJSON(w, http.StatusOK, binding)
}