1use std::collections::{BTreeMap, BTreeSet};
6
7use anyhow::{Context, Result, bail};
8use serde_json::{Value, json};
9
10use crate::git;
11use crate::settings;
12use crate::style;
13
14const METADATA_REF: &str = "refs/stk/metadata";
17const METADATA_FILE: &str = "stack.json";
18
19mod nav;
20mod restack;
21mod snapshot;
22
23pub use nav::{
24 behind_parent_hint, checkout_bottom, checkout_child, checkout_parent, checkout_top,
25 print_all_stacks, print_children, print_parent, print_stack,
26};
27pub use restack::{abort_restack, continue_restack, restack};
28pub use snapshot::{take as snapshot, undo};
29
30const PARENT_KEY: &str = "stkParent";
31const BASE_KEY: &str = "stkBase";
32const RENAMED_FROM_KEY: &str = "stkRenamedFrom";
35
36pub fn create_branch(branch: &str, dry_run: bool) -> Result<()> {
37 let parent = git::current_branch()?;
38 if git::local_branches()?
40 .iter()
41 .any(|existing| existing == branch)
42 {
43 bail!(
44 "branch {branch} already exists - adopt it onto {parent} \
45 with `git stk adopt {branch} --parent {parent}`"
46 );
47 }
48 if !dry_run {
49 git::create_branch(branch)?;
50 set_parent(branch, &parent)?;
51 record_base(branch, &parent);
52 }
53 anstream::println!(
54 "{} {} with parent {}",
55 if dry_run { "would create" } else { "created" },
56 style::branch(branch),
57 style::branch(&parent)
58 );
59 Ok(())
60}
61
62pub fn insert_branch(branch: &str, dry_run: bool) -> Result<()> {
67 ensure_absent(branch)?;
68 let current = git::current_branch()?;
69 let children = children_of(¤t)?;
70
71 if !dry_run {
72 snapshot::take("new --insert");
73 git::create_branch(branch)?; set_parent(branch, ¤t)?;
75 record_base(branch, ¤t);
76 for child in &children {
77 set_parent(child, branch)?;
78 record_base(child, branch);
79 }
80 }
81
82 anstream::println!(
83 "{} {} above {}",
84 if dry_run { "would insert" } else { "inserted" },
85 style::branch(branch),
86 style::branch(¤t)
87 );
88 for child in &children {
89 anstream::println!(
90 "{} {} -> {}",
91 if dry_run {
92 "would retarget"
93 } else {
94 "retargeted"
95 },
96 style::branch(child),
97 style::branch(branch)
98 );
99 }
100 Ok(())
101}
102
103pub fn prepend_branch(branch: &str, dry_run: bool) -> Result<()> {
107 ensure_absent(branch)?;
108 let current = git::current_branch()?;
109 let parent =
110 parent_of(¤t)?.context("current branch has no stack parent to prepend below")?;
111 if !git::worktree_is_clean()? {
112 bail!(
113 "working tree has uncommitted changes; commit or stash before `git stk new --prepend`"
114 );
115 }
116
117 if !dry_run {
118 snapshot::take("new --prepend");
119 git::checkout(&parent)?;
120 git::create_branch(branch)?; set_parent(branch, &parent)?;
122 record_base(branch, &parent);
123 set_parent(¤t, branch)?;
124 record_base(¤t, branch);
125 }
126
127 anstream::println!(
128 "{} {} between {} and {}",
129 if dry_run { "would insert" } else { "inserted" },
130 style::branch(branch),
131 style::branch(&parent),
132 style::branch(¤t)
133 );
134 anstream::println!(
135 "{} {} -> {}",
136 if dry_run {
137 "would retarget"
138 } else {
139 "retargeted"
140 },
141 style::branch(¤t),
142 style::branch(branch)
143 );
144 Ok(())
145}
146
147fn ensure_absent(branch: &str) -> Result<()> {
148 if git::local_branches()?
149 .iter()
150 .any(|existing| existing == branch)
151 {
152 bail!("branch {branch} already exists");
153 }
154 Ok(())
155}
156
157pub fn trunk_branch(branches: &[String]) -> Option<String> {
160 let remote = settings::remote().unwrap_or_else(|_| settings::DEFAULT_REMOTE.to_owned());
161 if let Some(default) = git::remote_default_branch(&remote) {
162 return Some(default);
163 }
164
165 ["main", "master"]
166 .iter()
167 .find(|name| branches.iter().any(|branch| branch == *name))
168 .map(|name| (*name).to_owned())
169}
170
171pub fn adopt_branch(branch: &str, parent: &str, dry_run: bool) -> Result<()> {
172 if branch == parent {
173 bail!("a branch cannot be its own stack parent");
174 }
175
176 let branches: BTreeSet<_> = git::local_branches()?.into_iter().collect();
177 if !branches.contains(branch) {
178 bail!("branch {branch} does not exist");
179 }
180 if !branches.contains(parent) {
181 bail!("parent branch {parent} does not exist");
182 }
183 if branch_and_descendants(branch)?
184 .iter()
185 .any(|descendant| descendant == parent)
186 {
187 bail!("{parent} is already below {branch} in the stack; that would form a cycle");
188 }
189
190 if !dry_run {
191 set_parent(branch, parent)?;
192 record_base(branch, parent);
193 }
194 anstream::println!(
195 "{} {} to {}",
196 if dry_run { "would attach" } else { "attached" },
197 style::branch(branch),
198 style::branch(parent)
199 );
200 Ok(())
201}
202
203pub fn detach_branch(branch: Option<&str>) -> Result<()> {
204 let branch = branch
205 .map(str::to_owned)
206 .map_or_else(git::current_branch, Ok)?;
207 unset_parent(&branch)?;
208 unset_base(&branch)?;
209 anstream::println!("detached {}", style::branch(&branch));
210 Ok(())
211}
212
213pub fn rename_branch(old: &str, new: &str, dry_run: bool) -> Result<()> {
217 let children = children_of(old)?;
218
219 if !dry_run {
220 snapshot::take("rename");
221 git::rename_branch(old, new)?;
222 }
223 anstream::println!(
224 "{} {} -> {}",
225 if dry_run { "would rename" } else { "renamed" },
226 style::branch(old),
227 style::branch(new)
228 );
229
230 for child in &children {
231 if !dry_run {
232 set_parent(child, new)?;
233 }
234 anstream::println!(
235 "{} {} -> {}",
236 if dry_run {
237 "would retarget"
238 } else {
239 "retargeted"
240 },
241 style::branch(child),
242 style::branch(new)
243 );
244 }
245 Ok(())
246}
247
248pub fn set_renamed_from(branch: &str, old: &str) -> Result<()> {
251 git::config_set(&renamed_from_key(branch), old)
252}
253
254pub fn renamed_from(branch: &str) -> Result<Option<String>> {
256 git::config_get(&renamed_from_key(branch))
257}
258
259pub fn clear_renamed_from(branch: &str) -> Result<()> {
261 git::config_unset(&renamed_from_key(branch))
262}
263
264pub fn record_base(branch: &str, parent: &str) {
267 if let Ok(base) = git::merge_base(parent, branch) {
268 let _ = git::config_set(&base_key(branch), &base);
269 }
270}
271
272pub fn stack_root(branch: &str) -> Result<String> {
274 let parents = parent_map()?;
275 Ok(root_for(branch, &parents))
276}
277
278pub fn branch_and_descendants(branch: &str) -> Result<Vec<String>> {
279 let parents = parent_map()?;
280 let children = children_map(&parents);
281 let mut branches = vec![branch.to_owned()];
282 let mut visited = BTreeSet::from([branch.to_owned()]);
283 collect_descendants(branch, &children, &mut branches, &mut visited);
284 Ok(branches)
285}
286
287pub fn stack_line(branch: &str) -> Result<Vec<String>> {
293 let trunk = trunk_branch(&git::local_branches()?);
297 if Some(branch) == trunk.as_deref() {
298 return Ok(Vec::new());
299 }
300
301 let mut line = path_from_root(branch)?; let above = branch_and_descendants(branch)?; line.extend(above.into_iter().skip(1)); line.retain(|candidate| Some(candidate) != trunk.as_ref());
308 Ok(line)
309}
310
311pub(crate) fn line_base(branch: &str) -> Result<String> {
319 Ok(path_from_root(branch)?
320 .into_iter()
321 .next()
322 .unwrap_or_else(|| branch.to_owned()))
323}
324
325pub fn current_stack_branches(branch: &str) -> Result<Vec<String>> {
333 let base = line_base(branch)?;
334 let trunk = trunk_branch(&git::local_branches()?);
335 Ok(branch_and_descendants(&base)?
336 .into_iter()
337 .filter(|candidate| Some(candidate) != trunk.as_ref())
338 .collect())
339}
340
341pub fn publish_metadata(remote: &str) {
345 if let Err(error) = try_publish_metadata(remote) {
346 anstream::eprintln!(
347 "{}",
348 style::warn(&format!("could not publish stack metadata: {error:#}"))
349 );
350 }
351}
352
353fn try_publish_metadata(remote: &str) -> Result<()> {
354 let current = git::current_branch()?;
355 let trunk = trunk_branch(&git::local_branches()?);
356
357 let mut parents = serde_json::Map::new();
358 for branch in current_stack_branches(¤t)? {
359 if let Some(parent) = parent_of(&branch)? {
360 parents.insert(branch, Value::String(parent));
361 }
362 }
363 if parents.is_empty() {
364 return Ok(());
365 }
366
367 let document = json!({ "trunk": trunk, "parents": parents });
368 git::write_blob_ref(METADATA_REF, METADATA_FILE, &document.to_string())?;
369 git::push_ref(remote, METADATA_REF)
370}
371
372pub fn apply_remote_metadata(remote: &str) -> Result<usize> {
375 git::fetch_ref(remote, METADATA_REF)
376 .context("no stack metadata on the remote - push it from the other machine first")?;
377 let Some(content) = git::read_ref_file(METADATA_REF, METADATA_FILE)? else {
378 bail!("the remote stack metadata is empty");
379 };
380
381 let document: Value =
382 serde_json::from_str(&content).context("failed to parse remote stack metadata")?;
383 let parents = document
384 .get("parents")
385 .and_then(Value::as_object)
386 .context("remote stack metadata is malformed")?;
387
388 let mut pairs = Vec::new();
393 for (branch, parent) in parents {
394 let Some(parent) = parent.as_str() else {
395 continue;
396 };
397 if !is_safe_ref_name(branch) || !is_safe_ref_name(parent) {
398 anstream::eprintln!(
399 "{}",
400 style::warn(&format!(
401 "skipping unsafe stack metadata entry: {branch:?} -> {parent:?}"
402 ))
403 );
404 continue;
405 }
406 pairs.push((branch.clone(), parent.to_owned()));
407 }
408
409 let local: BTreeSet<String> = git::local_branches()?.into_iter().collect();
412 for (branch, _) in &pairs {
413 if !local.contains(branch) {
414 git::fetch_branch(remote, branch)
415 .with_context(|| format!("failed to fetch {branch} from {remote}"))?;
416 }
417 }
418
419 let mut attached = 0;
420 for (branch, parent) in &pairs {
421 set_parent(branch, parent)?;
422 record_base(branch, parent);
423 attached += 1;
424 anstream::println!(
425 "attached {} to {}",
426 style::branch(branch),
427 style::branch(parent)
428 );
429 }
430 Ok(attached)
431}
432
433pub(crate) fn is_safe_ref_name(name: &str) -> bool {
438 !name.is_empty()
439 && !name.starts_with('-')
440 && !name.chars().any(|c| c.is_whitespace() || c.is_control())
441}
442
443pub fn path_from_root(branch: &str) -> Result<Vec<String>> {
446 let trunk = trunk_branch(&git::local_branches()?);
447 let mut path = vec![branch.to_owned()];
448 let mut seen = BTreeSet::from([branch.to_owned()]);
449
450 let mut cursor = branch.to_owned();
451 while let Some(parent) = parent_of(&cursor)? {
452 if Some(&parent) == trunk.as_ref() || !seen.insert(parent.clone()) {
453 break;
454 }
455 path.push(parent.clone());
456 cursor = parent;
457 }
458
459 path.reverse();
460 Ok(path)
461}
462
463pub fn branch_parents(branches: &[String]) -> Result<Vec<(String, String)>> {
466 let mut pairs = Vec::new();
467 for branch in branches {
468 if let Some(parent) = parent_of(branch)? {
469 pairs.push((branch.clone(), parent));
470 }
471 }
472 Ok(pairs)
473}
474
475fn parent_map() -> Result<BTreeMap<String, String>> {
476 let mut parents = BTreeMap::new();
477 for branch in git::local_branches()? {
478 if let Some(parent) = parent_of(&branch)? {
479 parents.insert(branch, parent);
480 }
481 }
482 Ok(parents)
483}
484
485fn collect_descendants(
486 branch: &str,
487 children: &BTreeMap<String, Vec<String>>,
488 branches: &mut Vec<String>,
489 visited: &mut BTreeSet<String>,
490) {
491 if let Some(branch_children) = children.get(branch) {
492 for child in branch_children {
493 if !visited.insert(child.to_owned()) {
494 continue; }
496 branches.push(child.to_owned());
497 collect_descendants(child, children, branches, visited);
498 }
499 }
500}
501
502pub(crate) fn children_of(parent: &str) -> Result<Vec<String>> {
503 Ok(parent_map()?
504 .into_iter()
505 .filter_map(|(branch, branch_parent)| (branch_parent == parent).then_some(branch))
506 .collect())
507}
508
509fn children_map(parents: &BTreeMap<String, String>) -> BTreeMap<String, Vec<String>> {
510 let mut children: BTreeMap<String, Vec<String>> = BTreeMap::new();
511 for (branch, parent) in parents {
512 children
513 .entry(parent.to_owned())
514 .or_default()
515 .push(branch.to_owned());
516 }
517 children
518}
519
520fn root_for(branch: &str, parents: &BTreeMap<String, String>) -> String {
521 let mut root = branch.to_owned();
522 let mut seen = BTreeSet::new();
523
524 while let Some(parent) = parents.get(&root) {
525 if !seen.insert(root.clone()) {
526 break;
527 }
528 root = parent.to_owned();
529 }
530
531 root
532}
533
534pub(crate) fn parent_of(branch: &str) -> Result<Option<String>> {
535 git::config_get(&parent_key(branch))
536}
537
538pub(crate) fn base_of(branch: &str) -> Result<Option<String>> {
539 git::config_get(&base_key(branch))
540}
541
542pub(crate) fn set_parent(branch: &str, parent: &str) -> Result<()> {
543 git::config_set(&parent_key(branch), parent)
544}
545
546pub(crate) fn unset_parent(branch: &str) -> Result<()> {
547 git::config_unset(&parent_key(branch))
548}
549
550pub(crate) fn set_base(branch: &str, base: &str) -> Result<()> {
551 git::config_set(&base_key(branch), base)
552}
553
554pub(crate) fn unset_base(branch: &str) -> Result<()> {
555 git::config_unset(&base_key(branch))
556}
557
558fn parent_key(branch: &str) -> String {
559 format!("branch.{branch}.{PARENT_KEY}")
560}
561
562fn base_key(branch: &str) -> String {
563 format!("branch.{branch}.{BASE_KEY}")
564}
565
566fn renamed_from_key(branch: &str) -> String {
567 format!("branch.{branch}.{RENAMED_FROM_KEY}")
568}
569
570#[cfg(test)]
571mod tests {
572 use super::is_safe_ref_name;
573
574 #[test]
575 fn safe_ref_names_pass() {
576 assert!(is_safe_ref_name("main"));
577 assert!(is_safe_ref_name("feature/a"));
578 assert!(is_safe_ref_name("user/fix-123"));
579 }
580
581 #[test]
582 fn unsafe_ref_names_are_rejected() {
583 assert!(!is_safe_ref_name("--upload-pack=touch /tmp/pwned"));
585 assert!(!is_safe_ref_name("-x"));
586 assert!(!is_safe_ref_name("a branch"));
588 assert!(!is_safe_ref_name("a\nb"));
589 assert!(!is_safe_ref_name("a\tb"));
590 assert!(!is_safe_ref_name(""));
591 }
592}