package flow import ( "context" "encoding/json" "errors" "strings" "time" sdk "github.com/memohai/twilight-ai/sdk" agentpkg "github.com/memohai/memoh/internal/agent" "github.com/memohai/memoh/internal/conversation" "github.com/memohai/memoh/internal/heartbeat" "github.com/memohai/memoh/internal/schedule" ) // TriggerSchedule executes a scheduled command via the internal agent. func (r *Resolver) TriggerSchedule(ctx context.Context, botID string, payload schedule.TriggerPayload, token string) (schedule.TriggerResult, error) { if strings.TrimSpace(botID) == "" { return schedule.TriggerResult{}, errors.New("bot id is required") } if strings.TrimSpace(payload.Command) == "" { return schedule.TriggerResult{}, errors.New("schedule command is required") } req := conversation.ChatRequest{ BotID: botID, ChatID: botID, SessionID: payload.SessionID, Query: payload.Command, UserID: payload.OwnerUserID, Token: token, } rc, err := r.resolve(ctx, req) if err != nil { return schedule.TriggerResult{}, err } cfg := rc.runConfig cfg.SessionType = "schedule" cfg.Identity.ChannelIdentityID = strings.TrimSpace(payload.OwnerUserID) schedulePrompt := agentpkg.GenerateSchedulePrompt(agentpkg.Schedule{ ID: payload.ID, Name: payload.Name, Description: payload.Description, Pattern: payload.Pattern, MaxCalls: payload.MaxCalls, Command: payload.Command, }) cfg.Messages = append(cfg.Messages, sdk.UserMessage(schedulePrompt)) cfg = r.prepareRunConfig(ctx, cfg) result, err := r.agent.Generate(ctx, cfg) if err != nil { return schedule.TriggerResult{}, err } outputMessages := sdkMessagesToModelMessages(result.Messages) roundMessages := prependUserMessage(req.Query, outputMessages) storeErr := r.storeRound(ctx, req, roundMessages, rc.model.ID) totalUsageJSON, _ := json.Marshal(result.Usage) return schedule.TriggerResult{ Status: "ok", Text: strings.TrimSpace(result.Text), UsageBytes: totalUsageJSON, ModelID: rc.model.ID, }, storeErr } // TriggerHeartbeat executes a heartbeat check via the internal agent. func (r *Resolver) TriggerHeartbeat(ctx context.Context, botID string, payload heartbeat.TriggerPayload, token string) (heartbeat.TriggerResult, error) { if strings.TrimSpace(botID) == "" { return heartbeat.TriggerResult{}, errors.New("bot id is required") } var heartbeatModel string if botSettings, err := r.loadBotSettings(ctx, botID); err == nil { heartbeatModel = strings.TrimSpace(botSettings.HeartbeatModelID) } req := conversation.ChatRequest{ BotID: botID, ChatID: botID, SessionID: payload.SessionID, Query: "heartbeat", UserID: payload.OwnerUserID, Token: token, Model: heartbeatModel, } rc, err := r.resolve(ctx, req) if err != nil { return heartbeat.TriggerResult{}, err } cfg := rc.runConfig cfg.SessionType = "heartbeat" cfg.Identity.ChannelIdentityID = strings.TrimSpace(payload.OwnerUserID) var checklist string if r.agent != nil { nowFn := time.Now if cfg.Identity.TimezoneLocation != nil { nowFn = func() time.Time { return time.Now().In(cfg.Identity.TimezoneLocation) } } fs := agentpkg.NewFSClient(r.agent.BridgeProvider(), botID, nowFn) checklist = fs.ReadTextSafe(ctx, "/data/HEARTBEAT.md") } now := time.Now().UTC() if cfg.Identity.TimezoneLocation != nil { now = now.In(cfg.Identity.TimezoneLocation) } heartbeatPrompt := agentpkg.GenerateHeartbeatPrompt(payload.Interval, checklist, now, payload.LastHeartbeatAt) cfg.Messages = append(cfg.Messages, sdk.UserMessage(heartbeatPrompt)) cfg = r.prepareRunConfig(ctx, cfg) result, err := r.agent.Generate(ctx, cfg) if err != nil { return heartbeat.TriggerResult{}, err } status := "alert" text := strings.TrimSpace(result.Text) if isHeartbeatOK(text) { status = "ok" } outputMessages := sdkMessagesToModelMessages(result.Messages) roundMessages := prependUserMessage(heartbeatPrompt, outputMessages) _ = r.storeRound(ctx, req, roundMessages, rc.model.ID) totalUsageJSON, _ := json.Marshal(result.Usage) return heartbeat.TriggerResult{ Status: status, Text: text, Usage: totalUsageJSON, UsageBytes: totalUsageJSON, ModelID: rc.model.ID, SessionID: payload.SessionID, }, nil } func isHeartbeatOK(text string) bool { t := strings.TrimSpace(text) return strings.HasPrefix(t, "HEARTBEAT_OK") || strings.HasSuffix(t, "HEARTBEAT_OK") || t == "HEARTBEAT_OK" }