1use std::collections::BTreeMap;
2use std::sync::LazyLock;
3use std::sync::RwLock;
4
5const PERMISSION_MODES: &[&str] = &["safe", "auto", "yolo", "plan"];
6const PROMPT_MODES: &[&str] = &["default", "prepend", "append"];
7const OUTPUT_MODES: &[&str] = &[
8 "text",
9 "stream-text",
10 "json",
11 "stream-json",
12 "formatted",
13 "stream-formatted",
14];
15const KIMI_OUTPUT_MODES: &[&str] = &[
16 "text",
17 "stream-text",
18 "stream-json",
19 "formatted",
20 "stream-formatted",
21];
22const TEXT_OUTPUT_MODES: &[&str] = &["text", "stream-text"];
23
24#[derive(Clone, Debug)]
25pub struct RunnerInfo {
26 pub binary: String,
27 pub extra_args: Vec<String>,
28 pub no_persist_flags: Vec<String>,
29 pub thinking_flags: BTreeMap<i32, Vec<String>>,
30 pub show_thinking_flags: BTreeMap<bool, Vec<String>>,
31 pub yolo_flags: Vec<String>,
32 pub provider_flag: String,
33 pub model_flag: String,
34 pub agent_flag: String,
35 pub prompt_flag: String,
36}
37
38#[derive(Clone, Debug, Default)]
39pub struct ParsedArgs {
40 pub runner: Option<String>,
41 pub thinking: Option<i32>,
42 pub show_thinking: Option<bool>,
43 pub print_config: bool,
44 pub sanitize_osc: Option<bool>,
45 pub output_log_path: Option<bool>,
46 pub output_mode: Option<String>,
47 pub forward_unknown_json: bool,
48 pub save_session: bool,
49 pub cleanup_session: bool,
50 pub yolo: bool,
51 pub permission_mode: Option<String>,
52 pub provider: Option<String>,
53 pub model: Option<String>,
54 pub alias: Option<String>,
55 pub prompt: String,
56 pub prompt_supplied: bool,
57}
58
59#[derive(Clone, Debug)]
60pub struct AliasDef {
61 pub runner: Option<String>,
62 pub thinking: Option<i32>,
63 pub show_thinking: Option<bool>,
64 pub sanitize_osc: Option<bool>,
65 pub output_mode: Option<String>,
66 pub provider: Option<String>,
67 pub model: Option<String>,
68 pub agent: Option<String>,
69 pub prompt: Option<String>,
70 pub prompt_mode: Option<String>,
71}
72
73impl Default for AliasDef {
74 fn default() -> Self {
75 Self {
76 runner: None,
77 thinking: None,
78 show_thinking: None,
79 sanitize_osc: None,
80 output_mode: None,
81 provider: None,
82 model: None,
83 agent: None,
84 prompt: None,
85 prompt_mode: None,
86 }
87 }
88}
89
90#[derive(Clone, Debug)]
91pub struct CccConfig {
92 pub default_runner: String,
93 pub default_provider: String,
94 pub default_model: String,
95 pub default_thinking: Option<i32>,
96 pub default_show_thinking: bool,
97 pub default_sanitize_osc: Option<bool>,
98 pub default_output_mode: String,
99 pub aliases: BTreeMap<String, AliasDef>,
100 pub abbreviations: BTreeMap<String, String>,
101}
102
103impl Default for CccConfig {
104 fn default() -> Self {
105 Self {
106 default_runner: "oc".to_string(),
107 default_provider: String::new(),
108 default_model: String::new(),
109 default_thinking: Some(1),
110 default_show_thinking: true,
111 default_sanitize_osc: None,
112 default_output_mode: "text".to_string(),
113 aliases: BTreeMap::new(),
114 abbreviations: BTreeMap::new(),
115 }
116 }
117}
118
119#[derive(Clone, Debug, PartialEq, Eq)]
120pub struct OutputPlan {
121 pub runner_name: String,
122 pub mode: String,
123 pub stream: bool,
124 pub formatted: bool,
125 pub schema: Option<String>,
126 pub argv_flags: Vec<String>,
127 pub warnings: Vec<String>,
128}
129
130pub static RUNNER_REGISTRY: LazyLock<RwLock<BTreeMap<String, RunnerInfo>>> = LazyLock::new(|| {
131 let mut m = BTreeMap::new();
132 let opencode = RunnerInfo {
133 binary: "opencode".into(),
134 extra_args: vec!["run".into()],
135 no_persist_flags: vec![],
136 thinking_flags: BTreeMap::new(),
137 show_thinking_flags: {
138 let mut tf = BTreeMap::new();
139 tf.insert(true, vec!["--thinking".into()]);
140 tf
141 },
142 yolo_flags: vec![],
143 provider_flag: String::new(),
144 model_flag: String::new(),
145 agent_flag: "--agent".into(),
146 prompt_flag: String::new(),
147 };
148 let claude = RunnerInfo {
149 binary: "claude".into(),
150 extra_args: vec!["-p".into()],
151 no_persist_flags: vec!["--no-session-persistence".into()],
152 thinking_flags: {
153 let mut tf = BTreeMap::new();
154 tf.insert(0, vec!["--thinking".into(), "disabled".into()]);
155 tf.insert(
156 1,
157 vec![
158 "--thinking".into(),
159 "enabled".into(),
160 "--effort".into(),
161 "low".into(),
162 ],
163 );
164 tf.insert(
165 2,
166 vec![
167 "--thinking".into(),
168 "enabled".into(),
169 "--effort".into(),
170 "medium".into(),
171 ],
172 );
173 tf.insert(
174 3,
175 vec![
176 "--thinking".into(),
177 "enabled".into(),
178 "--effort".into(),
179 "high".into(),
180 ],
181 );
182 tf.insert(
183 4,
184 vec![
185 "--thinking".into(),
186 "enabled".into(),
187 "--effort".into(),
188 "max".into(),
189 ],
190 );
191 tf
192 },
193 show_thinking_flags: {
194 let mut tf = BTreeMap::new();
195 tf.insert(
196 true,
197 vec![
198 "--thinking".into(),
199 "enabled".into(),
200 "--effort".into(),
201 "low".into(),
202 ],
203 );
204 tf
205 },
206 yolo_flags: vec!["--dangerously-skip-permissions".into()],
207 provider_flag: String::new(),
208 model_flag: "--model".into(),
209 agent_flag: "--agent".into(),
210 prompt_flag: String::new(),
211 };
212 let kimi = RunnerInfo {
213 binary: "kimi".into(),
214 extra_args: vec![],
215 no_persist_flags: vec![],
216 thinking_flags: {
217 let mut tf = BTreeMap::new();
218 tf.insert(0, vec!["--no-thinking".into()]);
219 tf.insert(1, vec!["--thinking".into()]);
220 tf.insert(2, vec!["--thinking".into()]);
221 tf.insert(3, vec!["--thinking".into()]);
222 tf.insert(4, vec!["--thinking".into()]);
223 tf
224 },
225 show_thinking_flags: {
226 let mut tf = BTreeMap::new();
227 tf.insert(true, vec!["--thinking".into()]);
228 tf
229 },
230 yolo_flags: vec!["--yolo".into()],
231 provider_flag: String::new(),
232 model_flag: "--model".into(),
233 agent_flag: "--agent".into(),
234 prompt_flag: "--prompt".into(),
235 };
236 let codex = RunnerInfo {
237 binary: "codex".into(),
238 extra_args: vec!["exec".into()],
239 no_persist_flags: vec!["--ephemeral".into()],
240 thinking_flags: BTreeMap::new(),
241 show_thinking_flags: BTreeMap::new(),
242 yolo_flags: vec!["--dangerously-bypass-approvals-and-sandbox".into()],
243 provider_flag: String::new(),
244 model_flag: "--model".into(),
245 agent_flag: String::new(),
246 prompt_flag: String::new(),
247 };
248 let roocode = RunnerInfo {
249 binary: "roocode".into(),
250 extra_args: vec![],
251 no_persist_flags: vec![],
252 thinking_flags: BTreeMap::new(),
253 show_thinking_flags: BTreeMap::new(),
254 yolo_flags: vec![],
255 provider_flag: String::new(),
256 model_flag: String::new(),
257 agent_flag: String::new(),
258 prompt_flag: String::new(),
259 };
260 let crush = RunnerInfo {
261 binary: "crush".into(),
262 extra_args: vec!["run".into()],
263 no_persist_flags: vec![],
264 thinking_flags: BTreeMap::new(),
265 show_thinking_flags: BTreeMap::new(),
266 yolo_flags: vec![],
267 provider_flag: String::new(),
268 model_flag: String::new(),
269 agent_flag: String::new(),
270 prompt_flag: String::new(),
271 };
272 let cursor = RunnerInfo {
273 binary: "cursor-agent".into(),
274 extra_args: vec!["--print".into(), "--trust".into()],
275 no_persist_flags: vec![],
276 thinking_flags: BTreeMap::new(),
277 show_thinking_flags: BTreeMap::new(),
278 yolo_flags: vec!["--yolo".into()],
279 provider_flag: String::new(),
280 model_flag: "--model".into(),
281 agent_flag: String::new(),
282 prompt_flag: String::new(),
283 };
284 let gemini = RunnerInfo {
285 binary: "gemini".into(),
286 extra_args: vec![],
287 no_persist_flags: vec![],
288 thinking_flags: BTreeMap::new(),
289 show_thinking_flags: BTreeMap::new(),
290 yolo_flags: vec![],
291 provider_flag: String::new(),
292 model_flag: "--model".into(),
293 agent_flag: String::new(),
294 prompt_flag: "--prompt".into(),
295 };
296
297 let claude_clone = claude.clone();
298 let kimi_clone = kimi.clone();
299 let opencode_clone = opencode.clone();
300
301 let codex_clone = codex.clone();
302 let roocode_clone = roocode.clone();
303 let crush_clone = crush.clone();
304 let cursor_clone = cursor.clone();
305 let gemini_clone = gemini.clone();
306
307 m.insert("opencode".into(), opencode);
308 m.insert("claude".into(), claude);
309 m.insert("kimi".into(), kimi);
310 m.insert("codex".into(), codex);
311 m.insert("roocode".into(), roocode);
312 m.insert("crush".into(), crush);
313 m.insert("cursor".into(), cursor);
314 m.insert("gemini".into(), gemini);
315
316 m.insert("oc".into(), opencode_clone);
317 m.insert("cc".into(), claude_clone.clone());
318 m.insert("c".into(), codex_clone.clone());
319 m.insert("cx".into(), codex_clone);
320 m.insert("k".into(), kimi_clone);
321 m.insert("rc".into(), roocode_clone.clone());
322 m.insert("cr".into(), crush_clone);
323 m.insert("cu".into(), cursor_clone);
324 m.insert("g".into(), gemini_clone);
325
326 RwLock::new(m)
327});
328
329static RUNNER_SELECTOR_STRS: &[&str] = &[
330 "oc", "cc", "c", "cx", "k", "rc", "cr", "cu", "g", "codex", "claude", "opencode", "kimi",
331 "roocode", "crush", "cursor", "gemini",
332];
333
334fn is_runner_selector(s: &str) -> bool {
335 RUNNER_SELECTOR_STRS
336 .iter()
337 .any(|&sel| sel.eq_ignore_ascii_case(s))
338}
339
340fn parse_thinking(s: &str) -> Option<i32> {
341 let rest = s.strip_prefix('+')?;
342 match rest.to_ascii_lowercase().as_str() {
343 "0" | "none" => Some(0),
344 "1" | "low" => Some(1),
345 "2" | "med" | "mid" | "medium" => Some(2),
346 "3" | "high" => Some(3),
347 "4" | "max" | "xhigh" => Some(4),
348 _ => None,
349 }
350}
351
352fn parse_provider_model(s: &str) -> Option<(&str, &str)> {
353 let rest = s.strip_prefix(':')?;
354 let parts: Vec<&str> = rest.splitn(2, ':').collect();
355 if parts.len() == 2 {
356 Some((parts[0], parts[1]))
357 } else {
358 None
359 }
360}
361
362fn parse_model_only(s: &str) -> Option<&str> {
363 let rest = s.strip_prefix(':')?;
364 if rest.contains(':') {
365 None
366 } else {
367 Some(rest)
368 }
369}
370
371fn parse_alias(s: &str) -> Option<&str> {
372 let rest = s.strip_prefix('@')?;
373 if rest
374 .chars()
375 .all(|c| c.is_ascii_alphanumeric() || c == '_' || c == '-')
376 {
377 Some(rest)
378 } else {
379 None
380 }
381}
382
383fn parse_output_mode(s: &str) -> Option<String> {
384 let mode = s.to_ascii_lowercase();
385 if OUTPUT_MODES.iter().any(|known| *known == mode) {
386 Some(mode)
387 } else {
388 None
389 }
390}
391
392fn parse_output_mode_sugar(s: &str) -> Option<String> {
393 match s.to_ascii_lowercase().as_str() {
394 ".text" => Some("text".to_string()),
395 "..text" => Some("stream-text".to_string()),
396 ".json" => Some("json".to_string()),
397 "..json" => Some("stream-json".to_string()),
398 ".fmt" => Some("formatted".to_string()),
399 "..fmt" => Some("stream-formatted".to_string()),
400 _ => None,
401 }
402}
403
404pub fn parse_args(argv: &[String]) -> ParsedArgs {
405 let mut parsed = ParsedArgs::default();
406 let mut positional: Vec<String> = Vec::new();
407 let mut force_prompt = false;
408 let mut index = 0;
409
410 while index < argv.len() {
411 let token = &argv[index];
412 if force_prompt || !positional.is_empty() {
413 positional.push(token.clone());
414 } else if token == "--" {
415 force_prompt = true;
416 index += 1;
417 continue;
418 } else if is_runner_selector(token) {
419 parsed.runner = Some(token.to_lowercase());
420 } else if let Some(level) = parse_thinking(token) {
421 parsed.thinking = Some(level);
422 } else if token == "--show-thinking" || token == "--no-show-thinking" {
423 parsed.show_thinking = Some(token == "--show-thinking");
424 } else if token == "--print-config" {
425 parsed.print_config = true;
426 } else if token == "--sanitize-osc" || token == "--no-sanitize-osc" {
427 parsed.sanitize_osc = Some(token == "--sanitize-osc");
428 } else if token == "--output-log-path" || token == "--no-output-log-path" {
429 parsed.output_log_path = Some(token == "--output-log-path");
430 } else if token == "--output-mode" || token == "-o" {
431 if index + 1 >= argv.len() {
432 parsed.output_mode = Some(String::new());
433 } else {
434 parsed.output_mode = Some(argv[index + 1].to_ascii_lowercase());
435 index += 1;
436 }
437 } else if token == "--forward-unknown-json" {
438 parsed.forward_unknown_json = true;
439 } else if token == "--save-session" {
440 parsed.save_session = true;
441 } else if token == "--cleanup-session" {
442 parsed.cleanup_session = true;
443 } else if let Some(mode) = parse_output_mode_sugar(token) {
444 parsed.output_mode = Some(mode);
445 } else if token == "--yolo" || token == "-y" {
446 parsed.yolo = true;
447 parsed.permission_mode = Some("yolo".to_string());
448 } else if token == "--permission-mode" {
449 if index + 1 >= argv.len() {
450 parsed.permission_mode = Some(String::new());
451 } else {
452 let mode = argv[index + 1].to_lowercase();
453 parsed.yolo = mode == "yolo";
454 parsed.permission_mode = Some(mode);
455 index += 1;
456 }
457 } else if let Some((provider, model)) = parse_provider_model(token) {
458 parsed.provider = Some(provider.to_string());
459 parsed.model = Some(model.to_string());
460 } else if let Some(model) = parse_model_only(token) {
461 parsed.model = Some(model.to_string());
462 } else if let Some(alias_name) = parse_alias(token) {
463 parsed.alias = Some(alias_name.to_string());
464 } else {
465 positional.push(token.clone());
466 }
467 index += 1;
468 }
469
470 parsed.prompt = positional.join(" ");
471 parsed.prompt_supplied = !positional.is_empty();
472 parsed
473}
474
475pub fn resolve_command(
476 parsed: &ParsedArgs,
477 config: Option<&CccConfig>,
478) -> Result<(Vec<String>, BTreeMap<String, String>, Vec<String>), String> {
479 let config = config.cloned().unwrap_or_default();
480 let registry = RUNNER_REGISTRY.read().unwrap();
481 let mut warnings = Vec::new();
482
483 let (effective_runner_name, effective_runner, alias_def) =
484 resolve_effective_runner(parsed, &config, ®istry)
485 .ok_or_else(|| "no runner found".to_string())?;
486 if parsed.save_session && parsed.cleanup_session {
487 return Err("--save-session and --cleanup-session are mutually exclusive".to_string());
488 }
489
490 let mut argv: Vec<String> = vec![effective_runner.binary.clone()];
491 argv.extend(effective_runner.extra_args.iter().cloned());
492 warnings.extend(session_persistence_warnings(
493 parsed,
494 &effective_runner_name,
495 effective_runner,
496 ));
497 let output_plan = resolve_output_plan(parsed, Some(&config)).map_err(str::to_string)?;
498 warnings.extend(output_plan.warnings.clone());
499 argv.extend(output_plan.argv_flags.iter().cloned());
500
501 let effective_thinking = parsed
502 .thinking
503 .or_else(|| alias_def.and_then(|a| a.thinking))
504 .or(config.default_thinking);
505
506 let mut thinking_flags_applied = false;
507 if let Some(level) = effective_thinking {
508 if let Some(flags) = effective_runner.thinking_flags.get(&level) {
509 argv.extend(flags.iter().cloned());
510 thinking_flags_applied = true;
511 }
512 }
513
514 let effective_show_thinking = resolve_show_thinking(parsed, Some(&config));
515
516 if !thinking_flags_applied && effective_show_thinking {
517 if let Some(flags) = effective_runner.show_thinking_flags.get(&true) {
518 argv.extend(flags.iter().cloned());
519 }
520 }
521
522 let effective_provider: Option<String> = parsed
523 .provider
524 .clone()
525 .or_else(|| alias_def.and_then(|a| a.provider.clone()))
526 .or_else(|| {
527 if config.default_provider.is_empty() {
528 None
529 } else {
530 Some(config.default_provider.clone())
531 }
532 });
533
534 let effective_model: Option<String> = parsed
535 .model
536 .clone()
537 .or_else(|| alias_def.and_then(|a| a.model.clone()))
538 .or_else(|| {
539 if config.default_model.is_empty() {
540 None
541 } else {
542 Some(config.default_model.clone())
543 }
544 });
545
546 let mut env_overrides = BTreeMap::new();
547 if let Some(ref provider) = effective_provider {
548 env_overrides.insert("CCC_PROVIDER".to_string(), provider.clone());
549 }
550 if matches!(effective_runner_name.as_str(), "oc" | "opencode") {
551 env_overrides.insert(
552 "OPENCODE_DISABLE_TERMINAL_TITLE".to_string(),
553 "true".to_string(),
554 );
555 }
556
557 if let Some(ref model) = effective_model {
558 if !effective_runner.model_flag.is_empty() {
559 argv.push(effective_runner.model_flag.clone());
560 argv.push(model.clone());
561 }
562 }
563
564 let effective_agent = if let Some(alias_def) = alias_def {
565 alias_def.agent.clone()
566 } else if unresolved_alias_runner_name(parsed, &config, ®istry).is_some() {
567 None
568 } else {
569 parsed.alias.clone()
570 };
571
572 if let Some(agent) = effective_agent {
573 if effective_runner.agent_flag.is_empty() {
574 warnings.push(format!(
575 "warning: runner \"{}\" does not support agents; ignoring @{}",
576 effective_runner_name, agent
577 ));
578 } else {
579 argv.push(effective_runner.agent_flag.clone());
580 argv.push(agent);
581 }
582 }
583
584 let effective_permission_mode = parsed.permission_mode.clone().or_else(|| {
585 if parsed.yolo {
586 Some("yolo".to_string())
587 } else {
588 None
589 }
590 });
591 if let Some(ref mode) = effective_permission_mode {
592 if mode.is_empty() {
593 return Err("permission mode requires one of: safe, auto, yolo, plan".to_string());
594 }
595 if !PERMISSION_MODES.iter().any(|known| known == mode) {
596 return Err("permission mode must be one of: safe, auto, yolo, plan".to_string());
597 }
598 }
599
600 if matches!(effective_permission_mode.as_deref(), Some("safe")) {
601 if matches!(effective_runner_name.as_str(), "cc" | "claude") {
602 argv.push("--permission-mode".to_string());
603 argv.push("default".to_string());
604 } else if matches!(effective_runner_name.as_str(), "oc" | "opencode") {
605 env_overrides.insert(
606 "OPENCODE_CONFIG_CONTENT".to_string(),
607 "{\"permission\":\"ask\"}".to_string(),
608 );
609 } else if matches!(effective_runner_name.as_str(), "cu" | "cursor") {
610 argv.push("--sandbox".to_string());
611 argv.push("enabled".to_string());
612 } else if matches!(effective_runner_name.as_str(), "g" | "gemini") {
613 argv.push("--approval-mode".to_string());
614 argv.push("default".to_string());
615 argv.push("--sandbox".to_string());
616 } else if matches!(effective_runner_name.as_str(), "rc" | "roocode") {
617 warnings.push(
618 "warning: runner \"roocode\" safe mode is unverified; leaving default permissions unchanged"
619 .to_string(),
620 );
621 }
622 } else if matches!(effective_permission_mode.as_deref(), Some("auto")) {
623 if matches!(effective_runner_name.as_str(), "cc" | "claude") {
624 argv.push("--permission-mode".to_string());
625 argv.push("auto".to_string());
626 } else if matches!(effective_runner_name.as_str(), "c" | "cx" | "codex") {
627 argv.push("--full-auto".to_string());
628 } else if matches!(effective_runner_name.as_str(), "g" | "gemini") {
629 argv.push("--approval-mode".to_string());
630 argv.push("auto_edit".to_string());
631 } else {
632 warnings.push(format!(
633 "warning: runner \"{}\" does not support permission mode \"auto\"; ignoring it",
634 effective_runner_name
635 ));
636 }
637 } else if matches!(effective_permission_mode.as_deref(), Some("yolo")) {
638 if matches!(effective_runner_name.as_str(), "g" | "gemini") {
639 argv.push("--approval-mode".to_string());
640 argv.push("yolo".to_string());
641 } else if !effective_runner.yolo_flags.is_empty() {
642 argv.extend(effective_runner.yolo_flags.iter().cloned());
643 } else if matches!(effective_runner_name.as_str(), "oc" | "opencode") {
644 env_overrides.insert(
645 "OPENCODE_CONFIG_CONTENT".to_string(),
646 "{\"permission\":\"allow\"}".to_string(),
647 );
648 } else if matches!(effective_runner_name.as_str(), "cr" | "crush") {
649 warnings.push(
650 "warning: runner \"crush\" does not support yolo mode in non-interactive run mode; ignoring --yolo".to_string(),
651 );
652 } else if matches!(effective_runner_name.as_str(), "rc" | "roocode") {
653 warnings.push(
654 "warning: runner \"roocode\" yolo mode is unverified; ignoring --yolo".to_string(),
655 );
656 }
657 } else if matches!(effective_permission_mode.as_deref(), Some("plan")) {
658 if matches!(effective_runner_name.as_str(), "cc" | "claude") {
659 argv.push("--permission-mode".to_string());
660 argv.push("plan".to_string());
661 } else if matches!(effective_runner_name.as_str(), "k" | "kimi") {
662 argv.push("--plan".to_string());
663 } else if matches!(effective_runner_name.as_str(), "cu" | "cursor") {
664 argv.push("--mode".to_string());
665 argv.push("plan".to_string());
666 } else if matches!(effective_runner_name.as_str(), "g" | "gemini") {
667 argv.push("--approval-mode".to_string());
668 argv.push("plan".to_string());
669 } else {
670 warnings.push(format!(
671 "warning: runner \"{}\" does not support permission mode \"plan\"; ignoring it",
672 effective_runner_name
673 ));
674 }
675 }
676
677 if !parsed.save_session {
678 argv.extend(effective_runner.no_persist_flags.iter().cloned());
679 }
680
681 let prompt = resolve_prompt(parsed, alias_def)?;
682 if effective_runner.prompt_flag.is_empty() {
683 argv.push(prompt);
684 } else {
685 argv.push(effective_runner.prompt_flag.clone());
686 argv.push(prompt);
687 }
688
689 Ok((argv, env_overrides, warnings))
690}
691
692fn canonical_runner_name(effective_runner_name: &str, info: &RunnerInfo) -> String {
693 match effective_runner_name {
694 "oc" | "opencode" => "opencode".to_string(),
695 "cc" | "claude" => "claude".to_string(),
696 "c" | "cx" | "codex" => "codex".to_string(),
697 "k" | "kimi" => "kimi".to_string(),
698 "cr" | "crush" => "crush".to_string(),
699 "rc" | "roocode" => "roocode".to_string(),
700 "cu" | "cursor" => "cursor".to_string(),
701 "g" | "gemini" => "gemini".to_string(),
702 _ => info.binary.clone(),
703 }
704}
705
706fn session_persistence_warnings(
707 parsed: &ParsedArgs,
708 effective_runner_name: &str,
709 info: &RunnerInfo,
710) -> Vec<String> {
711 if parsed.save_session || !info.no_persist_flags.is_empty() {
712 return Vec::new();
713 }
714 let display = canonical_runner_name(effective_runner_name, info);
715 if parsed.cleanup_session {
716 if display == "opencode" || display == "kimi" {
717 return Vec::new();
718 }
719 return vec![format!(
720 "warning: runner \"{display}\" does not support automatic session cleanup; pass --save-session to allow saved sessions explicitly"
721 )];
722 }
723 Vec::new()
724}
725
726fn resolve_prompt(parsed: &ParsedArgs, alias_def: Option<&AliasDef>) -> Result<String, String> {
727 let user_prompt = parsed.prompt.trim();
728 let alias_prompt = alias_def
729 .and_then(|alias| alias.prompt.as_deref())
730 .map(str::trim)
731 .unwrap_or("");
732 let prompt_mode = alias_def
733 .and_then(|alias| alias.prompt_mode.as_deref())
734 .map(str::trim)
735 .filter(|mode| !mode.is_empty())
736 .unwrap_or("default");
737
738 if !PROMPT_MODES.iter().any(|known| *known == prompt_mode) {
739 return Err("prompt_mode must be one of: default, prepend, append".to_string());
740 }
741
742 if prompt_mode == "default" {
743 let prompt = if user_prompt.is_empty() {
744 alias_prompt
745 } else {
746 user_prompt
747 };
748 if prompt.is_empty() {
749 return Err("prompt must not be empty".to_string());
750 }
751 return Ok(prompt.to_string());
752 }
753
754 if !parsed.prompt_supplied {
755 return Err(format!(
756 "prompt_mode {prompt_mode} requires an explicit prompt argument"
757 ));
758 }
759 if alias_prompt.is_empty() {
760 let alias_name = parsed.alias.as_deref().unwrap_or("<alias>");
761 return Err(format!(
762 "prompt_mode {prompt_mode} requires aliases.{alias_name}.prompt"
763 ));
764 }
765
766 let mut parts = vec![alias_prompt, user_prompt];
767 if prompt_mode == "append" {
768 parts.reverse();
769 }
770 Ok(parts
771 .into_iter()
772 .filter(|part| !part.is_empty())
773 .collect::<Vec<_>>()
774 .join("\n"))
775}
776
777fn resolve_alias_def<'a>(parsed: &'a ParsedArgs, config: &'a CccConfig) -> Option<&'a AliasDef> {
778 parsed
779 .alias
780 .as_ref()
781 .and_then(|alias| config.aliases.get(alias))
782}
783
784fn unresolved_alias_runner_name(
785 parsed: &ParsedArgs,
786 config: &CccConfig,
787 registry: &BTreeMap<String, RunnerInfo>,
788) -> Option<String> {
789 if parsed.runner.is_some() {
790 return None;
791 }
792 let alias = parsed.alias.as_ref()?;
793 if config.aliases.contains_key(alias) {
794 return None;
795 }
796 let exact = resolve_runner_name(Some(alias), config);
797 if registry.contains_key(&exact) {
798 return Some(exact);
799 }
800 let lower_alias = alias.to_ascii_lowercase();
801 let lower = resolve_runner_name(Some(&lower_alias), config);
802 registry.contains_key(&lower).then_some(lower)
803}
804
805fn resolve_runner_name(parsed_runner: Option<&str>, config: &CccConfig) -> String {
806 let runner_name = parsed_runner.unwrap_or(&config.default_runner);
807 config
808 .abbreviations
809 .get(runner_name)
810 .cloned()
811 .unwrap_or_else(|| runner_name.to_string())
812}
813
814fn resolve_effective_runner<'a>(
815 parsed: &'a ParsedArgs,
816 config: &'a CccConfig,
817 registry: &'a BTreeMap<String, RunnerInfo>,
818) -> Option<(String, &'a RunnerInfo, Option<&'a AliasDef>)> {
819 let alias_def = resolve_alias_def(parsed, config);
820 let mut runner_name = resolve_runner_name(parsed.runner.as_deref(), config);
821 if parsed.runner.is_none() {
822 if let Some(alias_runner) = alias_def.and_then(|alias| alias.runner.as_deref()) {
823 runner_name = resolve_runner_name(Some(alias_runner), config);
824 } else if let Some(alias_runner) = unresolved_alias_runner_name(parsed, config, registry) {
825 runner_name = alias_runner;
826 }
827 }
828 let info = registry
829 .get(&runner_name)
830 .or_else(|| registry.get(&config.default_runner))
831 .or_else(|| registry.get("opencode"))?;
832 Some((runner_name, info, alias_def))
833}
834
835fn supported_output_modes(runner_name: &str) -> &'static [&'static str] {
836 match runner_name {
837 "cc" | "claude" | "oc" | "opencode" | "c" | "cx" | "codex" | "cu" | "cursor" => {
838 OUTPUT_MODES
839 }
840 "k" | "kimi" => KIMI_OUTPUT_MODES,
841 "g" | "gemini" => &[
842 "text",
843 "stream-text",
844 "json",
845 "stream-json",
846 "formatted",
847 "stream-formatted",
848 ],
849 _ => TEXT_OUTPUT_MODES,
850 }
851}
852
853fn fallback_output_mode(supported: &[&str]) -> &'static str {
854 if supported.iter().any(|mode| *mode == "text") {
855 "text"
856 } else {
857 "stream-text"
858 }
859}
860
861pub fn resolve_output_mode(
862 parsed: &ParsedArgs,
863 config: Option<&CccConfig>,
864) -> Result<String, &'static str> {
865 resolve_output_mode_with_source(parsed, config).map(|(mode, _)| mode)
866}
867
868fn resolve_output_mode_with_source(
869 parsed: &ParsedArgs,
870 config: Option<&CccConfig>,
871) -> Result<(String, &'static str), &'static str> {
872 let config = config.cloned().unwrap_or_default();
873 let alias_def = resolve_alias_def(parsed, &config);
874 let mut mode = parsed.output_mode.clone();
875 let mut source = "argument";
876 if mode.is_none() {
877 mode = alias_def.and_then(|alias| alias.output_mode.clone());
878 if mode.is_some() {
879 source = "alias";
880 }
881 }
882 let mode = mode.unwrap_or_else(|| {
883 source = "configured";
884 config.default_output_mode.clone()
885 });
886 if mode.is_empty() {
887 return Err(
888 "output mode requires one of: text, stream-text, json, stream-json, formatted, stream-formatted",
889 );
890 }
891 match parse_output_mode(&mode) {
892 Some(mode) => Ok((mode, source)),
893 None => Err(
894 "output mode must be one of: text, stream-text, json, stream-json, formatted, stream-formatted",
895 ),
896 }
897}
898
899pub fn resolve_show_thinking(parsed: &ParsedArgs, config: Option<&CccConfig>) -> bool {
900 let config = config.cloned().unwrap_or_default();
901 parsed
902 .show_thinking
903 .or_else(|| resolve_alias_def(parsed, &config).and_then(|alias| alias.show_thinking))
904 .unwrap_or(config.default_show_thinking)
905}
906
907pub fn resolve_sanitize_osc(parsed: &ParsedArgs, config: Option<&CccConfig>) -> bool {
908 let config = config.cloned().unwrap_or_default();
909 parsed
910 .sanitize_osc
911 .or_else(|| resolve_alias_def(parsed, &config).and_then(|alias| alias.sanitize_osc))
912 .or(config.default_sanitize_osc)
913 .unwrap_or_else(|| {
914 resolve_output_plan(parsed, Some(&config))
915 .map(|plan| plan.mode.contains("formatted"))
916 .unwrap_or(false)
917 })
918}
919
920pub fn resolve_output_plan(
921 parsed: &ParsedArgs,
922 config: Option<&CccConfig>,
923) -> Result<OutputPlan, &'static str> {
924 let config = config.cloned().unwrap_or_default();
925 let registry = RUNNER_REGISTRY.read().unwrap();
926 let (runner_name, info, _) =
927 resolve_effective_runner(parsed, &config, ®istry).ok_or("no runner found")?;
928 let (mut mode, mode_source) = resolve_output_mode_with_source(parsed, Some(&config))?;
929 let supported = supported_output_modes(&runner_name);
930 let mut warnings = Vec::new();
931 if !supported.iter().any(|candidate| *candidate == mode) {
932 if mode_source == "argument" {
933 return Err("gemini runner does not support requested output mode");
934 }
935 let fallback = fallback_output_mode(supported);
936 warnings.push(format!(
937 "warning: runner \"{}\" does not support {} output mode \"{}\"; falling back to \"{}\"",
938 canonical_runner_name(&runner_name, info),
939 mode_source,
940 mode,
941 fallback,
942 ));
943 mode = fallback.to_string();
944 }
945
946 if matches!(mode.as_str(), "text" | "stream-text") {
947 return Ok(OutputPlan {
948 runner_name,
949 mode: mode.clone(),
950 stream: mode.starts_with("stream-"),
951 formatted: false,
952 schema: None,
953 argv_flags: Vec::new(),
954 warnings,
955 });
956 }
957
958 if matches!(runner_name.as_str(), "cc" | "claude") {
959 let mut argv_flags = if mode == "json" {
960 vec!["--output-format".into(), "json".into()]
961 } else {
962 vec![
963 "--verbose".into(),
964 "--output-format".into(),
965 "stream-json".into(),
966 ]
967 };
968 if mode == "stream-formatted" {
969 argv_flags.push("--include-partial-messages".into());
970 }
971 return Ok(OutputPlan {
972 runner_name,
973 mode: mode.clone(),
974 stream: mode.starts_with("stream-"),
975 formatted: mode.contains("formatted"),
976 schema: Some("claude-code".into()),
977 argv_flags,
978 warnings,
979 });
980 }
981
982 if matches!(runner_name.as_str(), "k" | "kimi") {
983 return Ok(OutputPlan {
984 runner_name,
985 mode: mode.clone(),
986 stream: mode.starts_with("stream-"),
987 formatted: mode.contains("formatted"),
988 schema: Some("kimi".into()),
989 argv_flags: vec![
990 "--print".into(),
991 "--output-format".into(),
992 "stream-json".into(),
993 ],
994 warnings,
995 });
996 }
997
998 if matches!(runner_name.as_str(), "oc" | "opencode") {
999 return Ok(OutputPlan {
1000 runner_name,
1001 mode: mode.clone(),
1002 stream: mode.starts_with("stream-"),
1003 formatted: mode.contains("formatted"),
1004 schema: Some("opencode".into()),
1005 argv_flags: vec!["--format".into(), "json".into()],
1006 warnings,
1007 });
1008 }
1009
1010 if matches!(runner_name.as_str(), "c" | "cx" | "codex") {
1011 return Ok(OutputPlan {
1012 runner_name,
1013 mode: mode.clone(),
1014 stream: mode.starts_with("stream-"),
1015 formatted: mode.contains("formatted"),
1016 schema: Some("codex".into()),
1017 argv_flags: vec!["--json".into()],
1018 warnings,
1019 });
1020 }
1021
1022 if matches!(runner_name.as_str(), "cu" | "cursor") {
1023 let argv_flags = if mode == "json" {
1024 vec!["--output-format".into(), "json".into()]
1025 } else {
1026 vec!["--output-format".into(), "stream-json".into()]
1027 };
1028 return Ok(OutputPlan {
1029 runner_name,
1030 mode: mode.clone(),
1031 stream: mode.starts_with("stream-"),
1032 formatted: mode.contains("formatted"),
1033 schema: Some("cursor-agent".into()),
1034 argv_flags,
1035 warnings,
1036 });
1037 }
1038
1039 if matches!(runner_name.as_str(), "g" | "gemini") {
1040 let argv_flags = if mode == "json" {
1041 vec!["--output-format".into(), "json".into()]
1042 } else {
1043 vec!["--output-format".into(), "stream-json".into()]
1044 };
1045 return Ok(OutputPlan {
1046 runner_name,
1047 mode: mode.clone(),
1048 stream: mode.starts_with("stream-"),
1049 formatted: mode.contains("formatted"),
1050 schema: Some("gemini".into()),
1051 argv_flags,
1052 warnings,
1053 });
1054 }
1055
1056 Ok(OutputPlan {
1057 runner_name,
1058 mode: mode.clone(),
1059 stream: mode.starts_with("stream-"),
1060 formatted: false,
1061 schema: None,
1062 argv_flags: Vec::new(),
1063 warnings,
1064 })
1065}