main.rs 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739
  1. mod input;
  2. mod render;
  3. use std::env;
  4. use std::io::{self, Write};
  5. use std::path::{Path, PathBuf};
  6. use api::{
  7. AnthropicClient, ContentBlockDelta, InputContentBlock, InputMessage, MessageRequest,
  8. MessageResponse, OutputContentBlock, StreamEvent as ApiStreamEvent, ToolChoice, ToolDefinition,
  9. ToolResultContentBlock,
  10. };
  11. use commands::handle_slash_command;
  12. use compat_harness::{extract_manifest, UpstreamPaths};
  13. use render::{Spinner, TerminalRenderer};
  14. use runtime::{
  15. load_system_prompt, ApiClient, ApiRequest, AssistantEvent, CompactionConfig, ContentBlock,
  16. ConversationMessage, ConversationRuntime, MessageRole, PermissionMode, PermissionPolicy,
  17. RuntimeError, Session, TokenUsage, ToolError, ToolExecutor,
  18. };
  19. use tools::{execute_tool, mvp_tool_specs};
  20. const DEFAULT_MODEL: &str = "claude-sonnet-4-20250514";
  21. const DEFAULT_MAX_TOKENS: u32 = 32;
  22. const DEFAULT_DATE: &str = "2026-03-31";
  23. fn main() {
  24. if let Err(error) = run() {
  25. eprintln!("{error}");
  26. std::process::exit(1);
  27. }
  28. }
  29. fn run() -> Result<(), Box<dyn std::error::Error>> {
  30. let args: Vec<String> = env::args().skip(1).collect();
  31. match parse_args(&args)? {
  32. CliAction::DumpManifests => dump_manifests(),
  33. CliAction::BootstrapPlan => print_bootstrap_plan(),
  34. CliAction::PrintSystemPrompt { cwd, date } => print_system_prompt(cwd, date),
  35. CliAction::ResumeSession {
  36. session_path,
  37. command,
  38. } => resume_session(&session_path, command),
  39. CliAction::Prompt { prompt, model } => LiveCli::new(model, false)?.run_turn(&prompt)?,
  40. CliAction::Repl { model } => run_repl(model)?,
  41. CliAction::Help => print_help(),
  42. }
  43. Ok(())
  44. }
  45. #[derive(Debug, Clone, PartialEq, Eq)]
  46. enum CliAction {
  47. DumpManifests,
  48. BootstrapPlan,
  49. PrintSystemPrompt {
  50. cwd: PathBuf,
  51. date: String,
  52. },
  53. ResumeSession {
  54. session_path: PathBuf,
  55. command: Option<String>,
  56. },
  57. Prompt {
  58. prompt: String,
  59. model: String,
  60. },
  61. Repl {
  62. model: String,
  63. },
  64. Help,
  65. }
  66. fn parse_args(args: &[String]) -> Result<CliAction, String> {
  67. let mut model = DEFAULT_MODEL.to_string();
  68. let mut rest = Vec::new();
  69. let mut index = 0;
  70. while index < args.len() {
  71. match args[index].as_str() {
  72. "--model" => {
  73. let value = args
  74. .get(index + 1)
  75. .ok_or_else(|| "missing value for --model".to_string())?;
  76. model = value.clone();
  77. index += 2;
  78. }
  79. flag if flag.starts_with("--model=") => {
  80. model = flag[8..].to_string();
  81. index += 1;
  82. }
  83. other => {
  84. rest.push(other.to_string());
  85. index += 1;
  86. }
  87. }
  88. }
  89. if rest.is_empty() {
  90. return Ok(CliAction::Repl { model });
  91. }
  92. if matches!(rest.first().map(String::as_str), Some("--help" | "-h")) {
  93. return Ok(CliAction::Help);
  94. }
  95. if rest.first().map(String::as_str) == Some("--resume") {
  96. return parse_resume_args(&rest[1..]);
  97. }
  98. match rest[0].as_str() {
  99. "dump-manifests" => Ok(CliAction::DumpManifests),
  100. "bootstrap-plan" => Ok(CliAction::BootstrapPlan),
  101. "system-prompt" => parse_system_prompt_args(&rest[1..]),
  102. "prompt" => {
  103. let prompt = rest[1..].join(" ");
  104. if prompt.trim().is_empty() {
  105. return Err("prompt subcommand requires a prompt string".to_string());
  106. }
  107. Ok(CliAction::Prompt { prompt, model })
  108. }
  109. other => Err(format!("unknown subcommand: {other}")),
  110. }
  111. }
  112. fn parse_system_prompt_args(args: &[String]) -> Result<CliAction, String> {
  113. let mut cwd = env::current_dir().map_err(|error| error.to_string())?;
  114. let mut date = DEFAULT_DATE.to_string();
  115. let mut index = 0;
  116. while index < args.len() {
  117. match args[index].as_str() {
  118. "--cwd" => {
  119. let value = args
  120. .get(index + 1)
  121. .ok_or_else(|| "missing value for --cwd".to_string())?;
  122. cwd = PathBuf::from(value);
  123. index += 2;
  124. }
  125. "--date" => {
  126. let value = args
  127. .get(index + 1)
  128. .ok_or_else(|| "missing value for --date".to_string())?;
  129. date.clone_from(value);
  130. index += 2;
  131. }
  132. other => return Err(format!("unknown system-prompt option: {other}")),
  133. }
  134. }
  135. Ok(CliAction::PrintSystemPrompt { cwd, date })
  136. }
  137. fn parse_resume_args(args: &[String]) -> Result<CliAction, String> {
  138. let session_path = args
  139. .first()
  140. .ok_or_else(|| "missing session path for --resume".to_string())
  141. .map(PathBuf::from)?;
  142. let command = args.get(1).cloned();
  143. if args.len() > 2 {
  144. return Err("--resume accepts at most one trailing slash command".to_string());
  145. }
  146. Ok(CliAction::ResumeSession {
  147. session_path,
  148. command,
  149. })
  150. }
  151. fn dump_manifests() {
  152. let workspace_dir = PathBuf::from(env!("CARGO_MANIFEST_DIR")).join("../..");
  153. let paths = UpstreamPaths::from_workspace_dir(&workspace_dir);
  154. match extract_manifest(&paths) {
  155. Ok(manifest) => {
  156. println!("commands: {}", manifest.commands.entries().len());
  157. println!("tools: {}", manifest.tools.entries().len());
  158. println!("bootstrap phases: {}", manifest.bootstrap.phases().len());
  159. }
  160. Err(error) => {
  161. eprintln!("failed to extract manifests: {error}");
  162. std::process::exit(1);
  163. }
  164. }
  165. }
  166. fn print_bootstrap_plan() {
  167. for phase in runtime::BootstrapPlan::claude_code_default().phases() {
  168. println!("- {phase:?}");
  169. }
  170. }
  171. fn print_system_prompt(cwd: PathBuf, date: String) {
  172. match load_system_prompt(cwd, date, env::consts::OS, "unknown") {
  173. Ok(sections) => println!("{}", sections.join("\n\n")),
  174. Err(error) => {
  175. eprintln!("failed to build system prompt: {error}");
  176. std::process::exit(1);
  177. }
  178. }
  179. }
  180. fn resume_session(session_path: &Path, command: Option<String>) {
  181. let session = match Session::load_from_path(session_path) {
  182. Ok(session) => session,
  183. Err(error) => {
  184. eprintln!("failed to restore session: {error}");
  185. std::process::exit(1);
  186. }
  187. };
  188. match command {
  189. Some(command) if command.starts_with('/') => {
  190. let Some(result) = handle_slash_command(
  191. &command,
  192. &session,
  193. CompactionConfig {
  194. max_estimated_tokens: 0,
  195. ..CompactionConfig::default()
  196. },
  197. ) else {
  198. eprintln!("unknown slash command: {command}");
  199. std::process::exit(2);
  200. };
  201. if let Err(error) = result.session.save_to_path(session_path) {
  202. eprintln!("failed to persist resumed session: {error}");
  203. std::process::exit(1);
  204. }
  205. println!("{}", result.message);
  206. }
  207. Some(other) => {
  208. eprintln!("unsupported resumed command: {other}");
  209. std::process::exit(2);
  210. }
  211. None => {
  212. println!(
  213. "Restored session from {} ({} messages).",
  214. session_path.display(),
  215. session.messages.len()
  216. );
  217. }
  218. }
  219. }
  220. fn run_repl(model: String) -> Result<(), Box<dyn std::error::Error>> {
  221. let mut cli = LiveCli::new(model, true)?;
  222. let editor = input::LineEditor::new("› ");
  223. println!("Rusty Claude CLI interactive mode");
  224. println!("Type /help for commands. Shift+Enter or Ctrl+J inserts a newline.");
  225. while let Some(input) = editor.read_line()? {
  226. let trimmed = input.trim();
  227. if trimmed.is_empty() {
  228. continue;
  229. }
  230. match trimmed {
  231. "/exit" | "/quit" => break,
  232. "/help" => {
  233. println!("Available commands:");
  234. println!(" /help Show help");
  235. println!(" /status Show session status");
  236. println!(" /compact Compact session history");
  237. println!(" /exit Quit the REPL");
  238. }
  239. "/status" => cli.print_status(),
  240. "/compact" => cli.compact()?,
  241. _ => cli.run_turn(trimmed)?,
  242. }
  243. }
  244. Ok(())
  245. }
  246. struct LiveCli {
  247. model: String,
  248. system_prompt: Vec<String>,
  249. runtime: ConversationRuntime<AnthropicRuntimeClient, CliToolExecutor>,
  250. }
  251. impl LiveCli {
  252. fn new(model: String, enable_tools: bool) -> Result<Self, Box<dyn std::error::Error>> {
  253. let system_prompt = build_system_prompt()?;
  254. let runtime = build_runtime(
  255. Session::new(),
  256. model.clone(),
  257. system_prompt.clone(),
  258. enable_tools,
  259. )?;
  260. Ok(Self {
  261. model,
  262. system_prompt,
  263. runtime,
  264. })
  265. }
  266. fn run_turn(&mut self, input: &str) -> Result<(), Box<dyn std::error::Error>> {
  267. let mut spinner = Spinner::new();
  268. let mut stdout = io::stdout();
  269. spinner.tick(
  270. "Waiting for Claude",
  271. TerminalRenderer::new().color_theme(),
  272. &mut stdout,
  273. )?;
  274. let result = self.runtime.run_turn(input, None);
  275. match result {
  276. Ok(_) => {
  277. spinner.finish(
  278. "Claude response complete",
  279. TerminalRenderer::new().color_theme(),
  280. &mut stdout,
  281. )?;
  282. println!();
  283. Ok(())
  284. }
  285. Err(error) => {
  286. spinner.fail(
  287. "Claude request failed",
  288. TerminalRenderer::new().color_theme(),
  289. &mut stdout,
  290. )?;
  291. Err(Box::new(error))
  292. }
  293. }
  294. }
  295. fn print_status(&self) {
  296. let usage = self.runtime.usage().cumulative_usage();
  297. println!(
  298. "status: messages={} turns={} input_tokens={} output_tokens={}",
  299. self.runtime.session().messages.len(),
  300. self.runtime.usage().turns(),
  301. usage.input_tokens,
  302. usage.output_tokens
  303. );
  304. }
  305. fn compact(&mut self) -> Result<(), Box<dyn std::error::Error>> {
  306. let result = self.runtime.compact(CompactionConfig::default());
  307. let removed = result.removed_message_count;
  308. self.runtime = build_runtime(
  309. result.compacted_session,
  310. self.model.clone(),
  311. self.system_prompt.clone(),
  312. true,
  313. )?;
  314. println!("Compacted {removed} messages.");
  315. Ok(())
  316. }
  317. }
  318. fn build_system_prompt() -> Result<Vec<String>, Box<dyn std::error::Error>> {
  319. Ok(load_system_prompt(
  320. env::current_dir()?,
  321. DEFAULT_DATE,
  322. env::consts::OS,
  323. "unknown",
  324. )?)
  325. }
  326. fn build_runtime(
  327. session: Session,
  328. model: String,
  329. system_prompt: Vec<String>,
  330. enable_tools: bool,
  331. ) -> Result<ConversationRuntime<AnthropicRuntimeClient, CliToolExecutor>, Box<dyn std::error::Error>>
  332. {
  333. Ok(ConversationRuntime::new(
  334. session,
  335. AnthropicRuntimeClient::new(model, enable_tools)?,
  336. CliToolExecutor::new(),
  337. permission_policy_from_env(),
  338. system_prompt,
  339. ))
  340. }
  341. struct AnthropicRuntimeClient {
  342. runtime: tokio::runtime::Runtime,
  343. client: AnthropicClient,
  344. model: String,
  345. enable_tools: bool,
  346. }
  347. impl AnthropicRuntimeClient {
  348. fn new(model: String, enable_tools: bool) -> Result<Self, Box<dyn std::error::Error>> {
  349. Ok(Self {
  350. runtime: tokio::runtime::Runtime::new()?,
  351. client: AnthropicClient::from_env()?,
  352. model,
  353. enable_tools,
  354. })
  355. }
  356. }
  357. impl ApiClient for AnthropicRuntimeClient {
  358. fn stream(&mut self, request: ApiRequest) -> Result<Vec<AssistantEvent>, RuntimeError> {
  359. let message_request = MessageRequest {
  360. model: self.model.clone(),
  361. max_tokens: DEFAULT_MAX_TOKENS,
  362. messages: convert_messages(&request.messages),
  363. system: (!request.system_prompt.is_empty()).then(|| request.system_prompt.join("\n\n")),
  364. tools: self.enable_tools.then(|| {
  365. mvp_tool_specs()
  366. .into_iter()
  367. .map(|spec| ToolDefinition {
  368. name: spec.name.to_string(),
  369. description: Some(spec.description.to_string()),
  370. input_schema: spec.input_schema,
  371. })
  372. .collect()
  373. }),
  374. tool_choice: self.enable_tools.then_some(ToolChoice::Auto),
  375. stream: true,
  376. };
  377. self.runtime.block_on(async {
  378. let mut stream = self
  379. .client
  380. .stream_message(&message_request)
  381. .await
  382. .map_err(|error| RuntimeError::new(error.to_string()))?;
  383. let mut stdout = io::stdout();
  384. let mut events = Vec::new();
  385. let mut pending_tool: Option<(String, String, String)> = None;
  386. let mut saw_stop = false;
  387. while let Some(event) = stream
  388. .next_event()
  389. .await
  390. .map_err(|error| RuntimeError::new(error.to_string()))?
  391. {
  392. match event {
  393. ApiStreamEvent::MessageStart(start) => {
  394. for block in start.message.content {
  395. push_output_block(block, &mut stdout, &mut events, &mut pending_tool)?;
  396. }
  397. }
  398. ApiStreamEvent::ContentBlockStart(start) => {
  399. push_output_block(
  400. start.content_block,
  401. &mut stdout,
  402. &mut events,
  403. &mut pending_tool,
  404. )?;
  405. }
  406. ApiStreamEvent::ContentBlockDelta(delta) => match delta.delta {
  407. ContentBlockDelta::TextDelta { text } => {
  408. if !text.is_empty() {
  409. write!(stdout, "{text}")
  410. .and_then(|_| stdout.flush())
  411. .map_err(|error| RuntimeError::new(error.to_string()))?;
  412. events.push(AssistantEvent::TextDelta(text));
  413. }
  414. }
  415. ContentBlockDelta::InputJsonDelta { partial_json } => {
  416. if let Some((_, _, input)) = &mut pending_tool {
  417. input.push_str(&partial_json);
  418. }
  419. }
  420. },
  421. ApiStreamEvent::ContentBlockStop(_) => {
  422. if let Some((id, name, input)) = pending_tool.take() {
  423. events.push(AssistantEvent::ToolUse { id, name, input });
  424. }
  425. }
  426. ApiStreamEvent::MessageDelta(delta) => {
  427. events.push(AssistantEvent::Usage(TokenUsage {
  428. input_tokens: delta.usage.input_tokens,
  429. output_tokens: delta.usage.output_tokens,
  430. cache_creation_input_tokens: 0,
  431. cache_read_input_tokens: 0,
  432. }));
  433. }
  434. ApiStreamEvent::MessageStop(_) => {
  435. saw_stop = true;
  436. events.push(AssistantEvent::MessageStop);
  437. }
  438. }
  439. }
  440. if !saw_stop
  441. && events.iter().any(|event| {
  442. matches!(event, AssistantEvent::TextDelta(text) if !text.is_empty())
  443. || matches!(event, AssistantEvent::ToolUse { .. })
  444. })
  445. {
  446. events.push(AssistantEvent::MessageStop);
  447. }
  448. if events
  449. .iter()
  450. .any(|event| matches!(event, AssistantEvent::MessageStop))
  451. {
  452. return Ok(events);
  453. }
  454. let response = self
  455. .client
  456. .send_message(&MessageRequest {
  457. stream: false,
  458. ..message_request.clone()
  459. })
  460. .await
  461. .map_err(|error| RuntimeError::new(error.to_string()))?;
  462. response_to_events(response, &mut stdout)
  463. })
  464. }
  465. }
  466. fn push_output_block(
  467. block: OutputContentBlock,
  468. out: &mut impl Write,
  469. events: &mut Vec<AssistantEvent>,
  470. pending_tool: &mut Option<(String, String, String)>,
  471. ) -> Result<(), RuntimeError> {
  472. match block {
  473. OutputContentBlock::Text { text } => {
  474. if !text.is_empty() {
  475. write!(out, "{text}")
  476. .and_then(|_| out.flush())
  477. .map_err(|error| RuntimeError::new(error.to_string()))?;
  478. events.push(AssistantEvent::TextDelta(text));
  479. }
  480. }
  481. OutputContentBlock::ToolUse { id, name, input } => {
  482. *pending_tool = Some((id, name, input.to_string()));
  483. }
  484. }
  485. Ok(())
  486. }
  487. fn response_to_events(
  488. response: MessageResponse,
  489. out: &mut impl Write,
  490. ) -> Result<Vec<AssistantEvent>, RuntimeError> {
  491. let mut events = Vec::new();
  492. let mut pending_tool = None;
  493. for block in response.content {
  494. push_output_block(block, out, &mut events, &mut pending_tool)?;
  495. if let Some((id, name, input)) = pending_tool.take() {
  496. events.push(AssistantEvent::ToolUse { id, name, input });
  497. }
  498. }
  499. events.push(AssistantEvent::Usage(TokenUsage {
  500. input_tokens: response.usage.input_tokens,
  501. output_tokens: response.usage.output_tokens,
  502. cache_creation_input_tokens: response.usage.cache_creation_input_tokens,
  503. cache_read_input_tokens: response.usage.cache_read_input_tokens,
  504. }));
  505. events.push(AssistantEvent::MessageStop);
  506. Ok(events)
  507. }
  508. struct CliToolExecutor {
  509. renderer: TerminalRenderer,
  510. }
  511. impl CliToolExecutor {
  512. fn new() -> Self {
  513. Self {
  514. renderer: TerminalRenderer::new(),
  515. }
  516. }
  517. }
  518. impl ToolExecutor for CliToolExecutor {
  519. fn execute(&mut self, tool_name: &str, input: &str) -> Result<String, ToolError> {
  520. let value = serde_json::from_str(input)
  521. .map_err(|error| ToolError::new(format!("invalid tool input JSON: {error}")))?;
  522. match execute_tool(tool_name, &value) {
  523. Ok(output) => {
  524. let markdown = format!("### Tool `{tool_name}`\n\n```json\n{output}\n```\n");
  525. self.renderer
  526. .stream_markdown(&markdown, &mut io::stdout())
  527. .map_err(|error| ToolError::new(error.to_string()))?;
  528. Ok(output)
  529. }
  530. Err(error) => Err(ToolError::new(error)),
  531. }
  532. }
  533. }
  534. fn permission_policy_from_env() -> PermissionPolicy {
  535. let mode =
  536. env::var("RUSTY_CLAUDE_PERMISSION_MODE").unwrap_or_else(|_| "workspace-write".to_string());
  537. match mode.as_str() {
  538. "read-only" => PermissionPolicy::new(PermissionMode::Deny)
  539. .with_tool_mode("read_file", PermissionMode::Allow)
  540. .with_tool_mode("glob_search", PermissionMode::Allow)
  541. .with_tool_mode("grep_search", PermissionMode::Allow),
  542. _ => PermissionPolicy::new(PermissionMode::Allow),
  543. }
  544. }
  545. fn convert_messages(messages: &[ConversationMessage]) -> Vec<InputMessage> {
  546. messages
  547. .iter()
  548. .filter_map(|message| {
  549. let role = match message.role {
  550. MessageRole::System | MessageRole::User | MessageRole::Tool => "user",
  551. MessageRole::Assistant => "assistant",
  552. };
  553. let content = message
  554. .blocks
  555. .iter()
  556. .map(|block| match block {
  557. ContentBlock::Text { text } => InputContentBlock::Text { text: text.clone() },
  558. ContentBlock::ToolUse { id, name, input } => InputContentBlock::ToolUse {
  559. id: id.clone(),
  560. name: name.clone(),
  561. input: serde_json::from_str(input)
  562. .unwrap_or_else(|_| serde_json::json!({ "raw": input })),
  563. },
  564. ContentBlock::ToolResult {
  565. tool_use_id,
  566. output,
  567. is_error,
  568. ..
  569. } => InputContentBlock::ToolResult {
  570. tool_use_id: tool_use_id.clone(),
  571. content: vec![ToolResultContentBlock::Text {
  572. text: output.clone(),
  573. }],
  574. is_error: *is_error,
  575. },
  576. })
  577. .collect::<Vec<_>>();
  578. (!content.is_empty()).then(|| InputMessage {
  579. role: role.to_string(),
  580. content,
  581. })
  582. })
  583. .collect()
  584. }
  585. fn print_help() {
  586. println!("rusty-claude-cli");
  587. println!();
  588. println!("Usage:");
  589. println!(" rusty-claude-cli [--model MODEL] Start interactive REPL");
  590. println!(
  591. " rusty-claude-cli [--model MODEL] prompt TEXT Send one prompt and stream the response"
  592. );
  593. println!(" rusty-claude-cli dump-manifests");
  594. println!(" rusty-claude-cli bootstrap-plan");
  595. println!(" rusty-claude-cli system-prompt [--cwd PATH] [--date YYYY-MM-DD]");
  596. println!(" rusty-claude-cli --resume SESSION.json [/compact]");
  597. }
  598. #[cfg(test)]
  599. mod tests {
  600. use super::{parse_args, CliAction, DEFAULT_MODEL};
  601. use runtime::{ContentBlock, ConversationMessage, MessageRole};
  602. use std::path::PathBuf;
  603. #[test]
  604. fn defaults_to_repl_when_no_args() {
  605. assert_eq!(
  606. parse_args(&[]).expect("args should parse"),
  607. CliAction::Repl {
  608. model: DEFAULT_MODEL.to_string(),
  609. }
  610. );
  611. }
  612. #[test]
  613. fn parses_prompt_subcommand() {
  614. let args = vec![
  615. "prompt".to_string(),
  616. "hello".to_string(),
  617. "world".to_string(),
  618. ];
  619. assert_eq!(
  620. parse_args(&args).expect("args should parse"),
  621. CliAction::Prompt {
  622. prompt: "hello world".to_string(),
  623. model: DEFAULT_MODEL.to_string(),
  624. }
  625. );
  626. }
  627. #[test]
  628. fn parses_system_prompt_options() {
  629. let args = vec![
  630. "system-prompt".to_string(),
  631. "--cwd".to_string(),
  632. "/tmp/project".to_string(),
  633. "--date".to_string(),
  634. "2026-04-01".to_string(),
  635. ];
  636. assert_eq!(
  637. parse_args(&args).expect("args should parse"),
  638. CliAction::PrintSystemPrompt {
  639. cwd: PathBuf::from("/tmp/project"),
  640. date: "2026-04-01".to_string(),
  641. }
  642. );
  643. }
  644. #[test]
  645. fn parses_resume_flag_with_slash_command() {
  646. let args = vec![
  647. "--resume".to_string(),
  648. "session.json".to_string(),
  649. "/compact".to_string(),
  650. ];
  651. assert_eq!(
  652. parse_args(&args).expect("args should parse"),
  653. CliAction::ResumeSession {
  654. session_path: PathBuf::from("session.json"),
  655. command: Some("/compact".to_string()),
  656. }
  657. );
  658. }
  659. #[test]
  660. fn converts_tool_roundtrip_messages() {
  661. let messages = vec![
  662. ConversationMessage::user_text("hello"),
  663. ConversationMessage::assistant(vec![ContentBlock::ToolUse {
  664. id: "tool-1".to_string(),
  665. name: "bash".to_string(),
  666. input: "{\"command\":\"pwd\"}".to_string(),
  667. }]),
  668. ConversationMessage {
  669. role: MessageRole::Tool,
  670. blocks: vec![ContentBlock::ToolResult {
  671. tool_use_id: "tool-1".to_string(),
  672. tool_name: "bash".to_string(),
  673. output: "ok".to_string(),
  674. is_error: false,
  675. }],
  676. usage: None,
  677. },
  678. ];
  679. let converted = super::convert_messages(&messages);
  680. assert_eq!(converted.len(), 3);
  681. assert_eq!(converted[1].role, "assistant");
  682. assert_eq!(converted[2].role, "user");
  683. }
  684. }