mirror of
https://github.com/memohai/Memoh.git
synced 2026-04-27 07:16:19 +09:00
149 lines
4.4 KiB
Go
149 lines
4.4 KiB
Go
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"
|
|
}
|