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 fn current_stack_branches(branch: &str) -> Result<Vec<String>> {
317 let root = stack_root(branch)?;
318 let trunk = trunk_branch(&git::local_branches()?);
319 Ok(branch_and_descendants(&root)?
320 .into_iter()
321 .filter(|candidate| Some(candidate) != trunk.as_ref())
322 .collect())
323}
324
325pub fn publish_metadata(remote: &str) {
329 if let Err(error) = try_publish_metadata(remote) {
330 anstream::eprintln!(
331 "{}",
332 style::warn(&format!("could not publish stack metadata: {error:#}"))
333 );
334 }
335}
336
337fn try_publish_metadata(remote: &str) -> Result<()> {
338 let current = git::current_branch()?;
339 let trunk = trunk_branch(&git::local_branches()?);
340
341 let mut parents = serde_json::Map::new();
342 for branch in current_stack_branches(¤t)? {
343 if let Some(parent) = parent_of(&branch)? {
344 parents.insert(branch, Value::String(parent));
345 }
346 }
347 if parents.is_empty() {
348 return Ok(());
349 }
350
351 let document = json!({ "trunk": trunk, "parents": parents });
352 git::write_blob_ref(METADATA_REF, METADATA_FILE, &document.to_string())?;
353 git::push_ref(remote, METADATA_REF)
354}
355
356pub fn apply_remote_metadata(remote: &str) -> Result<usize> {
359 git::fetch_ref(remote, METADATA_REF)
360 .context("no stack metadata on the remote - push it from the other machine first")?;
361 let Some(content) = git::read_ref_file(METADATA_REF, METADATA_FILE)? else {
362 bail!("the remote stack metadata is empty");
363 };
364
365 let document: Value =
366 serde_json::from_str(&content).context("failed to parse remote stack metadata")?;
367 let parents = document
368 .get("parents")
369 .and_then(Value::as_object)
370 .context("remote stack metadata is malformed")?;
371
372 let mut pairs = Vec::new();
377 for (branch, parent) in parents {
378 let Some(parent) = parent.as_str() else {
379 continue;
380 };
381 if !is_safe_ref_name(branch) || !is_safe_ref_name(parent) {
382 anstream::eprintln!(
383 "{}",
384 style::warn(&format!(
385 "skipping unsafe stack metadata entry: {branch:?} -> {parent:?}"
386 ))
387 );
388 continue;
389 }
390 pairs.push((branch.clone(), parent.to_owned()));
391 }
392
393 let local: BTreeSet<String> = git::local_branches()?.into_iter().collect();
396 for (branch, _) in &pairs {
397 if !local.contains(branch) {
398 git::fetch_branch(remote, branch)
399 .with_context(|| format!("failed to fetch {branch} from {remote}"))?;
400 }
401 }
402
403 let mut attached = 0;
404 for (branch, parent) in &pairs {
405 set_parent(branch, parent)?;
406 record_base(branch, parent);
407 attached += 1;
408 anstream::println!(
409 "attached {} to {}",
410 style::branch(branch),
411 style::branch(parent)
412 );
413 }
414 Ok(attached)
415}
416
417fn is_safe_ref_name(name: &str) -> bool {
422 !name.is_empty()
423 && !name.starts_with('-')
424 && !name.chars().any(|c| c.is_whitespace() || c.is_control())
425}
426
427pub fn path_from_root(branch: &str) -> Result<Vec<String>> {
430 let trunk = trunk_branch(&git::local_branches()?);
431 let mut path = vec![branch.to_owned()];
432 let mut seen = BTreeSet::from([branch.to_owned()]);
433
434 let mut cursor = branch.to_owned();
435 while let Some(parent) = parent_of(&cursor)? {
436 if Some(&parent) == trunk.as_ref() || !seen.insert(parent.clone()) {
437 break;
438 }
439 path.push(parent.clone());
440 cursor = parent;
441 }
442
443 path.reverse();
444 Ok(path)
445}
446
447pub fn branch_parents(branches: &[String]) -> Result<Vec<(String, String)>> {
450 let mut pairs = Vec::new();
451 for branch in branches {
452 if let Some(parent) = parent_of(branch)? {
453 pairs.push((branch.clone(), parent));
454 }
455 }
456 Ok(pairs)
457}
458
459fn parent_map() -> Result<BTreeMap<String, String>> {
460 let mut parents = BTreeMap::new();
461 for branch in git::local_branches()? {
462 if let Some(parent) = parent_of(&branch)? {
463 parents.insert(branch, parent);
464 }
465 }
466 Ok(parents)
467}
468
469fn collect_descendants(
470 branch: &str,
471 children: &BTreeMap<String, Vec<String>>,
472 branches: &mut Vec<String>,
473 visited: &mut BTreeSet<String>,
474) {
475 if let Some(branch_children) = children.get(branch) {
476 for child in branch_children {
477 if !visited.insert(child.to_owned()) {
478 continue; }
480 branches.push(child.to_owned());
481 collect_descendants(child, children, branches, visited);
482 }
483 }
484}
485
486pub(crate) fn children_of(parent: &str) -> Result<Vec<String>> {
487 Ok(parent_map()?
488 .into_iter()
489 .filter_map(|(branch, branch_parent)| (branch_parent == parent).then_some(branch))
490 .collect())
491}
492
493fn children_map(parents: &BTreeMap<String, String>) -> BTreeMap<String, Vec<String>> {
494 let mut children: BTreeMap<String, Vec<String>> = BTreeMap::new();
495 for (branch, parent) in parents {
496 children
497 .entry(parent.to_owned())
498 .or_default()
499 .push(branch.to_owned());
500 }
501 children
502}
503
504fn root_for(branch: &str, parents: &BTreeMap<String, String>) -> String {
505 let mut root = branch.to_owned();
506 let mut seen = BTreeSet::new();
507
508 while let Some(parent) = parents.get(&root) {
509 if !seen.insert(root.clone()) {
510 break;
511 }
512 root = parent.to_owned();
513 }
514
515 root
516}
517
518pub(crate) fn parent_of(branch: &str) -> Result<Option<String>> {
519 git::config_get(&parent_key(branch))
520}
521
522pub(crate) fn base_of(branch: &str) -> Result<Option<String>> {
523 git::config_get(&base_key(branch))
524}
525
526pub(crate) fn set_parent(branch: &str, parent: &str) -> Result<()> {
527 git::config_set(&parent_key(branch), parent)
528}
529
530pub(crate) fn unset_parent(branch: &str) -> Result<()> {
531 git::config_unset(&parent_key(branch))
532}
533
534pub(crate) fn set_base(branch: &str, base: &str) -> Result<()> {
535 git::config_set(&base_key(branch), base)
536}
537
538pub(crate) fn unset_base(branch: &str) -> Result<()> {
539 git::config_unset(&base_key(branch))
540}
541
542fn parent_key(branch: &str) -> String {
543 format!("branch.{branch}.{PARENT_KEY}")
544}
545
546fn base_key(branch: &str) -> String {
547 format!("branch.{branch}.{BASE_KEY}")
548}
549
550fn renamed_from_key(branch: &str) -> String {
551 format!("branch.{branch}.{RENAMED_FROM_KEY}")
552}
553
554#[cfg(test)]
555mod tests {
556 use super::is_safe_ref_name;
557
558 #[test]
559 fn safe_ref_names_pass() {
560 assert!(is_safe_ref_name("main"));
561 assert!(is_safe_ref_name("feature/a"));
562 assert!(is_safe_ref_name("user/fix-123"));
563 }
564
565 #[test]
566 fn unsafe_ref_names_are_rejected() {
567 assert!(!is_safe_ref_name("--upload-pack=touch /tmp/pwned"));
569 assert!(!is_safe_ref_name("-x"));
570 assert!(!is_safe_ref_name("a branch"));
572 assert!(!is_safe_ref_name("a\nb"));
573 assert!(!is_safe_ref_name("a\tb"));
574 assert!(!is_safe_ref_name(""));
575 }
576}