mirror of
https://github.com/multica-ai/multica.git
synced 2026-07-05 13:29:44 +02:00
184 lines
9.9 KiB
Go
184 lines
9.9 KiB
Go
package execenv
|
|
|
|
import "fmt"
|
|
|
|
// BuildNewCommentsHint returns the comment-reading pointer for the WARM path —
|
|
// the agent ran on this issue before, so there is a since-anchor. The server
|
|
// count is ISSUE-WIDE (every thread, not just the triggering one) and excludes
|
|
// the triggering comment itself because that body is already injected into the
|
|
// prompt. It ships only the COUNT and the cursor — never the comment bodies —
|
|
// so the server stays cheap and the agent pulls details on demand.
|
|
//
|
|
// The agent is told the full issue-wide volume but steered to read the
|
|
// triggering (parent) thread FIRST instead of blindly catching up on every
|
|
// thread. The issue-wide `--since` catch-up is kept as an explicit
|
|
// "only if you need it" fallback.
|
|
//
|
|
// Both the per-turn prompt (daemon.buildCommentPrompt) and the CLAUDE.md
|
|
// workflow (InjectRuntimeConfig) call this so the two surfaces cannot drift
|
|
// (hard requirement from PR #2816).
|
|
//
|
|
// Renders nothing on cold start (no prior run → newCommentsSince empty) or when
|
|
// there are no new comments (newCommentCount <= 0) or issueID is empty. In those
|
|
// cases the caller falls back to BuildResumedCommentsHint (when a prior session
|
|
// is active) or BuildColdCommentsHint.
|
|
func BuildNewCommentsHint(issueID, triggerCommentID, triggerThreadID, newCommentsSince string, newCommentCount int) string {
|
|
if newCommentCount <= 0 || newCommentsSince == "" || issueID == "" {
|
|
return ""
|
|
}
|
|
threadID := activeThreadID(triggerThreadID, triggerCommentID)
|
|
// When we know the triggering thread, steer the agent to read THAT thread
|
|
// first rather than blindly pulling every new comment issue-wide. The
|
|
// issue-wide --since catch-up is demoted to an only-if-needed fallback.
|
|
if threadID != "" {
|
|
return fmt.Sprintf(
|
|
"%d new comment(s) on this issue since your last run — don't read them all blindly. "+
|
|
"Start with the thread your triggering comment is in: "+
|
|
"`multica issue comment list %s --thread %s --since %s --output json` "+
|
|
"(swap `--since` for `--tail 30` if you need the full thread, not just the delta). "+
|
|
"Only if you need context from the other threads, catch up issue-wide: "+
|
|
"`multica issue comment list %s --since %s --output json`.\n\n",
|
|
newCommentCount, issueID, threadID, newCommentsSince, issueID, newCommentsSince,
|
|
)
|
|
}
|
|
// Defensive: comment triggers always carry a trigger id, but if one is
|
|
// missing there is no thread to anchor on, so fall back to the plain
|
|
// issue-wide catch-up.
|
|
return fmt.Sprintf(
|
|
"%d new comment(s) on this issue since your last run. Catch up: "+
|
|
"`multica issue comment list %s --since %s --output json`.\n\n",
|
|
newCommentCount, issueID, newCommentsSince,
|
|
)
|
|
}
|
|
|
|
// BuildResumedCommentsHint returns the comment-reading pointer for the WARM
|
|
// no-delta path: the daemon is resuming a prior provider session and the
|
|
// triggering comment body has already been injected into the per-turn prompt.
|
|
// newCommentCount == 0 here means no new comments arrived issue-wide since the
|
|
// last run (beyond the injected trigger and the agent's own replies). Keep the
|
|
// read bounded and conditional, but make it explicit that context-dependent
|
|
// replies should refresh the triggering conversation rather than trusting
|
|
// resumed memory alone.
|
|
func BuildResumedCommentsHint(issueID, triggerCommentID, triggerThreadID string) string {
|
|
threadID := activeThreadID(triggerThreadID, triggerCommentID)
|
|
if issueID == "" || threadID == "" {
|
|
return ""
|
|
}
|
|
return fmt.Sprintf(
|
|
"You're resuming the prior session, and the triggering comment is already included above. "+
|
|
"No other new comments on this issue since your last run. "+
|
|
"Use the active thread anchor `%s` and triggering comment ID `%s`. "+
|
|
"If your reply depends on thread context, do not rely only on resumed session memory — "+
|
|
"first pull the triggering conversation with: "+
|
|
"`multica issue comment list %s --thread %s --tail 30 --output json`.\n\n",
|
|
threadID, triggerCommentID, issueID, threadID,
|
|
)
|
|
}
|
|
|
|
// BuildColdCommentsHint returns the comment-reading pointer for the COLD path —
|
|
// the agent has no prior run on this issue, so there is no since-anchor and
|
|
// BuildNewCommentsHint renders nothing. Instead of dumping the whole flat
|
|
// timeline (oldest-first, server cap 2000), point the agent at the triggering
|
|
// CONVERSATION: `--thread <trigger> --tail 30` returns that thread's root plus
|
|
// its 30 newest replies (root is always included, even at --tail 0) — the
|
|
// context the triggering comment actually needs. A `--recent 20` pointer is kept
|
|
// for cross-thread background the agent can pull on judgment.
|
|
//
|
|
// Both surfaces call this so the cold fallback cannot drift between them (same
|
|
// single-source rule as BuildNewCommentsHint, PR #2816). Returns "" when there
|
|
// is no triggering comment to thread from, so the caller can keep a final plain
|
|
// fallback.
|
|
func BuildColdCommentsHint(issueID, triggerCommentID, triggerThreadID string) string {
|
|
threadID := activeThreadID(triggerThreadID, triggerCommentID)
|
|
if issueID == "" || threadID == "" {
|
|
return ""
|
|
}
|
|
return fmt.Sprintf(
|
|
"Read the triggering conversation first: "+
|
|
"`multica issue comment list %s --thread %s --tail 30 --output json` "+
|
|
"(that thread's root + its 30 newest replies). "+
|
|
"Need cross-thread background? `multica issue comment list %s --recent 20 --output json`.\n\n",
|
|
issueID, threadID, issueID,
|
|
)
|
|
}
|
|
|
|
func activeThreadID(triggerThreadID, triggerCommentID string) string {
|
|
if triggerThreadID != "" {
|
|
return triggerThreadID
|
|
}
|
|
return triggerCommentID
|
|
}
|
|
|
|
// BuildCommentReplyInstructions returns the canonical block telling an agent
|
|
// how to post its reply for a comment-triggered task. Both the per-turn
|
|
// prompt (daemon.buildCommentPrompt) and the CLAUDE.md workflow
|
|
// (InjectRuntimeConfig) call this so the trigger comment ID and the
|
|
// --parent value cannot drift between surfaces.
|
|
//
|
|
// The explicit "do not reuse --parent from previous turns" wording exists
|
|
// because resumed Claude sessions keep prior turns' tool calls in context
|
|
// and will otherwise copy the old --parent UUID forward.
|
|
//
|
|
// The template is platform-aware but provider-agnostic — the failure it
|
|
// guards against lives at the shell layer, so it cannot be scoped to one
|
|
// provider (MUL-2904):
|
|
//
|
|
// - Windows + any provider → write a UTF-8 file, post with `--content-file`.
|
|
// This is the only path that survives Windows shells (PowerShell 5.1
|
|
// defaults to ASCIIEncoding when piping to native commands and drops
|
|
// non-ASCII as `?`; cmd.exe is at the mercy of `chcp`). The original
|
|
// reports — #2198 (Chinese), #2236 (Chinese), #2376 (Cyrillic, observed
|
|
// on a non-Codex agent) — all match this signature.
|
|
// - Linux/macOS + any provider → `--content-stdin` with a QUOTED HEREDOC
|
|
// (`<<'COMMENT'`). The quoted delimiter stops the shell from expanding
|
|
// backticks, `$()`, or `$VAR` inside the body. Inlining `--content "..."`
|
|
// instead lets the shell rewrite the body BEFORE the CLI receives it: a
|
|
// backtick-wrapped token becomes a failed command substitution that is
|
|
// silently deleted, the stored comment no longer matches what the model
|
|
// intended, and a model that notices the mismatch can retry forever
|
|
// (MUL-2904 / OKK-497). It also sidesteps Codex's habit of emitting
|
|
// literal `\n` escapes inside `--content` (MUL-1467).
|
|
//
|
|
// provider is retained for caller symmetry and future per-provider tweaks; the
|
|
// guardrail itself is intentionally identical across providers.
|
|
func BuildCommentReplyInstructions(provider, issueID, triggerCommentID string) string {
|
|
if triggerCommentID == "" {
|
|
return ""
|
|
}
|
|
if runtimeGOOS == "windows" {
|
|
return fmt.Sprintf(
|
|
"If you decide to reply, post it as a comment — always use the trigger comment ID below, "+
|
|
"do NOT reuse --parent values from previous turns in this session.\n\n"+
|
|
"On Windows, write the reply body to a UTF-8 file with your file-write tool, then post it with `--content-file`. "+
|
|
"Do NOT pipe via `--content-stdin` — Windows PowerShell 5.1's `$OutputEncoding` defaults to ASCIIEncoding when piping to native commands and silently drops non-ASCII (Chinese, Japanese, Cyrillic, accents, emoji) as `?` before the bytes reach `multica.exe`. "+
|
|
"Do NOT use inline `--content`; it is easy to lose formatting or accidentally compress a structured reply into one line.\n\n"+
|
|
"Use this form, preserving the same issue ID and --parent value:\n\n"+
|
|
" # 1. Write the reply body to a UTF-8 file (e.g. reply.md) with your file-write tool.\n"+
|
|
" # 2. Then run:\n"+
|
|
" multica issue comment add %s --parent %s --content-file ./reply.md\n\n"+
|
|
"Do NOT write literal `\\n` escapes to simulate line breaks; the file preserves real newlines.\n",
|
|
issueID, triggerCommentID,
|
|
)
|
|
}
|
|
// Linux/macOS, any provider: `--content-stdin` with a quoted HEREDOC. The
|
|
// quoted delimiter (`<<'COMMENT'`) is what makes this safe — it stops the
|
|
// shell from running backtick / `$()` substitution or `$VAR` expansion on
|
|
// the body. Inlining `--content "..."` is what triggered the MUL-2904
|
|
// duplicate-comment loop, so it is banned for every provider here, not just
|
|
// Codex.
|
|
return fmt.Sprintf(
|
|
"If you decide to reply, post it as a comment — always use the trigger comment ID below, "+
|
|
"do NOT reuse --parent values from previous turns in this session.\n\n"+
|
|
"Always use `--content-stdin` with a HEREDOC for agent-authored issue comments, even when the reply is a single line. "+
|
|
"Do NOT use inline `--content`; the shell rewrites unescaped backticks, `$()`, `$VAR`, or quotes in the body before the CLI receives them, and it is easy to lose formatting or compress a structured reply into one line.\n\n"+
|
|
"Use this form, preserving the same issue ID and --parent value:\n\n"+
|
|
" cat <<'COMMENT' | multica issue comment add %s --parent %s --content-stdin\n"+
|
|
" First paragraph.\n"+
|
|
"\n"+
|
|
" Second paragraph.\n"+
|
|
" COMMENT\n\n"+
|
|
"Do NOT write literal `\\n` escapes to simulate line breaks; the HEREDOC preserves real newlines.\n",
|
|
issueID, triggerCommentID,
|
|
)
|
|
}
|