mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-04 00:34:02 +00:00
Adds the opt-in bundled GitHub Copilot agent runtime, pinned SDK install path, docs/inventory, SDK/tool/sandbox/auth wiring, and replay/tool-safety fixes.
Verification:
- Local: git diff --check; fnm exec --using 24.15.0 pnpm tsgo:extensions; fnm exec --using 24.15.0 pnpm check:test-types; fnm exec --using 24.15.0 pnpm build.
- Autoreview local: clean for the replay-safety fix; branch autoreview engine returned empty output twice, so local autoreview plus local/Crabbox/CI proof was used.
- Crabbox focused Copilot: run_2c0db9f48a4a, 19 files / 485 tests passed.
- Crabbox additional boundary shard: run_26a246a1aa24, prompt snapshots and plugin SDK boundary/export checks passed.
- Crabbox live Copilot: run_d128e4048b4e, real gpt-4.1 turn with live_echo phase-1-green and clean session-file check.
- GitHub checks: green on head 7cc8657e0d, including Dependency Guard after exact-head approval.
Co-authored-by: Ramraj Balasubramanian <ramrajba@microsoft.com>
221 lines
7.8 KiB
TypeScript
Executable File
221 lines
7.8 KiB
TypeScript
Executable File
/**
|
|
* Mirrors the AgentMessages produced by the copilot agent runtime into the
|
|
* OpenClaw audit transcript that sits next to (but is distinct from) the
|
|
* SDK's own session storage.
|
|
*
|
|
* The OpenClaw shell (src/agents/command/attempt-execution.ts) already
|
|
* writes the user prompt and the terminal assistant text into the
|
|
* transcript at the end of each attempt. That is the bare minimum to
|
|
* keep `/history` working. It does NOT capture tool calls, tool
|
|
* results, or intermediate assistant turns — those live only in the
|
|
* SDK's own session file.
|
|
*
|
|
* For audit/compliance and for the codex-parity guarantees we promised
|
|
* in the proposal, we mirror the full `messagesSnapshot` (user +
|
|
* assistant + toolResult) into the OpenClaw transcript via the same
|
|
* plugin-sdk primitives that the codex extension uses
|
|
* (extensions/codex/src/app-server/transcript-mirror.ts). Both writers
|
|
* cooperate via idempotency-key dedupe: each mirrored entry carries a
|
|
* stable `${idempotencyScope}:${identity}` key, and we skip any key
|
|
* already present in the transcript on disk before appending. Both
|
|
* attempt-execution's untagged entries (no idempotencyKey) and our
|
|
* tagged mirror entries can coexist; attempt-execution dedupes its own
|
|
* final-assistant append via `embeddedAssistantGapFill` content match.
|
|
*
|
|
* Failures (lock contention, fs errors, etc.) are swallowed by the
|
|
* caller-side `dualWriteCopilotTranscriptBestEffort` wrapper used
|
|
* in attempt.ts so they cannot break the attempt; this module itself
|
|
* throws on infrastructure failure so callers can choose policy.
|
|
*/
|
|
|
|
import { createHash } from "node:crypto";
|
|
import fs from "node:fs/promises";
|
|
import {
|
|
acquireSessionWriteLock,
|
|
appendSessionTranscriptMessage,
|
|
emitSessionTranscriptUpdate,
|
|
resolveSessionWriteLockAcquireTimeoutMs,
|
|
runAgentHarnessBeforeMessageWriteHook,
|
|
type AgentMessage,
|
|
type SessionWriteLockAcquireTimeoutConfig,
|
|
} from "openclaw/plugin-sdk/agent-harness-runtime";
|
|
|
|
type MirroredAgentMessage = Extract<AgentMessage, { role: "user" | "assistant" | "toolResult" }>;
|
|
|
|
const MIRROR_IDENTITY_META_KEY = "mirrorIdentity" as const;
|
|
|
|
/**
|
|
* Tag a message with a stable logical identity for mirror dedupe.
|
|
* Callers should use a value that is invariant for the same logical
|
|
* message across re-emits (e.g. `${sdkSessionId}:assistant:${turnIndex}`)
|
|
* but distinct for genuinely-distinct messages. When present this
|
|
* identity replaces the role/content fingerprint in the idempotency
|
|
* key, so the dedupe survives caller-scope rotation without collapsing
|
|
* distinct same-content turns. Symmetric to
|
|
* `attachCodexMirrorIdentity` in the codex extension.
|
|
*/
|
|
export function attachCopilotMirrorIdentity<T extends AgentMessage>(
|
|
message: T,
|
|
identity: string,
|
|
): T {
|
|
const record = message as unknown as Record<string, unknown>;
|
|
const existing = record["__openclaw"];
|
|
const baseMeta =
|
|
existing && typeof existing === "object" && !Array.isArray(existing)
|
|
? (existing as Record<string, unknown>)
|
|
: {};
|
|
return {
|
|
...record,
|
|
__openclaw: { ...baseMeta, [MIRROR_IDENTITY_META_KEY]: identity },
|
|
} as unknown as T;
|
|
}
|
|
|
|
function readMirrorIdentity(message: MirroredAgentMessage): string | undefined {
|
|
const record = message as unknown as { __openclaw?: unknown };
|
|
const meta = record["__openclaw"];
|
|
if (!meta || typeof meta !== "object" || Array.isArray(meta)) {
|
|
return undefined;
|
|
}
|
|
const id = (meta as Record<string, unknown>)[MIRROR_IDENTITY_META_KEY];
|
|
return typeof id === "string" && id.length > 0 ? id : undefined;
|
|
}
|
|
|
|
function fingerprintMirrorMessageContent(message: MirroredAgentMessage): string {
|
|
const payload = JSON.stringify({ role: message.role, content: message.content });
|
|
return createHash("sha256").update(payload).digest("hex").slice(0, 16);
|
|
}
|
|
|
|
function buildMirrorDedupeIdentity(message: MirroredAgentMessage): string {
|
|
const explicit = readMirrorIdentity(message);
|
|
if (explicit) {
|
|
return explicit;
|
|
}
|
|
return `${message.role}:${fingerprintMirrorMessageContent(message)}`;
|
|
}
|
|
|
|
export interface MirrorCopilotTranscriptParams {
|
|
sessionFile: string;
|
|
sessionKey?: string;
|
|
agentId?: string;
|
|
messages: AgentMessage[];
|
|
/**
|
|
* Stable per-harness/per-thread scope. The codex equivalent uses
|
|
* `codex-app-server:${threadId}`; we use `copilot:${sessionId}`
|
|
* by convention (see attempt.ts call site). Keeping the scope
|
|
* thread-stable (not per-turn) is what lets a re-emitted prior-turn
|
|
* entry collide with its existing on-disk key and be a true no-op.
|
|
*/
|
|
idempotencyScope?: string;
|
|
config?: SessionWriteLockAcquireTimeoutConfig;
|
|
}
|
|
|
|
export async function mirrorCopilotTranscript(
|
|
params: MirrorCopilotTranscriptParams,
|
|
): Promise<void> {
|
|
const messages = params.messages.filter(
|
|
(message): message is MirroredAgentMessage =>
|
|
message.role === "user" || message.role === "assistant" || message.role === "toolResult",
|
|
);
|
|
if (messages.length === 0) {
|
|
return;
|
|
}
|
|
|
|
const lock = await acquireSessionWriteLock({
|
|
sessionFile: params.sessionFile,
|
|
timeoutMs: resolveSessionWriteLockAcquireTimeoutMs(params.config),
|
|
});
|
|
try {
|
|
const existingIdempotencyKeys = await readTranscriptIdempotencyKeys(params.sessionFile);
|
|
for (const message of messages) {
|
|
const dedupeIdentity = buildMirrorDedupeIdentity(message);
|
|
const idempotencyKey = params.idempotencyScope
|
|
? `${params.idempotencyScope}:${dedupeIdentity}`
|
|
: undefined;
|
|
if (idempotencyKey && existingIdempotencyKeys.has(idempotencyKey)) {
|
|
continue;
|
|
}
|
|
const transcriptMessage = {
|
|
...message,
|
|
...(idempotencyKey ? { idempotencyKey } : {}),
|
|
} as AgentMessage;
|
|
const nextMessage = runAgentHarnessBeforeMessageWriteHook({
|
|
message: transcriptMessage,
|
|
agentId: params.agentId,
|
|
sessionKey: params.sessionKey,
|
|
});
|
|
if (!nextMessage) {
|
|
continue;
|
|
}
|
|
const messageToAppend = (
|
|
idempotencyKey
|
|
? {
|
|
...(nextMessage as unknown as Record<string, unknown>),
|
|
idempotencyKey,
|
|
}
|
|
: nextMessage
|
|
) as AgentMessage;
|
|
await appendSessionTranscriptMessage({
|
|
transcriptPath: params.sessionFile,
|
|
message: messageToAppend,
|
|
config: params.config,
|
|
});
|
|
if (idempotencyKey) {
|
|
existingIdempotencyKeys.add(idempotencyKey);
|
|
}
|
|
}
|
|
} finally {
|
|
await lock.release();
|
|
}
|
|
|
|
if (params.sessionKey) {
|
|
emitSessionTranscriptUpdate({ sessionFile: params.sessionFile, sessionKey: params.sessionKey });
|
|
} else {
|
|
emitSessionTranscriptUpdate(params.sessionFile);
|
|
}
|
|
}
|
|
|
|
async function readTranscriptIdempotencyKeys(sessionFile: string): Promise<Set<string>> {
|
|
const keys = new Set<string>();
|
|
let raw: string;
|
|
try {
|
|
raw = await fs.readFile(sessionFile, "utf8");
|
|
} catch (error) {
|
|
if ((error as NodeJS.ErrnoException).code !== "ENOENT") {
|
|
throw error;
|
|
}
|
|
return keys;
|
|
}
|
|
for (const line of raw.split(/\r?\n/)) {
|
|
if (!line.trim()) {
|
|
continue;
|
|
}
|
|
try {
|
|
const parsed = JSON.parse(line) as { message?: { idempotencyKey?: unknown } };
|
|
if (typeof parsed.message?.idempotencyKey === "string") {
|
|
keys.add(parsed.message.idempotencyKey);
|
|
}
|
|
} catch {
|
|
continue;
|
|
}
|
|
}
|
|
return keys;
|
|
}
|
|
|
|
/**
|
|
* Caller-side wrapper that swallows mirror failures. attempt.ts uses
|
|
* this so that a transient transcript-mirror failure (lock contention,
|
|
* disk full, etc.) never breaks an otherwise-successful attempt. The
|
|
* SDK's own session file remains the source of truth in that case;
|
|
* the OpenClaw audit trail just misses the intermediate messages for
|
|
* this turn.
|
|
*/
|
|
export async function dualWriteCopilotTranscriptBestEffort(
|
|
params: MirrorCopilotTranscriptParams,
|
|
): Promise<void> {
|
|
try {
|
|
await mirrorCopilotTranscript(params);
|
|
} catch (error) {
|
|
console.warn("[copilot-attempt] dual-write transcript mirror failed", error);
|
|
}
|
|
}
|