mirror of
https://github.com/tiennm99/goclaw.git
synced 2026-06-10 06:10:46 +00:00
29816db0ab
- CronPayload.WakeHeartbeat triggers heartbeat immediately after cron job completes - Cron tool supports wake_heartbeat param on add/update actions - Scheduler.HasActiveSessionsForAgent() detects busy agents for heartbeat skip - RunRequest.LightContext skips loading context files during heartbeat runs
126 lines
4.4 KiB
Go
126 lines
4.4 KiB
Go
package cmd
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/nextlevelbuilder/goclaw/internal/agent"
|
|
"github.com/nextlevelbuilder/goclaw/internal/bus"
|
|
"github.com/nextlevelbuilder/goclaw/internal/channels"
|
|
"github.com/nextlevelbuilder/goclaw/internal/config"
|
|
"github.com/nextlevelbuilder/goclaw/internal/scheduler"
|
|
"github.com/nextlevelbuilder/goclaw/internal/sessions"
|
|
"github.com/nextlevelbuilder/goclaw/internal/store"
|
|
)
|
|
|
|
// makeCronJobHandler creates a cron job handler that routes through the scheduler's cron lane.
|
|
// This ensures per-session concurrency control (same job can't run concurrently)
|
|
// and integration with /stop, /stopall commands.
|
|
// cronHeartbeatWakeFn holds the heartbeat wake function, set after ticker creation.
|
|
// Safe because cron jobs only fire after Start(), well after this is set.
|
|
var cronHeartbeatWakeFn func(agentID string)
|
|
|
|
func makeCronJobHandler(sched *scheduler.Scheduler, msgBus *bus.MessageBus, cfg *config.Config, channelMgr *channels.Manager) func(job *store.CronJob) (*store.CronJobResult, error) {
|
|
return func(job *store.CronJob) (*store.CronJobResult, error) {
|
|
agentID := job.AgentID
|
|
if agentID == "" {
|
|
agentID = cfg.ResolveDefaultAgentID()
|
|
} else {
|
|
agentID = config.NormalizeAgentID(agentID)
|
|
}
|
|
|
|
sessionKey := sessions.BuildCronSessionKey(agentID, job.ID)
|
|
channel := job.Payload.Channel
|
|
if channel == "" {
|
|
channel = "cron"
|
|
}
|
|
|
|
// Infer peer kind from the stored session metadata (group chats need it
|
|
// so that tools like message can route correctly via group APIs).
|
|
peerKind := resolveCronPeerKind(job)
|
|
|
|
// Resolve channel type for system prompt context.
|
|
channelType := resolveChannelType(channelMgr, channel)
|
|
|
|
// Build cron context so the agent knows delivery target and requester.
|
|
var extraPrompt string
|
|
if job.Payload.Deliver && job.Payload.Channel != "" && job.Payload.To != "" {
|
|
extraPrompt = fmt.Sprintf(
|
|
"[Cron Job]\nThis is scheduled job \"%s\" (ID: %s).\n"+
|
|
"Requester: user %s on channel \"%s\" (chat %s).\n"+
|
|
"Your response will be automatically delivered to that chat — just produce the content directly.",
|
|
job.Name, job.ID, job.UserID, job.Payload.Channel, job.Payload.To,
|
|
)
|
|
} else {
|
|
extraPrompt = fmt.Sprintf(
|
|
"[Cron Job]\nThis is scheduled job \"%s\" (ID: %s), created by user %s.\n"+
|
|
"Delivery is not configured — respond normally.",
|
|
job.Name, job.ID, job.UserID,
|
|
)
|
|
}
|
|
|
|
// Schedule through cron lane — scheduler handles agent resolution and concurrency
|
|
outCh := sched.Schedule(context.Background(), scheduler.LaneCron, agent.RunRequest{
|
|
SessionKey: sessionKey,
|
|
Message: job.Payload.Message,
|
|
Channel: channel,
|
|
ChannelType: channelType,
|
|
ChatID: job.Payload.To,
|
|
PeerKind: peerKind,
|
|
UserID: job.UserID,
|
|
RunID: fmt.Sprintf("cron:%s", job.ID),
|
|
Stream: false,
|
|
ExtraSystemPrompt: extraPrompt,
|
|
TraceName: fmt.Sprintf("Cron [%s] - %s", job.Name, agentID),
|
|
TraceTags: []string{"cron"},
|
|
})
|
|
|
|
// Block until the scheduled run completes
|
|
outcome := <-outCh
|
|
if outcome.Err != nil {
|
|
return nil, outcome.Err
|
|
}
|
|
|
|
result := outcome.Result
|
|
|
|
// If job wants delivery to a channel, send the agent response to the target chat.
|
|
if job.Payload.Deliver && job.Payload.Channel != "" && job.Payload.To != "" {
|
|
outMsg := bus.OutboundMessage{
|
|
Channel: job.Payload.Channel,
|
|
ChatID: job.Payload.To,
|
|
Content: result.Content,
|
|
}
|
|
if peerKind == "group" {
|
|
outMsg.Metadata = map[string]string{"group_id": job.Payload.To}
|
|
}
|
|
appendMediaToOutbound(&outMsg, result.Media)
|
|
msgBus.PublishOutbound(outMsg)
|
|
}
|
|
|
|
cronResult := &store.CronJobResult{
|
|
Content: result.Content,
|
|
}
|
|
if result.Usage != nil {
|
|
cronResult.InputTokens = result.Usage.PromptTokens
|
|
cronResult.OutputTokens = result.Usage.CompletionTokens
|
|
}
|
|
|
|
// wakeMode: trigger heartbeat after cron job completes
|
|
if job.Payload.WakeHeartbeat && cronHeartbeatWakeFn != nil {
|
|
cronHeartbeatWakeFn(agentID)
|
|
}
|
|
|
|
return cronResult, nil
|
|
}
|
|
}
|
|
|
|
// resolveCronPeerKind infers peer kind from the cron job's user ID.
|
|
// Group cron jobs have userID prefixed with "group:" or "guild:" (set during job creation).
|
|
func resolveCronPeerKind(job *store.CronJob) string {
|
|
if strings.HasPrefix(job.UserID, "group:") || strings.HasPrefix(job.UserID, "guild:") {
|
|
return "group"
|
|
}
|
|
return ""
|
|
}
|