173 lines
5.6 KiB
TypeScript
173 lines
5.6 KiB
TypeScript
|
|
import { ReviewDecision } from "../../utils/agent/review";
|
|||
|
|
// TODO: figure out why `cli-spinners` fails on Node v20.9.0
|
|||
|
|
// which is why we have to do this in the first place
|
|||
|
|
//
|
|||
|
|
// @ts-expect-error select.js is JavaScript and has no types
|
|||
|
|
import { Select } from "../vendor/ink-select/select";
|
|||
|
|
import TextInput from "../vendor/ink-text-input";
|
|||
|
|
import { Box, Text, useInput } from "ink";
|
|||
|
|
import React from "react";
|
|||
|
|
|
|||
|
|
// default deny‑reason:
|
|||
|
|
const DEFAULT_DENY_MESSAGE =
|
|||
|
|
"Don't do that, but keep trying to fix the problem";
|
|||
|
|
|
|||
|
|
export function TerminalChatCommandReview({
|
|||
|
|
confirmationPrompt,
|
|||
|
|
onReviewCommand,
|
|||
|
|
}: {
|
|||
|
|
confirmationPrompt: React.ReactNode;
|
|||
|
|
onReviewCommand: (decision: ReviewDecision, customMessage?: string) => void;
|
|||
|
|
}): React.ReactElement {
|
|||
|
|
const [mode, setMode] = React.useState<"select" | "input">("select");
|
|||
|
|
const [msg, setMsg] = React.useState<string>("");
|
|||
|
|
|
|||
|
|
// -------------------------------------------------------------------------
|
|||
|
|
// Determine whether the "always approve" option should be displayed. We
|
|||
|
|
// only hide it for the special `apply_patch` command since approving those
|
|||
|
|
// permanently would bypass the user's review of future file modifications.
|
|||
|
|
// The information is embedded in the `confirmationPrompt` React element –
|
|||
|
|
// we inspect the `commandForDisplay` prop exposed by
|
|||
|
|
// <TerminalChatToolCallCommand/> to extract the base command.
|
|||
|
|
// -------------------------------------------------------------------------
|
|||
|
|
|
|||
|
|
const showAlwaysApprove = React.useMemo(() => {
|
|||
|
|
if (
|
|||
|
|
React.isValidElement(confirmationPrompt) &&
|
|||
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|||
|
|
typeof (confirmationPrompt as any).props?.commandForDisplay === "string"
|
|||
|
|
) {
|
|||
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|||
|
|
const command: string = (confirmationPrompt as any).props
|
|||
|
|
.commandForDisplay;
|
|||
|
|
// Grab the first token of the first line – that corresponds to the base
|
|||
|
|
// command even when the string contains embedded newlines (e.g. diffs).
|
|||
|
|
const baseCmd = command.split("\n")[0]?.trim().split(/\s+/)[0] ?? "";
|
|||
|
|
return baseCmd !== "apply_patch";
|
|||
|
|
}
|
|||
|
|
// Default to showing the option when we cannot reliably detect the base
|
|||
|
|
// command.
|
|||
|
|
return true;
|
|||
|
|
}, [confirmationPrompt]);
|
|||
|
|
|
|||
|
|
// Memoize the list of selectable options to avoid recreating the array on
|
|||
|
|
// every render. This keeps <Select/> stable and prevents unnecessary work
|
|||
|
|
// inside Ink.
|
|||
|
|
const approvalOptions = React.useMemo(() => {
|
|||
|
|
const opts: Array<
|
|||
|
|
| { label: string; value: ReviewDecision }
|
|||
|
|
| { label: string; value: "edit" }
|
|||
|
|
> = [
|
|||
|
|
{
|
|||
|
|
label: "Yes (y)",
|
|||
|
|
value: ReviewDecision.YES,
|
|||
|
|
},
|
|||
|
|
];
|
|||
|
|
|
|||
|
|
if (showAlwaysApprove) {
|
|||
|
|
opts.push({
|
|||
|
|
label: "Yes, always approve this exact command for this session (a)",
|
|||
|
|
value: ReviewDecision.ALWAYS,
|
|||
|
|
});
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
opts.push(
|
|||
|
|
{
|
|||
|
|
label: "Edit or give feedback (e)",
|
|||
|
|
value: "edit",
|
|||
|
|
},
|
|||
|
|
{
|
|||
|
|
label: "No, and keep going (n)",
|
|||
|
|
value: ReviewDecision.NO_CONTINUE,
|
|||
|
|
},
|
|||
|
|
{
|
|||
|
|
label: "No, and stop for now (esc)",
|
|||
|
|
value: ReviewDecision.NO_EXIT,
|
|||
|
|
},
|
|||
|
|
);
|
|||
|
|
|
|||
|
|
return opts;
|
|||
|
|
}, [showAlwaysApprove]);
|
|||
|
|
|
|||
|
|
useInput((input, key) => {
|
|||
|
|
if (mode === "select") {
|
|||
|
|
if (input === "y") {
|
|||
|
|
onReviewCommand(ReviewDecision.YES);
|
|||
|
|
} else if (input === "e") {
|
|||
|
|
setMode("input");
|
|||
|
|
} else if (input === "n") {
|
|||
|
|
onReviewCommand(
|
|||
|
|
ReviewDecision.NO_CONTINUE,
|
|||
|
|
"Don't do that, keep going though",
|
|||
|
|
);
|
|||
|
|
} else if (input === "a" && showAlwaysApprove) {
|
|||
|
|
onReviewCommand(ReviewDecision.ALWAYS);
|
|||
|
|
} else if (key.escape) {
|
|||
|
|
onReviewCommand(ReviewDecision.NO_EXIT);
|
|||
|
|
}
|
|||
|
|
} else {
|
|||
|
|
// text entry mode
|
|||
|
|
if (key.return) {
|
|||
|
|
// if user hit enter on empty msg, fall back to DEFAULT_DENY_MESSAGE
|
|||
|
|
const custom = msg.trim() === "" ? DEFAULT_DENY_MESSAGE : msg;
|
|||
|
|
onReviewCommand(ReviewDecision.NO_CONTINUE, custom);
|
|||
|
|
} else if (key.escape) {
|
|||
|
|
// treat escape as denial with default message as well
|
|||
|
|
onReviewCommand(
|
|||
|
|
ReviewDecision.NO_CONTINUE,
|
|||
|
|
msg.trim() === "" ? DEFAULT_DENY_MESSAGE : msg,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
});
|
|||
|
|
|
|||
|
|
return (
|
|||
|
|
<Box flexDirection="column" gap={1} borderStyle="round" marginTop={1}>
|
|||
|
|
{confirmationPrompt}
|
|||
|
|
<Box flexDirection="column" gap={1}>
|
|||
|
|
{mode === "select" ? (
|
|||
|
|
<>
|
|||
|
|
<Text>Allow command?</Text>
|
|||
|
|
<Box paddingX={2} flexDirection="column" gap={1}>
|
|||
|
|
<Select
|
|||
|
|
onChange={(value: ReviewDecision | "edit") => {
|
|||
|
|
if (value === "edit") {
|
|||
|
|
setMode("input");
|
|||
|
|
} else {
|
|||
|
|
onReviewCommand(value);
|
|||
|
|
}
|
|||
|
|
}}
|
|||
|
|
options={approvalOptions}
|
|||
|
|
/>
|
|||
|
|
</Box>
|
|||
|
|
</>
|
|||
|
|
) : (
|
|||
|
|
<>
|
|||
|
|
<Text>Give the model feedback (↵ to submit):</Text>
|
|||
|
|
<Box borderStyle="round">
|
|||
|
|
<Box paddingX={1}>
|
|||
|
|
<TextInput
|
|||
|
|
value={msg}
|
|||
|
|
onChange={setMsg}
|
|||
|
|
placeholder="type a reason"
|
|||
|
|
showCursor
|
|||
|
|
focus
|
|||
|
|
/>
|
|||
|
|
</Box>
|
|||
|
|
</Box>
|
|||
|
|
|
|||
|
|
{msg.trim() === "" && (
|
|||
|
|
<Box paddingX={2} marginBottom={1}>
|
|||
|
|
<Text dimColor>
|
|||
|
|
default:
|
|||
|
|
<Text>{DEFAULT_DENY_MESSAGE}</Text>
|
|||
|
|
</Text>
|
|||
|
|
</Box>
|
|||
|
|
)}
|
|||
|
|
</>
|
|||
|
|
)}
|
|||
|
|
</Box>
|
|||
|
|
</Box>
|
|||
|
|
);
|
|||
|
|
}
|