main.rs 98 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988
  1. mod init;
  2. mod input;
  3. mod render;
  4. use std::collections::{BTreeMap, BTreeSet};
  5. use std::env;
  6. use std::fs;
  7. use std::io::{self, Read, Write};
  8. use std::net::TcpListener;
  9. use std::path::{Path, PathBuf};
  10. use std::process::Command;
  11. use std::time::{SystemTime, UNIX_EPOCH};
  12. use api::{
  13. resolve_startup_auth_source, AnthropicClient, AuthSource, ContentBlockDelta, InputContentBlock,
  14. InputMessage, MessageRequest, MessageResponse, OutputContentBlock,
  15. StreamEvent as ApiStreamEvent, ToolChoice, ToolDefinition, ToolResultContentBlock,
  16. };
  17. use commands::{
  18. render_slash_command_help, resume_supported_slash_commands, slash_command_specs, SlashCommand,
  19. };
  20. use compat_harness::{extract_manifest, UpstreamPaths};
  21. use init::initialize_repo;
  22. use render::{Spinner, TerminalRenderer};
  23. use runtime::{
  24. clear_oauth_credentials, generate_pkce_pair, generate_state, load_system_prompt,
  25. parse_oauth_callback_request_target, save_oauth_credentials, ApiClient, ApiRequest,
  26. AssistantEvent, CompactionConfig, ConfigLoader, ConfigSource, ContentBlock,
  27. ConversationMessage, ConversationRuntime, MessageRole, OAuthAuthorizationRequest,
  28. OAuthTokenExchangeRequest, PermissionMode, PermissionPolicy, ProjectContext, RuntimeError,
  29. Session, TokenUsage, ToolError, ToolExecutor, UsageTracker,
  30. };
  31. use serde_json::json;
  32. use tools::{execute_tool, mvp_tool_specs, ToolSpec};
  33. const DEFAULT_MODEL: &str = "claude-opus-4-6";
  34. fn max_tokens_for_model(model: &str) -> u32 {
  35. if model.contains("opus") {
  36. 32_000
  37. } else {
  38. 64_000
  39. }
  40. }
  41. const DEFAULT_DATE: &str = "2026-03-31";
  42. const DEFAULT_OAUTH_CALLBACK_PORT: u16 = 4545;
  43. const VERSION: &str = env!("CARGO_PKG_VERSION");
  44. const BUILD_TARGET: Option<&str> = option_env!("TARGET");
  45. const GIT_SHA: Option<&str> = option_env!("GIT_SHA");
  46. type AllowedToolSet = BTreeSet<String>;
  47. fn main() {
  48. if let Err(error) = run() {
  49. eprintln!(
  50. "error: {error}
  51. Run `claw --help` for usage."
  52. );
  53. std::process::exit(1);
  54. }
  55. }
  56. fn run() -> Result<(), Box<dyn std::error::Error>> {
  57. let args: Vec<String> = env::args().skip(1).collect();
  58. match parse_args(&args)? {
  59. CliAction::DumpManifests => dump_manifests(),
  60. CliAction::BootstrapPlan => print_bootstrap_plan(),
  61. CliAction::PrintSystemPrompt { cwd, date } => print_system_prompt(cwd, date),
  62. CliAction::Version => print_version(),
  63. CliAction::ResumeSession {
  64. session_path,
  65. commands,
  66. } => resume_session(&session_path, &commands),
  67. CliAction::Prompt {
  68. prompt,
  69. model,
  70. output_format,
  71. allowed_tools,
  72. permission_mode,
  73. } => LiveCli::new(model, false, allowed_tools, permission_mode)?
  74. .run_turn_with_output(&prompt, output_format)?,
  75. CliAction::Login => run_login()?,
  76. CliAction::Logout => run_logout()?,
  77. CliAction::Init => run_init()?,
  78. CliAction::Repl {
  79. model,
  80. allowed_tools,
  81. permission_mode,
  82. } => run_repl(model, allowed_tools, permission_mode)?,
  83. CliAction::Help => print_help(),
  84. }
  85. Ok(())
  86. }
  87. #[derive(Debug, Clone, PartialEq, Eq)]
  88. enum CliAction {
  89. DumpManifests,
  90. BootstrapPlan,
  91. PrintSystemPrompt {
  92. cwd: PathBuf,
  93. date: String,
  94. },
  95. Version,
  96. ResumeSession {
  97. session_path: PathBuf,
  98. commands: Vec<String>,
  99. },
  100. Prompt {
  101. prompt: String,
  102. model: String,
  103. output_format: CliOutputFormat,
  104. allowed_tools: Option<AllowedToolSet>,
  105. permission_mode: PermissionMode,
  106. },
  107. Login,
  108. Logout,
  109. Init,
  110. Repl {
  111. model: String,
  112. allowed_tools: Option<AllowedToolSet>,
  113. permission_mode: PermissionMode,
  114. },
  115. // prompt-mode formatting is only supported for non-interactive runs
  116. Help,
  117. }
  118. #[derive(Debug, Clone, Copy, PartialEq, Eq)]
  119. enum CliOutputFormat {
  120. Text,
  121. Json,
  122. }
  123. impl CliOutputFormat {
  124. fn parse(value: &str) -> Result<Self, String> {
  125. match value {
  126. "text" => Ok(Self::Text),
  127. "json" => Ok(Self::Json),
  128. other => Err(format!(
  129. "unsupported value for --output-format: {other} (expected text or json)"
  130. )),
  131. }
  132. }
  133. }
  134. #[allow(clippy::too_many_lines)]
  135. fn parse_args(args: &[String]) -> Result<CliAction, String> {
  136. let mut model = DEFAULT_MODEL.to_string();
  137. let mut output_format = CliOutputFormat::Text;
  138. let mut permission_mode = default_permission_mode();
  139. let mut wants_version = false;
  140. let mut allowed_tool_values = Vec::new();
  141. let mut rest = Vec::new();
  142. let mut index = 0;
  143. while index < args.len() {
  144. match args[index].as_str() {
  145. "--version" | "-V" => {
  146. wants_version = true;
  147. index += 1;
  148. }
  149. "--model" => {
  150. let value = args
  151. .get(index + 1)
  152. .ok_or_else(|| "missing value for --model".to_string())?;
  153. model = resolve_model_alias(value).to_string();
  154. index += 2;
  155. }
  156. flag if flag.starts_with("--model=") => {
  157. model = resolve_model_alias(&flag[8..]).to_string();
  158. index += 1;
  159. }
  160. "--output-format" => {
  161. let value = args
  162. .get(index + 1)
  163. .ok_or_else(|| "missing value for --output-format".to_string())?;
  164. output_format = CliOutputFormat::parse(value)?;
  165. index += 2;
  166. }
  167. "--permission-mode" => {
  168. let value = args
  169. .get(index + 1)
  170. .ok_or_else(|| "missing value for --permission-mode".to_string())?;
  171. permission_mode = parse_permission_mode_arg(value)?;
  172. index += 2;
  173. }
  174. flag if flag.starts_with("--output-format=") => {
  175. output_format = CliOutputFormat::parse(&flag[16..])?;
  176. index += 1;
  177. }
  178. flag if flag.starts_with("--permission-mode=") => {
  179. permission_mode = parse_permission_mode_arg(&flag[18..])?;
  180. index += 1;
  181. }
  182. "--dangerously-skip-permissions" => {
  183. permission_mode = PermissionMode::DangerFullAccess;
  184. index += 1;
  185. }
  186. "--allowedTools" | "--allowed-tools" => {
  187. let value = args
  188. .get(index + 1)
  189. .ok_or_else(|| "missing value for --allowedTools".to_string())?;
  190. allowed_tool_values.push(value.clone());
  191. index += 2;
  192. }
  193. flag if flag.starts_with("--allowedTools=") => {
  194. allowed_tool_values.push(flag[15..].to_string());
  195. index += 1;
  196. }
  197. flag if flag.starts_with("--allowed-tools=") => {
  198. allowed_tool_values.push(flag[16..].to_string());
  199. index += 1;
  200. }
  201. other => {
  202. rest.push(other.to_string());
  203. index += 1;
  204. }
  205. }
  206. }
  207. if wants_version {
  208. return Ok(CliAction::Version);
  209. }
  210. let allowed_tools = normalize_allowed_tools(&allowed_tool_values)?;
  211. if rest.is_empty() {
  212. return Ok(CliAction::Repl {
  213. model,
  214. allowed_tools,
  215. permission_mode,
  216. });
  217. }
  218. if matches!(rest.first().map(String::as_str), Some("--help" | "-h")) {
  219. return Ok(CliAction::Help);
  220. }
  221. if rest.first().map(String::as_str) == Some("--resume") {
  222. return parse_resume_args(&rest[1..]);
  223. }
  224. match rest[0].as_str() {
  225. "dump-manifests" => Ok(CliAction::DumpManifests),
  226. "bootstrap-plan" => Ok(CliAction::BootstrapPlan),
  227. "system-prompt" => parse_system_prompt_args(&rest[1..]),
  228. "login" => Ok(CliAction::Login),
  229. "logout" => Ok(CliAction::Logout),
  230. "init" => Ok(CliAction::Init),
  231. "prompt" => {
  232. let prompt = rest[1..].join(" ");
  233. if prompt.trim().is_empty() {
  234. return Err("prompt subcommand requires a prompt string".to_string());
  235. }
  236. Ok(CliAction::Prompt {
  237. prompt,
  238. model,
  239. output_format,
  240. allowed_tools,
  241. permission_mode,
  242. })
  243. }
  244. other if !other.starts_with('/') => Ok(CliAction::Prompt {
  245. prompt: rest.join(" "),
  246. model,
  247. output_format,
  248. allowed_tools,
  249. permission_mode,
  250. }),
  251. other => Err(format!("unknown subcommand: {other}")),
  252. }
  253. }
  254. fn resolve_model_alias(model: &str) -> &str {
  255. match model {
  256. "opus" => "claude-opus-4-6",
  257. "sonnet" => "claude-sonnet-4-6",
  258. "haiku" => "claude-haiku-4-5-20251213",
  259. _ => model,
  260. }
  261. }
  262. fn normalize_allowed_tools(values: &[String]) -> Result<Option<AllowedToolSet>, String> {
  263. if values.is_empty() {
  264. return Ok(None);
  265. }
  266. let canonical_names = mvp_tool_specs()
  267. .into_iter()
  268. .map(|spec| spec.name.to_string())
  269. .collect::<Vec<_>>();
  270. let mut name_map = canonical_names
  271. .iter()
  272. .map(|name| (normalize_tool_name(name), name.clone()))
  273. .collect::<BTreeMap<_, _>>();
  274. for (alias, canonical) in [
  275. ("read", "read_file"),
  276. ("write", "write_file"),
  277. ("edit", "edit_file"),
  278. ("glob", "glob_search"),
  279. ("grep", "grep_search"),
  280. ] {
  281. name_map.insert(alias.to_string(), canonical.to_string());
  282. }
  283. let mut allowed = AllowedToolSet::new();
  284. for value in values {
  285. for token in value
  286. .split(|ch: char| ch == ',' || ch.is_whitespace())
  287. .filter(|token| !token.is_empty())
  288. {
  289. let normalized = normalize_tool_name(token);
  290. let canonical = name_map.get(&normalized).ok_or_else(|| {
  291. format!(
  292. "unsupported tool in --allowedTools: {token} (expected one of: {})",
  293. canonical_names.join(", ")
  294. )
  295. })?;
  296. allowed.insert(canonical.clone());
  297. }
  298. }
  299. Ok(Some(allowed))
  300. }
  301. fn normalize_tool_name(value: &str) -> String {
  302. value.trim().replace('-', "_").to_ascii_lowercase()
  303. }
  304. fn parse_permission_mode_arg(value: &str) -> Result<PermissionMode, String> {
  305. normalize_permission_mode(value)
  306. .ok_or_else(|| {
  307. format!(
  308. "unsupported permission mode '{value}'. Use read-only, workspace-write, or danger-full-access."
  309. )
  310. })
  311. .map(permission_mode_from_label)
  312. }
  313. fn permission_mode_from_label(mode: &str) -> PermissionMode {
  314. match mode {
  315. "read-only" => PermissionMode::ReadOnly,
  316. "workspace-write" => PermissionMode::WorkspaceWrite,
  317. "danger-full-access" => PermissionMode::DangerFullAccess,
  318. other => panic!("unsupported permission mode label: {other}"),
  319. }
  320. }
  321. fn default_permission_mode() -> PermissionMode {
  322. env::var("RUSTY_CLAUDE_PERMISSION_MODE")
  323. .ok()
  324. .as_deref()
  325. .and_then(normalize_permission_mode)
  326. .map_or(PermissionMode::WorkspaceWrite, permission_mode_from_label)
  327. }
  328. fn filter_tool_specs(allowed_tools: Option<&AllowedToolSet>) -> Vec<tools::ToolSpec> {
  329. mvp_tool_specs()
  330. .into_iter()
  331. .filter(|spec| allowed_tools.is_none_or(|allowed| allowed.contains(spec.name)))
  332. .collect()
  333. }
  334. fn parse_system_prompt_args(args: &[String]) -> Result<CliAction, String> {
  335. let mut cwd = env::current_dir().map_err(|error| error.to_string())?;
  336. let mut date = DEFAULT_DATE.to_string();
  337. let mut index = 0;
  338. while index < args.len() {
  339. match args[index].as_str() {
  340. "--cwd" => {
  341. let value = args
  342. .get(index + 1)
  343. .ok_or_else(|| "missing value for --cwd".to_string())?;
  344. cwd = PathBuf::from(value);
  345. index += 2;
  346. }
  347. "--date" => {
  348. let value = args
  349. .get(index + 1)
  350. .ok_or_else(|| "missing value for --date".to_string())?;
  351. date.clone_from(value);
  352. index += 2;
  353. }
  354. other => return Err(format!("unknown system-prompt option: {other}")),
  355. }
  356. }
  357. Ok(CliAction::PrintSystemPrompt { cwd, date })
  358. }
  359. fn parse_resume_args(args: &[String]) -> Result<CliAction, String> {
  360. let session_path = args
  361. .first()
  362. .ok_or_else(|| "missing session path for --resume".to_string())
  363. .map(PathBuf::from)?;
  364. let commands = args[1..].to_vec();
  365. if commands
  366. .iter()
  367. .any(|command| !command.trim_start().starts_with('/'))
  368. {
  369. return Err("--resume trailing arguments must be slash commands".to_string());
  370. }
  371. Ok(CliAction::ResumeSession {
  372. session_path,
  373. commands,
  374. })
  375. }
  376. fn dump_manifests() {
  377. let workspace_dir = PathBuf::from(env!("CARGO_MANIFEST_DIR")).join("../..");
  378. let paths = UpstreamPaths::from_workspace_dir(&workspace_dir);
  379. match extract_manifest(&paths) {
  380. Ok(manifest) => {
  381. println!("commands: {}", manifest.commands.entries().len());
  382. println!("tools: {}", manifest.tools.entries().len());
  383. println!("bootstrap phases: {}", manifest.bootstrap.phases().len());
  384. }
  385. Err(error) => {
  386. eprintln!("failed to extract manifests: {error}");
  387. std::process::exit(1);
  388. }
  389. }
  390. }
  391. fn print_bootstrap_plan() {
  392. for phase in runtime::BootstrapPlan::claude_code_default().phases() {
  393. println!("- {phase:?}");
  394. }
  395. }
  396. fn run_login() -> Result<(), Box<dyn std::error::Error>> {
  397. let cwd = env::current_dir()?;
  398. let config = ConfigLoader::default_for(&cwd).load()?;
  399. let oauth = config.oauth().ok_or_else(|| {
  400. io::Error::new(
  401. io::ErrorKind::NotFound,
  402. "OAuth config is missing. Add settings.oauth.clientId/authorizeUrl/tokenUrl first.",
  403. )
  404. })?;
  405. let callback_port = oauth.callback_port.unwrap_or(DEFAULT_OAUTH_CALLBACK_PORT);
  406. let redirect_uri = runtime::loopback_redirect_uri(callback_port);
  407. let pkce = generate_pkce_pair()?;
  408. let state = generate_state()?;
  409. let authorize_url =
  410. OAuthAuthorizationRequest::from_config(oauth, redirect_uri.clone(), state.clone(), &pkce)
  411. .build_url();
  412. println!("Starting Claude OAuth login...");
  413. println!("Listening for callback on {redirect_uri}");
  414. if let Err(error) = open_browser(&authorize_url) {
  415. eprintln!("warning: failed to open browser automatically: {error}");
  416. println!("Open this URL manually:\n{authorize_url}");
  417. }
  418. let callback = wait_for_oauth_callback(callback_port)?;
  419. if let Some(error) = callback.error {
  420. let description = callback
  421. .error_description
  422. .unwrap_or_else(|| "authorization failed".to_string());
  423. return Err(io::Error::other(format!("{error}: {description}")).into());
  424. }
  425. let code = callback.code.ok_or_else(|| {
  426. io::Error::new(io::ErrorKind::InvalidData, "callback did not include code")
  427. })?;
  428. let returned_state = callback.state.ok_or_else(|| {
  429. io::Error::new(io::ErrorKind::InvalidData, "callback did not include state")
  430. })?;
  431. if returned_state != state {
  432. return Err(io::Error::new(io::ErrorKind::InvalidData, "oauth state mismatch").into());
  433. }
  434. let client = AnthropicClient::from_auth(AuthSource::None).with_base_url(api::read_base_url());
  435. let exchange_request =
  436. OAuthTokenExchangeRequest::from_config(oauth, code, state, pkce.verifier, redirect_uri);
  437. let runtime = tokio::runtime::Runtime::new()?;
  438. let token_set = runtime.block_on(client.exchange_oauth_code(oauth, &exchange_request))?;
  439. save_oauth_credentials(&runtime::OAuthTokenSet {
  440. access_token: token_set.access_token,
  441. refresh_token: token_set.refresh_token,
  442. expires_at: token_set.expires_at,
  443. scopes: token_set.scopes,
  444. })?;
  445. println!("Claude OAuth login complete.");
  446. Ok(())
  447. }
  448. fn run_logout() -> Result<(), Box<dyn std::error::Error>> {
  449. clear_oauth_credentials()?;
  450. println!("Claude OAuth credentials cleared.");
  451. Ok(())
  452. }
  453. fn open_browser(url: &str) -> io::Result<()> {
  454. let commands = if cfg!(target_os = "macos") {
  455. vec![("open", vec![url])]
  456. } else if cfg!(target_os = "windows") {
  457. vec![("cmd", vec!["/C", "start", "", url])]
  458. } else {
  459. vec![("xdg-open", vec![url])]
  460. };
  461. for (program, args) in commands {
  462. match Command::new(program).args(args).spawn() {
  463. Ok(_) => return Ok(()),
  464. Err(error) if error.kind() == io::ErrorKind::NotFound => {}
  465. Err(error) => return Err(error),
  466. }
  467. }
  468. Err(io::Error::new(
  469. io::ErrorKind::NotFound,
  470. "no supported browser opener command found",
  471. ))
  472. }
  473. fn wait_for_oauth_callback(
  474. port: u16,
  475. ) -> Result<runtime::OAuthCallbackParams, Box<dyn std::error::Error>> {
  476. let listener = TcpListener::bind(("127.0.0.1", port))?;
  477. let (mut stream, _) = listener.accept()?;
  478. let mut buffer = [0_u8; 4096];
  479. let bytes_read = stream.read(&mut buffer)?;
  480. let request = String::from_utf8_lossy(&buffer[..bytes_read]);
  481. let request_line = request.lines().next().ok_or_else(|| {
  482. io::Error::new(io::ErrorKind::InvalidData, "missing callback request line")
  483. })?;
  484. let target = request_line.split_whitespace().nth(1).ok_or_else(|| {
  485. io::Error::new(
  486. io::ErrorKind::InvalidData,
  487. "missing callback request target",
  488. )
  489. })?;
  490. let callback = parse_oauth_callback_request_target(target)
  491. .map_err(|error| io::Error::new(io::ErrorKind::InvalidData, error))?;
  492. let body = if callback.error.is_some() {
  493. "Claude OAuth login failed. You can close this window."
  494. } else {
  495. "Claude OAuth login succeeded. You can close this window."
  496. };
  497. let response = format!(
  498. "HTTP/1.1 200 OK\r\ncontent-type: text/plain; charset=utf-8\r\ncontent-length: {}\r\nconnection: close\r\n\r\n{}",
  499. body.len(),
  500. body
  501. );
  502. stream.write_all(response.as_bytes())?;
  503. Ok(callback)
  504. }
  505. fn print_system_prompt(cwd: PathBuf, date: String) {
  506. match load_system_prompt(cwd, date, env::consts::OS, "unknown") {
  507. Ok(sections) => println!("{}", sections.join("\n\n")),
  508. Err(error) => {
  509. eprintln!("failed to build system prompt: {error}");
  510. std::process::exit(1);
  511. }
  512. }
  513. }
  514. fn print_version() {
  515. println!("{}", render_version_report());
  516. }
  517. fn resume_session(session_path: &Path, commands: &[String]) {
  518. let session = match Session::load_from_path(session_path) {
  519. Ok(session) => session,
  520. Err(error) => {
  521. eprintln!("failed to restore session: {error}");
  522. std::process::exit(1);
  523. }
  524. };
  525. if commands.is_empty() {
  526. println!(
  527. "Restored session from {} ({} messages).",
  528. session_path.display(),
  529. session.messages.len()
  530. );
  531. return;
  532. }
  533. let mut session = session;
  534. for raw_command in commands {
  535. let Some(command) = SlashCommand::parse(raw_command) else {
  536. eprintln!("unsupported resumed command: {raw_command}");
  537. std::process::exit(2);
  538. };
  539. match run_resume_command(session_path, &session, &command) {
  540. Ok(ResumeCommandOutcome {
  541. session: next_session,
  542. message,
  543. }) => {
  544. session = next_session;
  545. if let Some(message) = message {
  546. println!("{message}");
  547. }
  548. }
  549. Err(error) => {
  550. eprintln!("{error}");
  551. std::process::exit(2);
  552. }
  553. }
  554. }
  555. }
  556. #[derive(Debug, Clone)]
  557. struct ResumeCommandOutcome {
  558. session: Session,
  559. message: Option<String>,
  560. }
  561. #[derive(Debug, Clone)]
  562. struct StatusContext {
  563. cwd: PathBuf,
  564. session_path: Option<PathBuf>,
  565. loaded_config_files: usize,
  566. discovered_config_files: usize,
  567. memory_file_count: usize,
  568. project_root: Option<PathBuf>,
  569. git_branch: Option<String>,
  570. }
  571. #[derive(Debug, Clone, Copy)]
  572. struct StatusUsage {
  573. message_count: usize,
  574. turns: u32,
  575. latest: TokenUsage,
  576. cumulative: TokenUsage,
  577. estimated_tokens: usize,
  578. }
  579. fn format_model_report(model: &str, message_count: usize, turns: u32) -> String {
  580. format!(
  581. "Model
  582. Current model {model}
  583. Session messages {message_count}
  584. Session turns {turns}
  585. Usage
  586. Inspect current model with /model
  587. Switch models with /model <name>"
  588. )
  589. }
  590. fn format_model_switch_report(previous: &str, next: &str, message_count: usize) -> String {
  591. format!(
  592. "Model updated
  593. Previous {previous}
  594. Current {next}
  595. Preserved msgs {message_count}"
  596. )
  597. }
  598. fn format_permissions_report(mode: &str) -> String {
  599. let modes = [
  600. ("read-only", "Read/search tools only", mode == "read-only"),
  601. (
  602. "workspace-write",
  603. "Edit files inside the workspace",
  604. mode == "workspace-write",
  605. ),
  606. (
  607. "danger-full-access",
  608. "Unrestricted tool access",
  609. mode == "danger-full-access",
  610. ),
  611. ]
  612. .into_iter()
  613. .map(|(name, description, is_current)| {
  614. let marker = if is_current {
  615. "● current"
  616. } else {
  617. "○ available"
  618. };
  619. format!(" {name:<18} {marker:<11} {description}")
  620. })
  621. .collect::<Vec<_>>()
  622. .join(
  623. "
  624. ",
  625. );
  626. format!(
  627. "Permissions
  628. Active mode {mode}
  629. Mode status live session default
  630. Modes
  631. {modes}
  632. Usage
  633. Inspect current mode with /permissions
  634. Switch modes with /permissions <mode>"
  635. )
  636. }
  637. fn format_permissions_switch_report(previous: &str, next: &str) -> String {
  638. format!(
  639. "Permissions updated
  640. Result mode switched
  641. Previous mode {previous}
  642. Active mode {next}
  643. Applies to subsequent tool calls
  644. Usage /permissions to inspect current mode"
  645. )
  646. }
  647. fn format_cost_report(usage: TokenUsage) -> String {
  648. format!(
  649. "Cost
  650. Input tokens {}
  651. Output tokens {}
  652. Cache create {}
  653. Cache read {}
  654. Total tokens {}",
  655. usage.input_tokens,
  656. usage.output_tokens,
  657. usage.cache_creation_input_tokens,
  658. usage.cache_read_input_tokens,
  659. usage.total_tokens(),
  660. )
  661. }
  662. fn format_resume_report(session_path: &str, message_count: usize, turns: u32) -> String {
  663. format!(
  664. "Session resumed
  665. Session file {session_path}
  666. Messages {message_count}
  667. Turns {turns}"
  668. )
  669. }
  670. fn format_compact_report(removed: usize, resulting_messages: usize, skipped: bool) -> String {
  671. if skipped {
  672. format!(
  673. "Compact
  674. Result skipped
  675. Reason session below compaction threshold
  676. Messages kept {resulting_messages}"
  677. )
  678. } else {
  679. format!(
  680. "Compact
  681. Result compacted
  682. Messages removed {removed}
  683. Messages kept {resulting_messages}"
  684. )
  685. }
  686. }
  687. fn parse_git_status_metadata(status: Option<&str>) -> (Option<PathBuf>, Option<String>) {
  688. let Some(status) = status else {
  689. return (None, None);
  690. };
  691. let branch = status.lines().next().and_then(|line| {
  692. line.strip_prefix("## ")
  693. .map(|line| {
  694. line.split(['.', ' '])
  695. .next()
  696. .unwrap_or_default()
  697. .to_string()
  698. })
  699. .filter(|value| !value.is_empty())
  700. });
  701. let project_root = find_git_root().ok();
  702. (project_root, branch)
  703. }
  704. fn find_git_root() -> Result<PathBuf, Box<dyn std::error::Error>> {
  705. let output = std::process::Command::new("git")
  706. .args(["rev-parse", "--show-toplevel"])
  707. .current_dir(env::current_dir()?)
  708. .output()?;
  709. if !output.status.success() {
  710. return Err("not a git repository".into());
  711. }
  712. let path = String::from_utf8(output.stdout)?.trim().to_string();
  713. if path.is_empty() {
  714. return Err("empty git root".into());
  715. }
  716. Ok(PathBuf::from(path))
  717. }
  718. #[allow(clippy::too_many_lines)]
  719. fn run_resume_command(
  720. session_path: &Path,
  721. session: &Session,
  722. command: &SlashCommand,
  723. ) -> Result<ResumeCommandOutcome, Box<dyn std::error::Error>> {
  724. match command {
  725. SlashCommand::Help => Ok(ResumeCommandOutcome {
  726. session: session.clone(),
  727. message: Some(render_repl_help()),
  728. }),
  729. SlashCommand::Compact => {
  730. let result = runtime::compact_session(
  731. session,
  732. CompactionConfig {
  733. max_estimated_tokens: 0,
  734. ..CompactionConfig::default()
  735. },
  736. );
  737. let removed = result.removed_message_count;
  738. let kept = result.compacted_session.messages.len();
  739. let skipped = removed == 0;
  740. result.compacted_session.save_to_path(session_path)?;
  741. Ok(ResumeCommandOutcome {
  742. session: result.compacted_session,
  743. message: Some(format_compact_report(removed, kept, skipped)),
  744. })
  745. }
  746. SlashCommand::Clear { confirm } => {
  747. if !confirm {
  748. return Ok(ResumeCommandOutcome {
  749. session: session.clone(),
  750. message: Some(
  751. "clear: confirmation required; rerun with /clear --confirm".to_string(),
  752. ),
  753. });
  754. }
  755. let cleared = Session::new();
  756. cleared.save_to_path(session_path)?;
  757. Ok(ResumeCommandOutcome {
  758. session: cleared,
  759. message: Some(format!(
  760. "Cleared resumed session file {}.",
  761. session_path.display()
  762. )),
  763. })
  764. }
  765. SlashCommand::Status => {
  766. let tracker = UsageTracker::from_session(session);
  767. let usage = tracker.cumulative_usage();
  768. Ok(ResumeCommandOutcome {
  769. session: session.clone(),
  770. message: Some(format_status_report(
  771. "restored-session",
  772. StatusUsage {
  773. message_count: session.messages.len(),
  774. turns: tracker.turns(),
  775. latest: tracker.current_turn_usage(),
  776. cumulative: usage,
  777. estimated_tokens: 0,
  778. },
  779. default_permission_mode().as_str(),
  780. &status_context(Some(session_path))?,
  781. )),
  782. })
  783. }
  784. SlashCommand::Cost => {
  785. let usage = UsageTracker::from_session(session).cumulative_usage();
  786. Ok(ResumeCommandOutcome {
  787. session: session.clone(),
  788. message: Some(format_cost_report(usage)),
  789. })
  790. }
  791. SlashCommand::Config { section } => Ok(ResumeCommandOutcome {
  792. session: session.clone(),
  793. message: Some(render_config_report(section.as_deref())?),
  794. }),
  795. SlashCommand::Memory => Ok(ResumeCommandOutcome {
  796. session: session.clone(),
  797. message: Some(render_memory_report()?),
  798. }),
  799. SlashCommand::Init => Ok(ResumeCommandOutcome {
  800. session: session.clone(),
  801. message: Some(init_claude_md()?),
  802. }),
  803. SlashCommand::Diff => Ok(ResumeCommandOutcome {
  804. session: session.clone(),
  805. message: Some(render_diff_report()?),
  806. }),
  807. SlashCommand::Version => Ok(ResumeCommandOutcome {
  808. session: session.clone(),
  809. message: Some(render_version_report()),
  810. }),
  811. SlashCommand::Export { path } => {
  812. let export_path = resolve_export_path(path.as_deref(), session)?;
  813. fs::write(&export_path, render_export_text(session))?;
  814. Ok(ResumeCommandOutcome {
  815. session: session.clone(),
  816. message: Some(format!(
  817. "Export\n Result wrote transcript\n File {}\n Messages {}",
  818. export_path.display(),
  819. session.messages.len(),
  820. )),
  821. })
  822. }
  823. SlashCommand::Resume { .. }
  824. | SlashCommand::Model { .. }
  825. | SlashCommand::Permissions { .. }
  826. | SlashCommand::Session { .. }
  827. | SlashCommand::Unknown(_) => Err("unsupported resumed slash command".into()),
  828. }
  829. }
  830. fn run_repl(
  831. model: String,
  832. allowed_tools: Option<AllowedToolSet>,
  833. permission_mode: PermissionMode,
  834. ) -> Result<(), Box<dyn std::error::Error>> {
  835. let mut cli = LiveCli::new(model, true, allowed_tools, permission_mode)?;
  836. let mut editor = input::LineEditor::new("> ", slash_command_completion_candidates());
  837. println!("{}", cli.startup_banner());
  838. loop {
  839. match editor.read_line()? {
  840. input::ReadOutcome::Submit(input) => {
  841. let trimmed = input.trim().to_string();
  842. if trimmed.is_empty() {
  843. continue;
  844. }
  845. if matches!(trimmed.as_str(), "/exit" | "/quit") {
  846. cli.persist_session()?;
  847. break;
  848. }
  849. if let Some(command) = SlashCommand::parse(&trimmed) {
  850. if cli.handle_repl_command(command)? {
  851. cli.persist_session()?;
  852. }
  853. continue;
  854. }
  855. editor.push_history(input);
  856. cli.run_turn(&trimmed)?;
  857. }
  858. input::ReadOutcome::Cancel => {}
  859. input::ReadOutcome::Exit => {
  860. cli.persist_session()?;
  861. break;
  862. }
  863. }
  864. }
  865. Ok(())
  866. }
  867. #[derive(Debug, Clone)]
  868. struct SessionHandle {
  869. id: String,
  870. path: PathBuf,
  871. }
  872. #[derive(Debug, Clone)]
  873. struct ManagedSessionSummary {
  874. id: String,
  875. path: PathBuf,
  876. modified_epoch_secs: u64,
  877. message_count: usize,
  878. }
  879. struct LiveCli {
  880. model: String,
  881. allowed_tools: Option<AllowedToolSet>,
  882. permission_mode: PermissionMode,
  883. system_prompt: Vec<String>,
  884. runtime: ConversationRuntime<AnthropicRuntimeClient, CliToolExecutor>,
  885. session: SessionHandle,
  886. }
  887. impl LiveCli {
  888. fn new(
  889. model: String,
  890. enable_tools: bool,
  891. allowed_tools: Option<AllowedToolSet>,
  892. permission_mode: PermissionMode,
  893. ) -> Result<Self, Box<dyn std::error::Error>> {
  894. let system_prompt = build_system_prompt()?;
  895. let session = create_managed_session_handle()?;
  896. let runtime = build_runtime(
  897. Session::new(),
  898. model.clone(),
  899. system_prompt.clone(),
  900. enable_tools,
  901. allowed_tools.clone(),
  902. permission_mode,
  903. )?;
  904. let cli = Self {
  905. model,
  906. allowed_tools,
  907. permission_mode,
  908. system_prompt,
  909. runtime,
  910. session,
  911. };
  912. cli.persist_session()?;
  913. Ok(cli)
  914. }
  915. fn startup_banner(&self) -> String {
  916. let cwd = env::current_dir().map_or_else(
  917. |_| "<unknown>".to_string(),
  918. |path| path.display().to_string(),
  919. );
  920. format!(
  921. "\x1b[38;5;196m\
  922. ██████╗██╗ █████╗ ██╗ ██╗\n\
  923. ██╔════╝██║ ██╔══██╗██║ ██║\n\
  924. ██║ ██║ ███████║██║ █╗ ██║\n\
  925. ██║ ██║ ██╔══██║██║███╗██║\n\
  926. ╚██████╗███████╗██║ ██║╚███╔███╔╝\n\
  927. ╚═════╝╚══════╝╚═╝ ╚═╝ ╚══╝╚══╝\x1b[0m \x1b[38;5;208mCode\x1b[0m 🦞\n\n\
  928. \x1b[2mModel\x1b[0m {}\n\
  929. \x1b[2mPermissions\x1b[0m {}\n\
  930. \x1b[2mDirectory\x1b[0m {}\n\
  931. \x1b[2mSession\x1b[0m {}\n\n\
  932. Type \x1b[1m/help\x1b[0m for commands · \x1b[2mShift+Enter\x1b[0m for newline",
  933. self.model,
  934. self.permission_mode.as_str(),
  935. cwd,
  936. self.session.id,
  937. )
  938. }
  939. fn run_turn(&mut self, input: &str) -> Result<(), Box<dyn std::error::Error>> {
  940. let mut spinner = Spinner::new();
  941. let mut stdout = io::stdout();
  942. spinner.tick(
  943. "🦀 Thinking...",
  944. TerminalRenderer::new().color_theme(),
  945. &mut stdout,
  946. )?;
  947. let mut permission_prompter = CliPermissionPrompter::new(self.permission_mode);
  948. let result = self.runtime.run_turn(input, Some(&mut permission_prompter));
  949. match result {
  950. Ok(_) => {
  951. spinner.finish(
  952. "✨ Done",
  953. TerminalRenderer::new().color_theme(),
  954. &mut stdout,
  955. )?;
  956. println!();
  957. self.persist_session()?;
  958. Ok(())
  959. }
  960. Err(error) => {
  961. spinner.fail(
  962. "❌ Request failed",
  963. TerminalRenderer::new().color_theme(),
  964. &mut stdout,
  965. )?;
  966. Err(Box::new(error))
  967. }
  968. }
  969. }
  970. fn run_turn_with_output(
  971. &mut self,
  972. input: &str,
  973. output_format: CliOutputFormat,
  974. ) -> Result<(), Box<dyn std::error::Error>> {
  975. match output_format {
  976. CliOutputFormat::Text => self.run_turn(input),
  977. CliOutputFormat::Json => self.run_prompt_json(input),
  978. }
  979. }
  980. fn run_prompt_json(&mut self, input: &str) -> Result<(), Box<dyn std::error::Error>> {
  981. let client = AnthropicClient::from_auth(resolve_cli_auth_source()?)
  982. .with_base_url(api::read_base_url());
  983. let request = MessageRequest {
  984. model: self.model.clone(),
  985. max_tokens: max_tokens_for_model(&self.model),
  986. messages: vec![InputMessage {
  987. role: "user".to_string(),
  988. content: vec![InputContentBlock::Text {
  989. text: input.to_string(),
  990. }],
  991. }],
  992. system: (!self.system_prompt.is_empty()).then(|| self.system_prompt.join("\n\n")),
  993. tools: None,
  994. tool_choice: None,
  995. stream: false,
  996. };
  997. let runtime = tokio::runtime::Runtime::new()?;
  998. let response = runtime.block_on(client.send_message(&request))?;
  999. let text = response
  1000. .content
  1001. .iter()
  1002. .filter_map(|block| match block {
  1003. OutputContentBlock::Text { text } => Some(text.as_str()),
  1004. OutputContentBlock::ToolUse { .. } => None,
  1005. })
  1006. .collect::<Vec<_>>()
  1007. .join("");
  1008. println!(
  1009. "{}",
  1010. json!({
  1011. "message": text,
  1012. "model": self.model,
  1013. "usage": {
  1014. "input_tokens": response.usage.input_tokens,
  1015. "output_tokens": response.usage.output_tokens,
  1016. "cache_creation_input_tokens": response.usage.cache_creation_input_tokens,
  1017. "cache_read_input_tokens": response.usage.cache_read_input_tokens,
  1018. }
  1019. })
  1020. );
  1021. Ok(())
  1022. }
  1023. fn handle_repl_command(
  1024. &mut self,
  1025. command: SlashCommand,
  1026. ) -> Result<bool, Box<dyn std::error::Error>> {
  1027. Ok(match command {
  1028. SlashCommand::Help => {
  1029. println!("{}", render_repl_help());
  1030. false
  1031. }
  1032. SlashCommand::Status => {
  1033. self.print_status();
  1034. false
  1035. }
  1036. SlashCommand::Compact => {
  1037. self.compact()?;
  1038. false
  1039. }
  1040. SlashCommand::Model { model } => self.set_model(model)?,
  1041. SlashCommand::Permissions { mode } => self.set_permissions(mode)?,
  1042. SlashCommand::Clear { confirm } => self.clear_session(confirm)?,
  1043. SlashCommand::Cost => {
  1044. self.print_cost();
  1045. false
  1046. }
  1047. SlashCommand::Resume { session_path } => self.resume_session(session_path)?,
  1048. SlashCommand::Config { section } => {
  1049. Self::print_config(section.as_deref())?;
  1050. false
  1051. }
  1052. SlashCommand::Memory => {
  1053. Self::print_memory()?;
  1054. false
  1055. }
  1056. SlashCommand::Init => {
  1057. run_init()?;
  1058. false
  1059. }
  1060. SlashCommand::Diff => {
  1061. Self::print_diff()?;
  1062. false
  1063. }
  1064. SlashCommand::Version => {
  1065. Self::print_version();
  1066. false
  1067. }
  1068. SlashCommand::Export { path } => {
  1069. self.export_session(path.as_deref())?;
  1070. false
  1071. }
  1072. SlashCommand::Session { action, target } => {
  1073. self.handle_session_command(action.as_deref(), target.as_deref())?
  1074. }
  1075. SlashCommand::Unknown(name) => {
  1076. eprintln!("unknown slash command: /{name}");
  1077. false
  1078. }
  1079. })
  1080. }
  1081. fn persist_session(&self) -> Result<(), Box<dyn std::error::Error>> {
  1082. self.runtime.session().save_to_path(&self.session.path)?;
  1083. Ok(())
  1084. }
  1085. fn print_status(&self) {
  1086. let cumulative = self.runtime.usage().cumulative_usage();
  1087. let latest = self.runtime.usage().current_turn_usage();
  1088. println!(
  1089. "{}",
  1090. format_status_report(
  1091. &self.model,
  1092. StatusUsage {
  1093. message_count: self.runtime.session().messages.len(),
  1094. turns: self.runtime.usage().turns(),
  1095. latest,
  1096. cumulative,
  1097. estimated_tokens: self.runtime.estimated_tokens(),
  1098. },
  1099. self.permission_mode.as_str(),
  1100. &status_context(Some(&self.session.path)).expect("status context should load"),
  1101. )
  1102. );
  1103. }
  1104. fn set_model(&mut self, model: Option<String>) -> Result<bool, Box<dyn std::error::Error>> {
  1105. let Some(model) = model else {
  1106. println!(
  1107. "{}",
  1108. format_model_report(
  1109. &self.model,
  1110. self.runtime.session().messages.len(),
  1111. self.runtime.usage().turns(),
  1112. )
  1113. );
  1114. return Ok(false);
  1115. };
  1116. let model = resolve_model_alias(&model).to_string();
  1117. if model == self.model {
  1118. println!(
  1119. "{}",
  1120. format_model_report(
  1121. &self.model,
  1122. self.runtime.session().messages.len(),
  1123. self.runtime.usage().turns(),
  1124. )
  1125. );
  1126. return Ok(false);
  1127. }
  1128. let previous = self.model.clone();
  1129. let session = self.runtime.session().clone();
  1130. let message_count = session.messages.len();
  1131. self.runtime = build_runtime(
  1132. session,
  1133. model.clone(),
  1134. self.system_prompt.clone(),
  1135. true,
  1136. self.allowed_tools.clone(),
  1137. self.permission_mode,
  1138. )?;
  1139. self.model.clone_from(&model);
  1140. println!(
  1141. "{}",
  1142. format_model_switch_report(&previous, &model, message_count)
  1143. );
  1144. Ok(true)
  1145. }
  1146. fn set_permissions(
  1147. &mut self,
  1148. mode: Option<String>,
  1149. ) -> Result<bool, Box<dyn std::error::Error>> {
  1150. let Some(mode) = mode else {
  1151. println!(
  1152. "{}",
  1153. format_permissions_report(self.permission_mode.as_str())
  1154. );
  1155. return Ok(false);
  1156. };
  1157. let normalized = normalize_permission_mode(&mode).ok_or_else(|| {
  1158. format!(
  1159. "unsupported permission mode '{mode}'. Use read-only, workspace-write, or danger-full-access."
  1160. )
  1161. })?;
  1162. if normalized == self.permission_mode.as_str() {
  1163. println!("{}", format_permissions_report(normalized));
  1164. return Ok(false);
  1165. }
  1166. let previous = self.permission_mode.as_str().to_string();
  1167. let session = self.runtime.session().clone();
  1168. self.permission_mode = permission_mode_from_label(normalized);
  1169. self.runtime = build_runtime(
  1170. session,
  1171. self.model.clone(),
  1172. self.system_prompt.clone(),
  1173. true,
  1174. self.allowed_tools.clone(),
  1175. self.permission_mode,
  1176. )?;
  1177. println!(
  1178. "{}",
  1179. format_permissions_switch_report(&previous, normalized)
  1180. );
  1181. Ok(true)
  1182. }
  1183. fn clear_session(&mut self, confirm: bool) -> Result<bool, Box<dyn std::error::Error>> {
  1184. if !confirm {
  1185. println!(
  1186. "clear: confirmation required; run /clear --confirm to start a fresh session."
  1187. );
  1188. return Ok(false);
  1189. }
  1190. self.session = create_managed_session_handle()?;
  1191. self.runtime = build_runtime(
  1192. Session::new(),
  1193. self.model.clone(),
  1194. self.system_prompt.clone(),
  1195. true,
  1196. self.allowed_tools.clone(),
  1197. self.permission_mode,
  1198. )?;
  1199. println!(
  1200. "Session cleared\n Mode fresh session\n Preserved model {}\n Permission mode {}\n Session {}",
  1201. self.model,
  1202. self.permission_mode.as_str(),
  1203. self.session.id,
  1204. );
  1205. Ok(true)
  1206. }
  1207. fn print_cost(&self) {
  1208. let cumulative = self.runtime.usage().cumulative_usage();
  1209. println!("{}", format_cost_report(cumulative));
  1210. }
  1211. fn resume_session(
  1212. &mut self,
  1213. session_path: Option<String>,
  1214. ) -> Result<bool, Box<dyn std::error::Error>> {
  1215. let Some(session_ref) = session_path else {
  1216. println!("Usage: /resume <session-path>");
  1217. return Ok(false);
  1218. };
  1219. let handle = resolve_session_reference(&session_ref)?;
  1220. let session = Session::load_from_path(&handle.path)?;
  1221. let message_count = session.messages.len();
  1222. self.runtime = build_runtime(
  1223. session,
  1224. self.model.clone(),
  1225. self.system_prompt.clone(),
  1226. true,
  1227. self.allowed_tools.clone(),
  1228. self.permission_mode,
  1229. )?;
  1230. self.session = handle;
  1231. println!(
  1232. "{}",
  1233. format_resume_report(
  1234. &self.session.path.display().to_string(),
  1235. message_count,
  1236. self.runtime.usage().turns(),
  1237. )
  1238. );
  1239. Ok(true)
  1240. }
  1241. fn print_config(section: Option<&str>) -> Result<(), Box<dyn std::error::Error>> {
  1242. println!("{}", render_config_report(section)?);
  1243. Ok(())
  1244. }
  1245. fn print_memory() -> Result<(), Box<dyn std::error::Error>> {
  1246. println!("{}", render_memory_report()?);
  1247. Ok(())
  1248. }
  1249. fn print_diff() -> Result<(), Box<dyn std::error::Error>> {
  1250. println!("{}", render_diff_report()?);
  1251. Ok(())
  1252. }
  1253. fn print_version() {
  1254. println!("{}", render_version_report());
  1255. }
  1256. fn export_session(
  1257. &self,
  1258. requested_path: Option<&str>,
  1259. ) -> Result<(), Box<dyn std::error::Error>> {
  1260. let export_path = resolve_export_path(requested_path, self.runtime.session())?;
  1261. fs::write(&export_path, render_export_text(self.runtime.session()))?;
  1262. println!(
  1263. "Export\n Result wrote transcript\n File {}\n Messages {}",
  1264. export_path.display(),
  1265. self.runtime.session().messages.len(),
  1266. );
  1267. Ok(())
  1268. }
  1269. fn handle_session_command(
  1270. &mut self,
  1271. action: Option<&str>,
  1272. target: Option<&str>,
  1273. ) -> Result<bool, Box<dyn std::error::Error>> {
  1274. match action {
  1275. None | Some("list") => {
  1276. println!("{}", render_session_list(&self.session.id)?);
  1277. Ok(false)
  1278. }
  1279. Some("switch") => {
  1280. let Some(target) = target else {
  1281. println!("Usage: /session switch <session-id>");
  1282. return Ok(false);
  1283. };
  1284. let handle = resolve_session_reference(target)?;
  1285. let session = Session::load_from_path(&handle.path)?;
  1286. let message_count = session.messages.len();
  1287. self.runtime = build_runtime(
  1288. session,
  1289. self.model.clone(),
  1290. self.system_prompt.clone(),
  1291. true,
  1292. self.allowed_tools.clone(),
  1293. self.permission_mode,
  1294. )?;
  1295. self.session = handle;
  1296. println!(
  1297. "Session switched\n Active session {}\n File {}\n Messages {}",
  1298. self.session.id,
  1299. self.session.path.display(),
  1300. message_count,
  1301. );
  1302. Ok(true)
  1303. }
  1304. Some(other) => {
  1305. println!("Unknown /session action '{other}'. Use /session list or /session switch <session-id>.");
  1306. Ok(false)
  1307. }
  1308. }
  1309. }
  1310. fn compact(&mut self) -> Result<(), Box<dyn std::error::Error>> {
  1311. let result = self.runtime.compact(CompactionConfig::default());
  1312. let removed = result.removed_message_count;
  1313. let kept = result.compacted_session.messages.len();
  1314. let skipped = removed == 0;
  1315. self.runtime = build_runtime(
  1316. result.compacted_session,
  1317. self.model.clone(),
  1318. self.system_prompt.clone(),
  1319. true,
  1320. self.allowed_tools.clone(),
  1321. self.permission_mode,
  1322. )?;
  1323. self.persist_session()?;
  1324. println!("{}", format_compact_report(removed, kept, skipped));
  1325. Ok(())
  1326. }
  1327. }
  1328. fn sessions_dir() -> Result<PathBuf, Box<dyn std::error::Error>> {
  1329. let cwd = env::current_dir()?;
  1330. let path = cwd.join(".claude").join("sessions");
  1331. fs::create_dir_all(&path)?;
  1332. Ok(path)
  1333. }
  1334. fn create_managed_session_handle() -> Result<SessionHandle, Box<dyn std::error::Error>> {
  1335. let id = generate_session_id();
  1336. let path = sessions_dir()?.join(format!("{id}.json"));
  1337. Ok(SessionHandle { id, path })
  1338. }
  1339. fn generate_session_id() -> String {
  1340. let millis = SystemTime::now()
  1341. .duration_since(UNIX_EPOCH)
  1342. .map(|duration| duration.as_millis())
  1343. .unwrap_or_default();
  1344. format!("session-{millis}")
  1345. }
  1346. fn resolve_session_reference(reference: &str) -> Result<SessionHandle, Box<dyn std::error::Error>> {
  1347. let direct = PathBuf::from(reference);
  1348. let path = if direct.exists() {
  1349. direct
  1350. } else {
  1351. sessions_dir()?.join(format!("{reference}.json"))
  1352. };
  1353. if !path.exists() {
  1354. return Err(format!("session not found: {reference}").into());
  1355. }
  1356. let id = path
  1357. .file_stem()
  1358. .and_then(|value| value.to_str())
  1359. .unwrap_or(reference)
  1360. .to_string();
  1361. Ok(SessionHandle { id, path })
  1362. }
  1363. fn list_managed_sessions() -> Result<Vec<ManagedSessionSummary>, Box<dyn std::error::Error>> {
  1364. let mut sessions = Vec::new();
  1365. for entry in fs::read_dir(sessions_dir()?)? {
  1366. let entry = entry?;
  1367. let path = entry.path();
  1368. if path.extension().and_then(|ext| ext.to_str()) != Some("json") {
  1369. continue;
  1370. }
  1371. let metadata = entry.metadata()?;
  1372. let modified_epoch_secs = metadata
  1373. .modified()
  1374. .ok()
  1375. .and_then(|time| time.duration_since(UNIX_EPOCH).ok())
  1376. .map(|duration| duration.as_secs())
  1377. .unwrap_or_default();
  1378. let message_count = Session::load_from_path(&path)
  1379. .map(|session| session.messages.len())
  1380. .unwrap_or_default();
  1381. let id = path
  1382. .file_stem()
  1383. .and_then(|value| value.to_str())
  1384. .unwrap_or("unknown")
  1385. .to_string();
  1386. sessions.push(ManagedSessionSummary {
  1387. id,
  1388. path,
  1389. modified_epoch_secs,
  1390. message_count,
  1391. });
  1392. }
  1393. sessions.sort_by(|left, right| right.modified_epoch_secs.cmp(&left.modified_epoch_secs));
  1394. Ok(sessions)
  1395. }
  1396. fn render_session_list(active_session_id: &str) -> Result<String, Box<dyn std::error::Error>> {
  1397. let sessions = list_managed_sessions()?;
  1398. let mut lines = vec![
  1399. "Sessions".to_string(),
  1400. format!(" Directory {}", sessions_dir()?.display()),
  1401. ];
  1402. if sessions.is_empty() {
  1403. lines.push(" No managed sessions saved yet.".to_string());
  1404. return Ok(lines.join("\n"));
  1405. }
  1406. for session in sessions {
  1407. let marker = if session.id == active_session_id {
  1408. "● current"
  1409. } else {
  1410. "○ saved"
  1411. };
  1412. lines.push(format!(
  1413. " {id:<20} {marker:<10} msgs={msgs:<4} modified={modified} path={path}",
  1414. id = session.id,
  1415. msgs = session.message_count,
  1416. modified = session.modified_epoch_secs,
  1417. path = session.path.display(),
  1418. ));
  1419. }
  1420. Ok(lines.join("\n"))
  1421. }
  1422. fn render_repl_help() -> String {
  1423. [
  1424. "REPL".to_string(),
  1425. " /exit Quit the REPL".to_string(),
  1426. " /quit Quit the REPL".to_string(),
  1427. " Up/Down Navigate prompt history".to_string(),
  1428. " Tab Complete slash commands".to_string(),
  1429. " Ctrl-C Clear input (or exit on empty prompt)".to_string(),
  1430. " Shift+Enter/Ctrl+J Insert a newline".to_string(),
  1431. String::new(),
  1432. render_slash_command_help(),
  1433. ]
  1434. .join(
  1435. "
  1436. ",
  1437. )
  1438. }
  1439. fn status_context(
  1440. session_path: Option<&Path>,
  1441. ) -> Result<StatusContext, Box<dyn std::error::Error>> {
  1442. let cwd = env::current_dir()?;
  1443. let loader = ConfigLoader::default_for(&cwd);
  1444. let discovered_config_files = loader.discover().len();
  1445. let runtime_config = loader.load()?;
  1446. let project_context = ProjectContext::discover_with_git(&cwd, DEFAULT_DATE)?;
  1447. let (project_root, git_branch) =
  1448. parse_git_status_metadata(project_context.git_status.as_deref());
  1449. Ok(StatusContext {
  1450. cwd,
  1451. session_path: session_path.map(Path::to_path_buf),
  1452. loaded_config_files: runtime_config.loaded_entries().len(),
  1453. discovered_config_files,
  1454. memory_file_count: project_context.instruction_files.len(),
  1455. project_root,
  1456. git_branch,
  1457. })
  1458. }
  1459. fn format_status_report(
  1460. model: &str,
  1461. usage: StatusUsage,
  1462. permission_mode: &str,
  1463. context: &StatusContext,
  1464. ) -> String {
  1465. [
  1466. format!(
  1467. "Status
  1468. Model {model}
  1469. Permission mode {permission_mode}
  1470. Messages {}
  1471. Turns {}
  1472. Estimated tokens {}",
  1473. usage.message_count, usage.turns, usage.estimated_tokens,
  1474. ),
  1475. format!(
  1476. "Usage
  1477. Latest total {}
  1478. Cumulative input {}
  1479. Cumulative output {}
  1480. Cumulative total {}",
  1481. usage.latest.total_tokens(),
  1482. usage.cumulative.input_tokens,
  1483. usage.cumulative.output_tokens,
  1484. usage.cumulative.total_tokens(),
  1485. ),
  1486. format!(
  1487. "Workspace
  1488. Cwd {}
  1489. Project root {}
  1490. Git branch {}
  1491. Session {}
  1492. Config files loaded {}/{}
  1493. Memory files {}",
  1494. context.cwd.display(),
  1495. context
  1496. .project_root
  1497. .as_ref()
  1498. .map_or_else(|| "unknown".to_string(), |path| path.display().to_string()),
  1499. context.git_branch.as_deref().unwrap_or("unknown"),
  1500. context.session_path.as_ref().map_or_else(
  1501. || "live-repl".to_string(),
  1502. |path| path.display().to_string()
  1503. ),
  1504. context.loaded_config_files,
  1505. context.discovered_config_files,
  1506. context.memory_file_count,
  1507. ),
  1508. ]
  1509. .join(
  1510. "
  1511. ",
  1512. )
  1513. }
  1514. fn render_config_report(section: Option<&str>) -> Result<String, Box<dyn std::error::Error>> {
  1515. let cwd = env::current_dir()?;
  1516. let loader = ConfigLoader::default_for(&cwd);
  1517. let discovered = loader.discover();
  1518. let runtime_config = loader.load()?;
  1519. let mut lines = vec![
  1520. format!(
  1521. "Config
  1522. Working directory {}
  1523. Loaded files {}
  1524. Merged keys {}",
  1525. cwd.display(),
  1526. runtime_config.loaded_entries().len(),
  1527. runtime_config.merged().len()
  1528. ),
  1529. "Discovered files".to_string(),
  1530. ];
  1531. for entry in discovered {
  1532. let source = match entry.source {
  1533. ConfigSource::User => "user",
  1534. ConfigSource::Project => "project",
  1535. ConfigSource::Local => "local",
  1536. };
  1537. let status = if runtime_config
  1538. .loaded_entries()
  1539. .iter()
  1540. .any(|loaded_entry| loaded_entry.path == entry.path)
  1541. {
  1542. "loaded"
  1543. } else {
  1544. "missing"
  1545. };
  1546. lines.push(format!(
  1547. " {source:<7} {status:<7} {}",
  1548. entry.path.display()
  1549. ));
  1550. }
  1551. if let Some(section) = section {
  1552. lines.push(format!("Merged section: {section}"));
  1553. let value = match section {
  1554. "env" => runtime_config.get("env"),
  1555. "hooks" => runtime_config.get("hooks"),
  1556. "model" => runtime_config.get("model"),
  1557. other => {
  1558. lines.push(format!(
  1559. " Unsupported config section '{other}'. Use env, hooks, or model."
  1560. ));
  1561. return Ok(lines.join(
  1562. "
  1563. ",
  1564. ));
  1565. }
  1566. };
  1567. lines.push(format!(
  1568. " {}",
  1569. match value {
  1570. Some(value) => value.render(),
  1571. None => "<unset>".to_string(),
  1572. }
  1573. ));
  1574. return Ok(lines.join(
  1575. "
  1576. ",
  1577. ));
  1578. }
  1579. lines.push("Merged JSON".to_string());
  1580. lines.push(format!(" {}", runtime_config.as_json().render()));
  1581. Ok(lines.join(
  1582. "
  1583. ",
  1584. ))
  1585. }
  1586. fn render_memory_report() -> Result<String, Box<dyn std::error::Error>> {
  1587. let cwd = env::current_dir()?;
  1588. let project_context = ProjectContext::discover(&cwd, DEFAULT_DATE)?;
  1589. let mut lines = vec![format!(
  1590. "Memory
  1591. Working directory {}
  1592. Instruction files {}",
  1593. cwd.display(),
  1594. project_context.instruction_files.len()
  1595. )];
  1596. if project_context.instruction_files.is_empty() {
  1597. lines.push("Discovered files".to_string());
  1598. lines.push(
  1599. " No CLAUDE instruction files discovered in the current directory ancestry."
  1600. .to_string(),
  1601. );
  1602. } else {
  1603. lines.push("Discovered files".to_string());
  1604. for (index, file) in project_context.instruction_files.iter().enumerate() {
  1605. let preview = file.content.lines().next().unwrap_or("").trim();
  1606. let preview = if preview.is_empty() {
  1607. "<empty>"
  1608. } else {
  1609. preview
  1610. };
  1611. lines.push(format!(" {}. {}", index + 1, file.path.display(),));
  1612. lines.push(format!(
  1613. " lines={} preview={}",
  1614. file.content.lines().count(),
  1615. preview
  1616. ));
  1617. }
  1618. }
  1619. Ok(lines.join(
  1620. "
  1621. ",
  1622. ))
  1623. }
  1624. fn init_claude_md() -> Result<String, Box<dyn std::error::Error>> {
  1625. let cwd = env::current_dir()?;
  1626. Ok(initialize_repo(&cwd)?.render())
  1627. }
  1628. fn run_init() -> Result<(), Box<dyn std::error::Error>> {
  1629. println!("{}", init_claude_md()?);
  1630. Ok(())
  1631. }
  1632. fn normalize_permission_mode(mode: &str) -> Option<&'static str> {
  1633. match mode.trim() {
  1634. "read-only" => Some("read-only"),
  1635. "workspace-write" => Some("workspace-write"),
  1636. "danger-full-access" => Some("danger-full-access"),
  1637. _ => None,
  1638. }
  1639. }
  1640. fn render_diff_report() -> Result<String, Box<dyn std::error::Error>> {
  1641. let output = std::process::Command::new("git")
  1642. .args(["diff", "--", ":(exclude).omx"])
  1643. .current_dir(env::current_dir()?)
  1644. .output()?;
  1645. if !output.status.success() {
  1646. let stderr = String::from_utf8_lossy(&output.stderr).trim().to_string();
  1647. return Err(format!("git diff failed: {stderr}").into());
  1648. }
  1649. let diff = String::from_utf8(output.stdout)?;
  1650. if diff.trim().is_empty() {
  1651. return Ok(
  1652. "Diff\n Result clean working tree\n Detail no current changes"
  1653. .to_string(),
  1654. );
  1655. }
  1656. Ok(format!("Diff\n\n{}", diff.trim_end()))
  1657. }
  1658. fn render_version_report() -> String {
  1659. let git_sha = GIT_SHA.unwrap_or("unknown");
  1660. let target = BUILD_TARGET.unwrap_or("unknown");
  1661. format!(
  1662. "Claw Code\n Version {VERSION}\n Git SHA {git_sha}\n Target {target}\n Build date {DEFAULT_DATE}"
  1663. )
  1664. }
  1665. fn render_export_text(session: &Session) -> String {
  1666. let mut lines = vec!["# Conversation Export".to_string(), String::new()];
  1667. for (index, message) in session.messages.iter().enumerate() {
  1668. let role = match message.role {
  1669. MessageRole::System => "system",
  1670. MessageRole::User => "user",
  1671. MessageRole::Assistant => "assistant",
  1672. MessageRole::Tool => "tool",
  1673. };
  1674. lines.push(format!("## {}. {role}", index + 1));
  1675. for block in &message.blocks {
  1676. match block {
  1677. ContentBlock::Text { text } => lines.push(text.clone()),
  1678. ContentBlock::ToolUse { id, name, input } => {
  1679. lines.push(format!("[tool_use id={id} name={name}] {input}"));
  1680. }
  1681. ContentBlock::ToolResult {
  1682. tool_use_id,
  1683. tool_name,
  1684. output,
  1685. is_error,
  1686. } => {
  1687. lines.push(format!(
  1688. "[tool_result id={tool_use_id} name={tool_name} error={is_error}] {output}"
  1689. ));
  1690. }
  1691. }
  1692. }
  1693. lines.push(String::new());
  1694. }
  1695. lines.join("\n")
  1696. }
  1697. fn default_export_filename(session: &Session) -> String {
  1698. let stem = session
  1699. .messages
  1700. .iter()
  1701. .find_map(|message| match message.role {
  1702. MessageRole::User => message.blocks.iter().find_map(|block| match block {
  1703. ContentBlock::Text { text } => Some(text.as_str()),
  1704. _ => None,
  1705. }),
  1706. _ => None,
  1707. })
  1708. .map_or("conversation", |text| {
  1709. text.lines().next().unwrap_or("conversation")
  1710. })
  1711. .chars()
  1712. .map(|ch| {
  1713. if ch.is_ascii_alphanumeric() {
  1714. ch.to_ascii_lowercase()
  1715. } else {
  1716. '-'
  1717. }
  1718. })
  1719. .collect::<String>()
  1720. .split('-')
  1721. .filter(|part| !part.is_empty())
  1722. .take(8)
  1723. .collect::<Vec<_>>()
  1724. .join("-");
  1725. let fallback = if stem.is_empty() {
  1726. "conversation"
  1727. } else {
  1728. &stem
  1729. };
  1730. format!("{fallback}.txt")
  1731. }
  1732. fn resolve_export_path(
  1733. requested_path: Option<&str>,
  1734. session: &Session,
  1735. ) -> Result<PathBuf, Box<dyn std::error::Error>> {
  1736. let cwd = env::current_dir()?;
  1737. let file_name =
  1738. requested_path.map_or_else(|| default_export_filename(session), ToOwned::to_owned);
  1739. let final_name = if Path::new(&file_name)
  1740. .extension()
  1741. .is_some_and(|ext| ext.eq_ignore_ascii_case("txt"))
  1742. {
  1743. file_name
  1744. } else {
  1745. format!("{file_name}.txt")
  1746. };
  1747. Ok(cwd.join(final_name))
  1748. }
  1749. fn build_system_prompt() -> Result<Vec<String>, Box<dyn std::error::Error>> {
  1750. Ok(load_system_prompt(
  1751. env::current_dir()?,
  1752. DEFAULT_DATE,
  1753. env::consts::OS,
  1754. "unknown",
  1755. )?)
  1756. }
  1757. fn build_runtime(
  1758. session: Session,
  1759. model: String,
  1760. system_prompt: Vec<String>,
  1761. enable_tools: bool,
  1762. allowed_tools: Option<AllowedToolSet>,
  1763. permission_mode: PermissionMode,
  1764. ) -> Result<ConversationRuntime<AnthropicRuntimeClient, CliToolExecutor>, Box<dyn std::error::Error>>
  1765. {
  1766. Ok(ConversationRuntime::new(
  1767. session,
  1768. AnthropicRuntimeClient::new(model, enable_tools, allowed_tools.clone())?,
  1769. CliToolExecutor::new(allowed_tools),
  1770. permission_policy(permission_mode),
  1771. system_prompt,
  1772. ))
  1773. }
  1774. struct CliPermissionPrompter {
  1775. current_mode: PermissionMode,
  1776. }
  1777. impl CliPermissionPrompter {
  1778. fn new(current_mode: PermissionMode) -> Self {
  1779. Self { current_mode }
  1780. }
  1781. }
  1782. impl runtime::PermissionPrompter for CliPermissionPrompter {
  1783. fn decide(
  1784. &mut self,
  1785. request: &runtime::PermissionRequest,
  1786. ) -> runtime::PermissionPromptDecision {
  1787. println!();
  1788. println!("Permission approval required");
  1789. println!(" Tool {}", request.tool_name);
  1790. println!(" Current mode {}", self.current_mode.as_str());
  1791. println!(" Required mode {}", request.required_mode.as_str());
  1792. println!(" Input {}", request.input);
  1793. print!("Approve this tool call? [y/N]: ");
  1794. let _ = io::stdout().flush();
  1795. let mut response = String::new();
  1796. match io::stdin().read_line(&mut response) {
  1797. Ok(_) => {
  1798. let normalized = response.trim().to_ascii_lowercase();
  1799. if matches!(normalized.as_str(), "y" | "yes") {
  1800. runtime::PermissionPromptDecision::Allow
  1801. } else {
  1802. runtime::PermissionPromptDecision::Deny {
  1803. reason: format!(
  1804. "tool '{}' denied by user approval prompt",
  1805. request.tool_name
  1806. ),
  1807. }
  1808. }
  1809. }
  1810. Err(error) => runtime::PermissionPromptDecision::Deny {
  1811. reason: format!("permission approval failed: {error}"),
  1812. },
  1813. }
  1814. }
  1815. }
  1816. struct AnthropicRuntimeClient {
  1817. runtime: tokio::runtime::Runtime,
  1818. client: AnthropicClient,
  1819. model: String,
  1820. enable_tools: bool,
  1821. allowed_tools: Option<AllowedToolSet>,
  1822. }
  1823. impl AnthropicRuntimeClient {
  1824. fn new(
  1825. model: String,
  1826. enable_tools: bool,
  1827. allowed_tools: Option<AllowedToolSet>,
  1828. ) -> Result<Self, Box<dyn std::error::Error>> {
  1829. Ok(Self {
  1830. runtime: tokio::runtime::Runtime::new()?,
  1831. client: AnthropicClient::from_auth(resolve_cli_auth_source()?)
  1832. .with_base_url(api::read_base_url()),
  1833. model,
  1834. enable_tools,
  1835. allowed_tools,
  1836. })
  1837. }
  1838. }
  1839. fn resolve_cli_auth_source() -> Result<AuthSource, Box<dyn std::error::Error>> {
  1840. Ok(resolve_startup_auth_source(|| {
  1841. let cwd = env::current_dir().map_err(api::ApiError::from)?;
  1842. let config = ConfigLoader::default_for(&cwd).load().map_err(|error| {
  1843. api::ApiError::Auth(format!("failed to load runtime OAuth config: {error}"))
  1844. })?;
  1845. Ok(config.oauth().cloned())
  1846. })?)
  1847. }
  1848. impl ApiClient for AnthropicRuntimeClient {
  1849. #[allow(clippy::too_many_lines)]
  1850. fn stream(&mut self, request: ApiRequest) -> Result<Vec<AssistantEvent>, RuntimeError> {
  1851. let message_request = MessageRequest {
  1852. model: self.model.clone(),
  1853. max_tokens: max_tokens_for_model(&self.model),
  1854. messages: convert_messages(&request.messages),
  1855. system: (!request.system_prompt.is_empty()).then(|| request.system_prompt.join("\n\n")),
  1856. tools: self.enable_tools.then(|| {
  1857. filter_tool_specs(self.allowed_tools.as_ref())
  1858. .into_iter()
  1859. .map(|spec| ToolDefinition {
  1860. name: spec.name.to_string(),
  1861. description: Some(spec.description.to_string()),
  1862. input_schema: spec.input_schema,
  1863. })
  1864. .collect()
  1865. }),
  1866. tool_choice: self.enable_tools.then_some(ToolChoice::Auto),
  1867. stream: true,
  1868. };
  1869. self.runtime.block_on(async {
  1870. let mut stream = self
  1871. .client
  1872. .stream_message(&message_request)
  1873. .await
  1874. .map_err(|error| RuntimeError::new(error.to_string()))?;
  1875. let mut stdout = io::stdout();
  1876. let mut events = Vec::new();
  1877. let mut pending_tool: Option<(String, String, String)> = None;
  1878. let mut saw_stop = false;
  1879. while let Some(event) = stream
  1880. .next_event()
  1881. .await
  1882. .map_err(|error| RuntimeError::new(error.to_string()))?
  1883. {
  1884. match event {
  1885. ApiStreamEvent::MessageStart(start) => {
  1886. for block in start.message.content {
  1887. push_output_block(block, &mut stdout, &mut events, &mut pending_tool)?;
  1888. }
  1889. }
  1890. ApiStreamEvent::ContentBlockStart(start) => {
  1891. push_output_block(
  1892. start.content_block,
  1893. &mut stdout,
  1894. &mut events,
  1895. &mut pending_tool,
  1896. )?;
  1897. }
  1898. ApiStreamEvent::ContentBlockDelta(delta) => match delta.delta {
  1899. ContentBlockDelta::TextDelta { text } => {
  1900. if !text.is_empty() {
  1901. write!(stdout, "{text}")
  1902. .and_then(|()| stdout.flush())
  1903. .map_err(|error| RuntimeError::new(error.to_string()))?;
  1904. events.push(AssistantEvent::TextDelta(text));
  1905. }
  1906. }
  1907. ContentBlockDelta::InputJsonDelta { partial_json } => {
  1908. if let Some((_, _, input)) = &mut pending_tool {
  1909. input.push_str(&partial_json);
  1910. }
  1911. }
  1912. },
  1913. ApiStreamEvent::ContentBlockStop(_) => {
  1914. if let Some((id, name, input)) = pending_tool.take() {
  1915. events.push(AssistantEvent::ToolUse { id, name, input });
  1916. }
  1917. }
  1918. ApiStreamEvent::MessageDelta(delta) => {
  1919. events.push(AssistantEvent::Usage(TokenUsage {
  1920. input_tokens: delta.usage.input_tokens,
  1921. output_tokens: delta.usage.output_tokens,
  1922. cache_creation_input_tokens: 0,
  1923. cache_read_input_tokens: 0,
  1924. }));
  1925. }
  1926. ApiStreamEvent::MessageStop(_) => {
  1927. saw_stop = true;
  1928. events.push(AssistantEvent::MessageStop);
  1929. }
  1930. }
  1931. }
  1932. if !saw_stop
  1933. && events.iter().any(|event| {
  1934. matches!(event, AssistantEvent::TextDelta(text) if !text.is_empty())
  1935. || matches!(event, AssistantEvent::ToolUse { .. })
  1936. })
  1937. {
  1938. events.push(AssistantEvent::MessageStop);
  1939. }
  1940. if events
  1941. .iter()
  1942. .any(|event| matches!(event, AssistantEvent::MessageStop))
  1943. {
  1944. return Ok(events);
  1945. }
  1946. let response = self
  1947. .client
  1948. .send_message(&MessageRequest {
  1949. stream: false,
  1950. ..message_request.clone()
  1951. })
  1952. .await
  1953. .map_err(|error| RuntimeError::new(error.to_string()))?;
  1954. response_to_events(response, &mut stdout)
  1955. })
  1956. }
  1957. }
  1958. fn slash_command_completion_candidates() -> Vec<String> {
  1959. slash_command_specs()
  1960. .iter()
  1961. .map(|spec| format!("/{}", spec.name))
  1962. .collect()
  1963. }
  1964. fn format_tool_call_start(name: &str, input: &str) -> String {
  1965. let parsed: serde_json::Value =
  1966. serde_json::from_str(input).unwrap_or(serde_json::Value::String(input.to_string()));
  1967. let detail = match name {
  1968. "bash" | "Bash" => parsed
  1969. .get("command")
  1970. .and_then(|v| v.as_str())
  1971. .map(|cmd| truncate_for_summary(cmd, 120))
  1972. .unwrap_or_default(),
  1973. "read_file" | "Read" => parsed
  1974. .get("file_path")
  1975. .or_else(|| parsed.get("path"))
  1976. .and_then(|v| v.as_str())
  1977. .unwrap_or("?")
  1978. .to_string(),
  1979. "write_file" | "Write" => {
  1980. let path = parsed
  1981. .get("file_path")
  1982. .or_else(|| parsed.get("path"))
  1983. .and_then(|v| v.as_str())
  1984. .unwrap_or("?");
  1985. let lines = parsed
  1986. .get("content")
  1987. .and_then(|v| v.as_str())
  1988. .map(|c| c.lines().count())
  1989. .unwrap_or(0);
  1990. format!("{path} ({lines} lines)")
  1991. }
  1992. "edit_file" | "Edit" => {
  1993. let path = parsed
  1994. .get("file_path")
  1995. .or_else(|| parsed.get("path"))
  1996. .and_then(|v| v.as_str())
  1997. .unwrap_or("?");
  1998. path.to_string()
  1999. }
  2000. "glob_search" | "Glob" => parsed
  2001. .get("pattern")
  2002. .and_then(|v| v.as_str())
  2003. .unwrap_or("?")
  2004. .to_string(),
  2005. "grep_search" | "Grep" => parsed
  2006. .get("pattern")
  2007. .and_then(|v| v.as_str())
  2008. .unwrap_or("?")
  2009. .to_string(),
  2010. "web_search" | "WebSearch" => parsed
  2011. .get("query")
  2012. .and_then(|v| v.as_str())
  2013. .unwrap_or("?")
  2014. .to_string(),
  2015. _ => summarize_tool_payload(input),
  2016. };
  2017. let border = "─".repeat(name.len() + 6);
  2018. format!(
  2019. "\x1b[38;5;245m╭─ \x1b[1;36m{name}\x1b[0;38;5;245m ─╮\x1b[0m\n\x1b[38;5;245m│\x1b[0m {detail}\n\x1b[38;5;245m╰{border}╯\x1b[0m"
  2020. )
  2021. }
  2022. fn format_tool_result(name: &str, output: &str, is_error: bool) -> String {
  2023. let icon = if is_error {
  2024. "\x1b[1;31m✗\x1b[0m"
  2025. } else {
  2026. "\x1b[1;32m✓\x1b[0m"
  2027. };
  2028. let summary = truncate_for_summary(output.trim(), 200);
  2029. format!("{icon} \x1b[38;5;245m{name}:\x1b[0m {summary}")
  2030. }
  2031. fn summarize_tool_payload(payload: &str) -> String {
  2032. let compact = match serde_json::from_str::<serde_json::Value>(payload) {
  2033. Ok(value) => value.to_string(),
  2034. Err(_) => payload.trim().to_string(),
  2035. };
  2036. truncate_for_summary(&compact, 96)
  2037. }
  2038. fn prettify_tool_payload(payload: &str) -> String {
  2039. match serde_json::from_str::<serde_json::Value>(payload) {
  2040. Ok(value) => serde_json::to_string_pretty(&value).unwrap_or_else(|_| payload.to_string()),
  2041. Err(_) => payload.to_string(),
  2042. }
  2043. }
  2044. fn truncate_for_summary(value: &str, limit: usize) -> String {
  2045. let mut chars = value.chars();
  2046. let truncated = chars.by_ref().take(limit).collect::<String>();
  2047. if chars.next().is_some() {
  2048. format!("{truncated}…")
  2049. } else {
  2050. truncated
  2051. }
  2052. }
  2053. fn push_output_block(
  2054. block: OutputContentBlock,
  2055. out: &mut impl Write,
  2056. events: &mut Vec<AssistantEvent>,
  2057. pending_tool: &mut Option<(String, String, String)>,
  2058. ) -> Result<(), RuntimeError> {
  2059. match block {
  2060. OutputContentBlock::Text { text } => {
  2061. if !text.is_empty() {
  2062. write!(out, "{text}")
  2063. .and_then(|()| out.flush())
  2064. .map_err(|error| RuntimeError::new(error.to_string()))?;
  2065. events.push(AssistantEvent::TextDelta(text));
  2066. }
  2067. }
  2068. OutputContentBlock::ToolUse { id, name, input } => {
  2069. writeln!(
  2070. out,
  2071. "
  2072. {}",
  2073. format_tool_call_start(&name, &input.to_string())
  2074. )
  2075. .and_then(|()| out.flush())
  2076. .map_err(|error| RuntimeError::new(error.to_string()))?;
  2077. *pending_tool = Some((id, name, input.to_string()));
  2078. }
  2079. }
  2080. Ok(())
  2081. }
  2082. fn response_to_events(
  2083. response: MessageResponse,
  2084. out: &mut impl Write,
  2085. ) -> Result<Vec<AssistantEvent>, RuntimeError> {
  2086. let mut events = Vec::new();
  2087. let mut pending_tool = None;
  2088. for block in response.content {
  2089. push_output_block(block, out, &mut events, &mut pending_tool)?;
  2090. if let Some((id, name, input)) = pending_tool.take() {
  2091. events.push(AssistantEvent::ToolUse { id, name, input });
  2092. }
  2093. }
  2094. events.push(AssistantEvent::Usage(TokenUsage {
  2095. input_tokens: response.usage.input_tokens,
  2096. output_tokens: response.usage.output_tokens,
  2097. cache_creation_input_tokens: response.usage.cache_creation_input_tokens,
  2098. cache_read_input_tokens: response.usage.cache_read_input_tokens,
  2099. }));
  2100. events.push(AssistantEvent::MessageStop);
  2101. Ok(events)
  2102. }
  2103. struct CliToolExecutor {
  2104. renderer: TerminalRenderer,
  2105. allowed_tools: Option<AllowedToolSet>,
  2106. }
  2107. impl CliToolExecutor {
  2108. fn new(allowed_tools: Option<AllowedToolSet>) -> Self {
  2109. Self {
  2110. renderer: TerminalRenderer::new(),
  2111. allowed_tools,
  2112. }
  2113. }
  2114. }
  2115. impl ToolExecutor for CliToolExecutor {
  2116. fn execute(&mut self, tool_name: &str, input: &str) -> Result<String, ToolError> {
  2117. if self
  2118. .allowed_tools
  2119. .as_ref()
  2120. .is_some_and(|allowed| !allowed.contains(tool_name))
  2121. {
  2122. return Err(ToolError::new(format!(
  2123. "tool `{tool_name}` is not enabled by the current --allowedTools setting"
  2124. )));
  2125. }
  2126. let value = serde_json::from_str(input)
  2127. .map_err(|error| ToolError::new(format!("invalid tool input JSON: {error}")))?;
  2128. match execute_tool(tool_name, &value) {
  2129. Ok(output) => {
  2130. let markdown = format_tool_result(tool_name, &output, false);
  2131. self.renderer
  2132. .stream_markdown(&markdown, &mut io::stdout())
  2133. .map_err(|error| ToolError::new(error.to_string()))?;
  2134. Ok(output)
  2135. }
  2136. Err(error) => {
  2137. let markdown = format_tool_result(tool_name, &error, true);
  2138. self.renderer
  2139. .stream_markdown(&markdown, &mut io::stdout())
  2140. .map_err(|stream_error| ToolError::new(stream_error.to_string()))?;
  2141. Err(ToolError::new(error))
  2142. }
  2143. }
  2144. }
  2145. }
  2146. fn permission_policy(mode: PermissionMode) -> PermissionPolicy {
  2147. tool_permission_specs()
  2148. .into_iter()
  2149. .fold(PermissionPolicy::new(mode), |policy, spec| {
  2150. policy.with_tool_requirement(spec.name, spec.required_permission)
  2151. })
  2152. }
  2153. fn tool_permission_specs() -> Vec<ToolSpec> {
  2154. mvp_tool_specs()
  2155. }
  2156. fn convert_messages(messages: &[ConversationMessage]) -> Vec<InputMessage> {
  2157. messages
  2158. .iter()
  2159. .filter_map(|message| {
  2160. let role = match message.role {
  2161. MessageRole::System | MessageRole::User | MessageRole::Tool => "user",
  2162. MessageRole::Assistant => "assistant",
  2163. };
  2164. let content = message
  2165. .blocks
  2166. .iter()
  2167. .map(|block| match block {
  2168. ContentBlock::Text { text } => InputContentBlock::Text { text: text.clone() },
  2169. ContentBlock::ToolUse { id, name, input } => InputContentBlock::ToolUse {
  2170. id: id.clone(),
  2171. name: name.clone(),
  2172. input: serde_json::from_str(input)
  2173. .unwrap_or_else(|_| serde_json::json!({ "raw": input })),
  2174. },
  2175. ContentBlock::ToolResult {
  2176. tool_use_id,
  2177. output,
  2178. is_error,
  2179. ..
  2180. } => InputContentBlock::ToolResult {
  2181. tool_use_id: tool_use_id.clone(),
  2182. content: vec![ToolResultContentBlock::Text {
  2183. text: output.clone(),
  2184. }],
  2185. is_error: *is_error,
  2186. },
  2187. })
  2188. .collect::<Vec<_>>();
  2189. (!content.is_empty()).then(|| InputMessage {
  2190. role: role.to_string(),
  2191. content,
  2192. })
  2193. })
  2194. .collect()
  2195. }
  2196. fn print_help_to(out: &mut impl Write) -> io::Result<()> {
  2197. writeln!(out, "claw v{VERSION}")?;
  2198. writeln!(out)?;
  2199. writeln!(out, "Usage:")?;
  2200. writeln!(
  2201. out,
  2202. " claw [--model MODEL] [--allowedTools TOOL[,TOOL...]]"
  2203. )?;
  2204. writeln!(out, " Start the interactive REPL")?;
  2205. writeln!(
  2206. out,
  2207. " claw [--model MODEL] [--output-format text|json] prompt TEXT"
  2208. )?;
  2209. writeln!(out, " Send one prompt and exit")?;
  2210. writeln!(
  2211. out,
  2212. " claw [--model MODEL] [--output-format text|json] TEXT"
  2213. )?;
  2214. writeln!(out, " Shorthand non-interactive prompt mode")?;
  2215. writeln!(
  2216. out,
  2217. " claw --resume SESSION.json [/status] [/compact] [...]"
  2218. )?;
  2219. writeln!(
  2220. out,
  2221. " Inspect or maintain a saved session without entering the REPL"
  2222. )?;
  2223. writeln!(out, " claw dump-manifests")?;
  2224. writeln!(out, " claw bootstrap-plan")?;
  2225. writeln!(out, " claw system-prompt [--cwd PATH] [--date YYYY-MM-DD]")?;
  2226. writeln!(out, " claw login")?;
  2227. writeln!(out, " claw logout")?;
  2228. writeln!(out, " claw init")?;
  2229. writeln!(out)?;
  2230. writeln!(out, "Flags:")?;
  2231. writeln!(
  2232. out,
  2233. " --model MODEL Override the active model"
  2234. )?;
  2235. writeln!(
  2236. out,
  2237. " --output-format FORMAT Non-interactive output format: text or json"
  2238. )?;
  2239. writeln!(
  2240. out,
  2241. " --permission-mode MODE Set read-only, workspace-write, or danger-full-access"
  2242. )?;
  2243. writeln!(out, " --dangerously-skip-permissions Skip all permission checks")?;
  2244. writeln!(out, " --allowedTools TOOLS Restrict enabled tools (repeatable; comma-separated aliases supported)")?;
  2245. writeln!(
  2246. out,
  2247. " --version, -V Print version and build information locally"
  2248. )?;
  2249. writeln!(out)?;
  2250. writeln!(out, "Interactive slash commands:")?;
  2251. writeln!(out, "{}", render_slash_command_help())?;
  2252. writeln!(out)?;
  2253. let resume_commands = resume_supported_slash_commands()
  2254. .into_iter()
  2255. .map(|spec| match spec.argument_hint {
  2256. Some(argument_hint) => format!("/{} {}", spec.name, argument_hint),
  2257. None => format!("/{}", spec.name),
  2258. })
  2259. .collect::<Vec<_>>()
  2260. .join(", ");
  2261. writeln!(out, "Resume-safe commands: {resume_commands}")?;
  2262. writeln!(out, "Examples:")?;
  2263. writeln!(out, " claw --model claude-opus \"summarize this repo\"")?;
  2264. writeln!(
  2265. out,
  2266. " claw --output-format json prompt \"explain src/main.rs\""
  2267. )?;
  2268. writeln!(
  2269. out,
  2270. " claw --allowedTools read,glob \"summarize Cargo.toml\""
  2271. )?;
  2272. writeln!(
  2273. out,
  2274. " claw --resume session.json /status /diff /export notes.txt"
  2275. )?;
  2276. writeln!(out, " claw login")?;
  2277. writeln!(out, " claw init")?;
  2278. Ok(())
  2279. }
  2280. fn print_help() {
  2281. let _ = print_help_to(&mut io::stdout());
  2282. }
  2283. #[cfg(test)]
  2284. mod tests {
  2285. use super::{
  2286. filter_tool_specs, format_compact_report, format_cost_report, format_model_report,
  2287. format_model_switch_report, format_permissions_report, format_permissions_switch_report,
  2288. format_resume_report, format_status_report, format_tool_call_start, format_tool_result,
  2289. normalize_permission_mode, parse_args, parse_git_status_metadata, print_help_to,
  2290. render_config_report, render_memory_report, render_repl_help, resolve_model_alias,
  2291. resume_supported_slash_commands, status_context, CliAction, CliOutputFormat, SlashCommand,
  2292. StatusUsage, DEFAULT_MODEL,
  2293. };
  2294. use runtime::{ContentBlock, ConversationMessage, MessageRole, PermissionMode};
  2295. use std::path::PathBuf;
  2296. #[test]
  2297. fn defaults_to_repl_when_no_args() {
  2298. assert_eq!(
  2299. parse_args(&[]).expect("args should parse"),
  2300. CliAction::Repl {
  2301. model: DEFAULT_MODEL.to_string(),
  2302. allowed_tools: None,
  2303. permission_mode: PermissionMode::WorkspaceWrite,
  2304. }
  2305. );
  2306. }
  2307. #[test]
  2308. fn parses_prompt_subcommand() {
  2309. let args = vec![
  2310. "prompt".to_string(),
  2311. "hello".to_string(),
  2312. "world".to_string(),
  2313. ];
  2314. assert_eq!(
  2315. parse_args(&args).expect("args should parse"),
  2316. CliAction::Prompt {
  2317. prompt: "hello world".to_string(),
  2318. model: DEFAULT_MODEL.to_string(),
  2319. output_format: CliOutputFormat::Text,
  2320. allowed_tools: None,
  2321. permission_mode: PermissionMode::WorkspaceWrite,
  2322. }
  2323. );
  2324. }
  2325. #[test]
  2326. fn parses_bare_prompt_and_json_output_flag() {
  2327. let args = vec![
  2328. "--output-format=json".to_string(),
  2329. "--model".to_string(),
  2330. "claude-opus".to_string(),
  2331. "explain".to_string(),
  2332. "this".to_string(),
  2333. ];
  2334. assert_eq!(
  2335. parse_args(&args).expect("args should parse"),
  2336. CliAction::Prompt {
  2337. prompt: "explain this".to_string(),
  2338. model: "claude-opus".to_string(),
  2339. output_format: CliOutputFormat::Json,
  2340. allowed_tools: None,
  2341. permission_mode: PermissionMode::WorkspaceWrite,
  2342. }
  2343. );
  2344. }
  2345. #[test]
  2346. fn resolves_model_aliases_in_args() {
  2347. let args = vec![
  2348. "--model".to_string(),
  2349. "opus".to_string(),
  2350. "explain".to_string(),
  2351. "this".to_string(),
  2352. ];
  2353. assert_eq!(
  2354. parse_args(&args).expect("args should parse"),
  2355. CliAction::Prompt {
  2356. prompt: "explain this".to_string(),
  2357. model: "claude-opus-4-6".to_string(),
  2358. output_format: CliOutputFormat::Text,
  2359. allowed_tools: None,
  2360. permission_mode: PermissionMode::WorkspaceWrite,
  2361. }
  2362. );
  2363. }
  2364. #[test]
  2365. fn resolves_known_model_aliases() {
  2366. assert_eq!(resolve_model_alias("opus"), "claude-opus-4-6");
  2367. assert_eq!(resolve_model_alias("sonnet"), "claude-sonnet-4-6");
  2368. assert_eq!(resolve_model_alias("haiku"), "claude-haiku-3-5-20241022");
  2369. assert_eq!(resolve_model_alias("claude-opus"), "claude-opus");
  2370. }
  2371. #[test]
  2372. fn parses_version_flags_without_initializing_prompt_mode() {
  2373. assert_eq!(
  2374. parse_args(&["--version".to_string()]).expect("args should parse"),
  2375. CliAction::Version
  2376. );
  2377. assert_eq!(
  2378. parse_args(&["-V".to_string()]).expect("args should parse"),
  2379. CliAction::Version
  2380. );
  2381. }
  2382. #[test]
  2383. fn parses_permission_mode_flag() {
  2384. let args = vec!["--permission-mode=read-only".to_string()];
  2385. assert_eq!(
  2386. parse_args(&args).expect("args should parse"),
  2387. CliAction::Repl {
  2388. model: DEFAULT_MODEL.to_string(),
  2389. allowed_tools: None,
  2390. permission_mode: PermissionMode::ReadOnly,
  2391. }
  2392. );
  2393. }
  2394. #[test]
  2395. fn parses_allowed_tools_flags_with_aliases_and_lists() {
  2396. let args = vec![
  2397. "--allowedTools".to_string(),
  2398. "read,glob".to_string(),
  2399. "--allowed-tools=write_file".to_string(),
  2400. ];
  2401. assert_eq!(
  2402. parse_args(&args).expect("args should parse"),
  2403. CliAction::Repl {
  2404. model: DEFAULT_MODEL.to_string(),
  2405. allowed_tools: Some(
  2406. ["glob_search", "read_file", "write_file"]
  2407. .into_iter()
  2408. .map(str::to_string)
  2409. .collect()
  2410. ),
  2411. permission_mode: PermissionMode::WorkspaceWrite,
  2412. }
  2413. );
  2414. }
  2415. #[test]
  2416. fn rejects_unknown_allowed_tools() {
  2417. let error = parse_args(&["--allowedTools".to_string(), "teleport".to_string()])
  2418. .expect_err("tool should be rejected");
  2419. assert!(error.contains("unsupported tool in --allowedTools: teleport"));
  2420. }
  2421. #[test]
  2422. fn parses_system_prompt_options() {
  2423. let args = vec![
  2424. "system-prompt".to_string(),
  2425. "--cwd".to_string(),
  2426. "/tmp/project".to_string(),
  2427. "--date".to_string(),
  2428. "2026-04-01".to_string(),
  2429. ];
  2430. assert_eq!(
  2431. parse_args(&args).expect("args should parse"),
  2432. CliAction::PrintSystemPrompt {
  2433. cwd: PathBuf::from("/tmp/project"),
  2434. date: "2026-04-01".to_string(),
  2435. }
  2436. );
  2437. }
  2438. #[test]
  2439. fn parses_login_and_logout_subcommands() {
  2440. assert_eq!(
  2441. parse_args(&["login".to_string()]).expect("login should parse"),
  2442. CliAction::Login
  2443. );
  2444. assert_eq!(
  2445. parse_args(&["logout".to_string()]).expect("logout should parse"),
  2446. CliAction::Logout
  2447. );
  2448. assert_eq!(
  2449. parse_args(&["init".to_string()]).expect("init should parse"),
  2450. CliAction::Init
  2451. );
  2452. }
  2453. #[test]
  2454. fn parses_resume_flag_with_slash_command() {
  2455. let args = vec![
  2456. "--resume".to_string(),
  2457. "session.json".to_string(),
  2458. "/compact".to_string(),
  2459. ];
  2460. assert_eq!(
  2461. parse_args(&args).expect("args should parse"),
  2462. CliAction::ResumeSession {
  2463. session_path: PathBuf::from("session.json"),
  2464. commands: vec!["/compact".to_string()],
  2465. }
  2466. );
  2467. }
  2468. #[test]
  2469. fn parses_resume_flag_with_multiple_slash_commands() {
  2470. let args = vec![
  2471. "--resume".to_string(),
  2472. "session.json".to_string(),
  2473. "/status".to_string(),
  2474. "/compact".to_string(),
  2475. "/cost".to_string(),
  2476. ];
  2477. assert_eq!(
  2478. parse_args(&args).expect("args should parse"),
  2479. CliAction::ResumeSession {
  2480. session_path: PathBuf::from("session.json"),
  2481. commands: vec![
  2482. "/status".to_string(),
  2483. "/compact".to_string(),
  2484. "/cost".to_string(),
  2485. ],
  2486. }
  2487. );
  2488. }
  2489. #[test]
  2490. fn filtered_tool_specs_respect_allowlist() {
  2491. let allowed = ["read_file", "grep_search"]
  2492. .into_iter()
  2493. .map(str::to_string)
  2494. .collect();
  2495. let filtered = filter_tool_specs(Some(&allowed));
  2496. let names = filtered
  2497. .into_iter()
  2498. .map(|spec| spec.name)
  2499. .collect::<Vec<_>>();
  2500. assert_eq!(names, vec!["read_file", "grep_search"]);
  2501. }
  2502. #[test]
  2503. fn shared_help_uses_resume_annotation_copy() {
  2504. let help = commands::render_slash_command_help();
  2505. assert!(help.contains("Slash commands"));
  2506. assert!(help.contains("works with --resume SESSION.json"));
  2507. }
  2508. #[test]
  2509. fn repl_help_includes_shared_commands_and_exit() {
  2510. let help = render_repl_help();
  2511. assert!(help.contains("REPL"));
  2512. assert!(help.contains("/help"));
  2513. assert!(help.contains("/status"));
  2514. assert!(help.contains("/model [model]"));
  2515. assert!(help.contains("/permissions [read-only|workspace-write|danger-full-access]"));
  2516. assert!(help.contains("/clear [--confirm]"));
  2517. assert!(help.contains("/cost"));
  2518. assert!(help.contains("/resume <session-path>"));
  2519. assert!(help.contains("/config [env|hooks|model]"));
  2520. assert!(help.contains("/memory"));
  2521. assert!(help.contains("/init"));
  2522. assert!(help.contains("/diff"));
  2523. assert!(help.contains("/version"));
  2524. assert!(help.contains("/export [file]"));
  2525. assert!(help.contains("/session [list|switch <session-id>]"));
  2526. assert!(help.contains("/exit"));
  2527. }
  2528. #[test]
  2529. fn resume_supported_command_list_matches_expected_surface() {
  2530. let names = resume_supported_slash_commands()
  2531. .into_iter()
  2532. .map(|spec| spec.name)
  2533. .collect::<Vec<_>>();
  2534. assert_eq!(
  2535. names,
  2536. vec![
  2537. "help", "status", "compact", "clear", "cost", "config", "memory", "init", "diff",
  2538. "version", "export",
  2539. ]
  2540. );
  2541. }
  2542. #[test]
  2543. fn resume_report_uses_sectioned_layout() {
  2544. let report = format_resume_report("session.json", 14, 6);
  2545. assert!(report.contains("Session resumed"));
  2546. assert!(report.contains("Session file session.json"));
  2547. assert!(report.contains("Messages 14"));
  2548. assert!(report.contains("Turns 6"));
  2549. }
  2550. #[test]
  2551. fn compact_report_uses_structured_output() {
  2552. let compacted = format_compact_report(8, 5, false);
  2553. assert!(compacted.contains("Compact"));
  2554. assert!(compacted.contains("Result compacted"));
  2555. assert!(compacted.contains("Messages removed 8"));
  2556. let skipped = format_compact_report(0, 3, true);
  2557. assert!(skipped.contains("Result skipped"));
  2558. }
  2559. #[test]
  2560. fn cost_report_uses_sectioned_layout() {
  2561. let report = format_cost_report(runtime::TokenUsage {
  2562. input_tokens: 20,
  2563. output_tokens: 8,
  2564. cache_creation_input_tokens: 3,
  2565. cache_read_input_tokens: 1,
  2566. });
  2567. assert!(report.contains("Cost"));
  2568. assert!(report.contains("Input tokens 20"));
  2569. assert!(report.contains("Output tokens 8"));
  2570. assert!(report.contains("Cache create 3"));
  2571. assert!(report.contains("Cache read 1"));
  2572. assert!(report.contains("Total tokens 32"));
  2573. }
  2574. #[test]
  2575. fn permissions_report_uses_sectioned_layout() {
  2576. let report = format_permissions_report("workspace-write");
  2577. assert!(report.contains("Permissions"));
  2578. assert!(report.contains("Active mode workspace-write"));
  2579. assert!(report.contains("Modes"));
  2580. assert!(report.contains("read-only ○ available Read/search tools only"));
  2581. assert!(report.contains("workspace-write ● current Edit files inside the workspace"));
  2582. assert!(report.contains("danger-full-access ○ available Unrestricted tool access"));
  2583. }
  2584. #[test]
  2585. fn permissions_switch_report_is_structured() {
  2586. let report = format_permissions_switch_report("read-only", "workspace-write");
  2587. assert!(report.contains("Permissions updated"));
  2588. assert!(report.contains("Result mode switched"));
  2589. assert!(report.contains("Previous mode read-only"));
  2590. assert!(report.contains("Active mode workspace-write"));
  2591. assert!(report.contains("Applies to subsequent tool calls"));
  2592. }
  2593. #[test]
  2594. fn init_help_mentions_direct_subcommand() {
  2595. let mut help = Vec::new();
  2596. print_help_to(&mut help).expect("help should render");
  2597. let help = String::from_utf8(help).expect("help should be utf8");
  2598. assert!(help.contains("claw init"));
  2599. }
  2600. #[test]
  2601. fn model_report_uses_sectioned_layout() {
  2602. let report = format_model_report("claude-sonnet", 12, 4);
  2603. assert!(report.contains("Model"));
  2604. assert!(report.contains("Current model claude-sonnet"));
  2605. assert!(report.contains("Session messages 12"));
  2606. assert!(report.contains("Switch models with /model <name>"));
  2607. }
  2608. #[test]
  2609. fn model_switch_report_preserves_context_summary() {
  2610. let report = format_model_switch_report("claude-sonnet", "claude-opus", 9);
  2611. assert!(report.contains("Model updated"));
  2612. assert!(report.contains("Previous claude-sonnet"));
  2613. assert!(report.contains("Current claude-opus"));
  2614. assert!(report.contains("Preserved msgs 9"));
  2615. }
  2616. #[test]
  2617. fn status_line_reports_model_and_token_totals() {
  2618. let status = format_status_report(
  2619. "claude-sonnet",
  2620. StatusUsage {
  2621. message_count: 7,
  2622. turns: 3,
  2623. latest: runtime::TokenUsage {
  2624. input_tokens: 5,
  2625. output_tokens: 4,
  2626. cache_creation_input_tokens: 1,
  2627. cache_read_input_tokens: 0,
  2628. },
  2629. cumulative: runtime::TokenUsage {
  2630. input_tokens: 20,
  2631. output_tokens: 8,
  2632. cache_creation_input_tokens: 2,
  2633. cache_read_input_tokens: 1,
  2634. },
  2635. estimated_tokens: 128,
  2636. },
  2637. "workspace-write",
  2638. &super::StatusContext {
  2639. cwd: PathBuf::from("/tmp/project"),
  2640. session_path: Some(PathBuf::from("session.json")),
  2641. loaded_config_files: 2,
  2642. discovered_config_files: 3,
  2643. memory_file_count: 4,
  2644. project_root: Some(PathBuf::from("/tmp")),
  2645. git_branch: Some("main".to_string()),
  2646. },
  2647. );
  2648. assert!(status.contains("Status"));
  2649. assert!(status.contains("Model claude-sonnet"));
  2650. assert!(status.contains("Permission mode workspace-write"));
  2651. assert!(status.contains("Messages 7"));
  2652. assert!(status.contains("Latest total 10"));
  2653. assert!(status.contains("Cumulative total 31"));
  2654. assert!(status.contains("Cwd /tmp/project"));
  2655. assert!(status.contains("Project root /tmp"));
  2656. assert!(status.contains("Git branch main"));
  2657. assert!(status.contains("Session session.json"));
  2658. assert!(status.contains("Config files loaded 2/3"));
  2659. assert!(status.contains("Memory files 4"));
  2660. }
  2661. #[test]
  2662. fn config_report_supports_section_views() {
  2663. let report = render_config_report(Some("env")).expect("config report should render");
  2664. assert!(report.contains("Merged section: env"));
  2665. }
  2666. #[test]
  2667. fn memory_report_uses_sectioned_layout() {
  2668. let report = render_memory_report().expect("memory report should render");
  2669. assert!(report.contains("Memory"));
  2670. assert!(report.contains("Working directory"));
  2671. assert!(report.contains("Instruction files"));
  2672. assert!(report.contains("Discovered files"));
  2673. }
  2674. #[test]
  2675. fn config_report_uses_sectioned_layout() {
  2676. let report = render_config_report(None).expect("config report should render");
  2677. assert!(report.contains("Config"));
  2678. assert!(report.contains("Discovered files"));
  2679. assert!(report.contains("Merged JSON"));
  2680. }
  2681. #[test]
  2682. fn parses_git_status_metadata() {
  2683. let (root, branch) = parse_git_status_metadata(Some(
  2684. "## rcc/cli...origin/rcc/cli
  2685. M src/main.rs",
  2686. ));
  2687. assert_eq!(branch.as_deref(), Some("rcc/cli"));
  2688. let _ = root;
  2689. }
  2690. #[test]
  2691. fn status_context_reads_real_workspace_metadata() {
  2692. let context = status_context(None).expect("status context should load");
  2693. assert!(context.cwd.is_absolute());
  2694. assert_eq!(context.discovered_config_files, 5);
  2695. assert!(context.loaded_config_files <= context.discovered_config_files);
  2696. }
  2697. #[test]
  2698. fn normalizes_supported_permission_modes() {
  2699. assert_eq!(normalize_permission_mode("read-only"), Some("read-only"));
  2700. assert_eq!(
  2701. normalize_permission_mode("workspace-write"),
  2702. Some("workspace-write")
  2703. );
  2704. assert_eq!(
  2705. normalize_permission_mode("danger-full-access"),
  2706. Some("danger-full-access")
  2707. );
  2708. assert_eq!(normalize_permission_mode("unknown"), None);
  2709. }
  2710. #[test]
  2711. fn clear_command_requires_explicit_confirmation_flag() {
  2712. assert_eq!(
  2713. SlashCommand::parse("/clear"),
  2714. Some(SlashCommand::Clear { confirm: false })
  2715. );
  2716. assert_eq!(
  2717. SlashCommand::parse("/clear --confirm"),
  2718. Some(SlashCommand::Clear { confirm: true })
  2719. );
  2720. }
  2721. #[test]
  2722. fn parses_resume_and_config_slash_commands() {
  2723. assert_eq!(
  2724. SlashCommand::parse("/resume saved-session.json"),
  2725. Some(SlashCommand::Resume {
  2726. session_path: Some("saved-session.json".to_string())
  2727. })
  2728. );
  2729. assert_eq!(
  2730. SlashCommand::parse("/clear --confirm"),
  2731. Some(SlashCommand::Clear { confirm: true })
  2732. );
  2733. assert_eq!(
  2734. SlashCommand::parse("/config"),
  2735. Some(SlashCommand::Config { section: None })
  2736. );
  2737. assert_eq!(
  2738. SlashCommand::parse("/config env"),
  2739. Some(SlashCommand::Config {
  2740. section: Some("env".to_string())
  2741. })
  2742. );
  2743. assert_eq!(SlashCommand::parse("/memory"), Some(SlashCommand::Memory));
  2744. assert_eq!(SlashCommand::parse("/init"), Some(SlashCommand::Init));
  2745. }
  2746. #[test]
  2747. fn init_template_mentions_detected_rust_workspace() {
  2748. let rendered = crate::init::render_init_claude_md(std::path::Path::new("."));
  2749. assert!(rendered.contains("# CLAUDE.md"));
  2750. assert!(rendered.contains("cargo clippy --workspace --all-targets -- -D warnings"));
  2751. }
  2752. #[test]
  2753. fn converts_tool_roundtrip_messages() {
  2754. let messages = vec![
  2755. ConversationMessage::user_text("hello"),
  2756. ConversationMessage::assistant(vec![ContentBlock::ToolUse {
  2757. id: "tool-1".to_string(),
  2758. name: "bash".to_string(),
  2759. input: "{\"command\":\"pwd\"}".to_string(),
  2760. }]),
  2761. ConversationMessage {
  2762. role: MessageRole::Tool,
  2763. blocks: vec![ContentBlock::ToolResult {
  2764. tool_use_id: "tool-1".to_string(),
  2765. tool_name: "bash".to_string(),
  2766. output: "ok".to_string(),
  2767. is_error: false,
  2768. }],
  2769. usage: None,
  2770. },
  2771. ];
  2772. let converted = super::convert_messages(&messages);
  2773. assert_eq!(converted.len(), 3);
  2774. assert_eq!(converted[1].role, "assistant");
  2775. assert_eq!(converted[2].role, "user");
  2776. }
  2777. #[test]
  2778. fn repl_help_mentions_history_completion_and_multiline() {
  2779. let help = render_repl_help();
  2780. assert!(help.contains("Up/Down"));
  2781. assert!(help.contains("Tab"));
  2782. assert!(help.contains("Shift+Enter/Ctrl+J"));
  2783. }
  2784. #[test]
  2785. fn tool_rendering_helpers_compact_output() {
  2786. let start = format_tool_call_start("read_file", r#"{"path":"src/main.rs"}"#);
  2787. assert!(start.contains("Tool call"));
  2788. assert!(start.contains("src/main.rs"));
  2789. let done = format_tool_result("read_file", r#"{"contents":"hello"}"#, false);
  2790. assert!(done.contains("Tool `read_file`"));
  2791. assert!(done.contains("contents"));
  2792. }
  2793. }