-
-
Notifications
You must be signed in to change notification settings - Fork 344
/
Copy pathmerge.rs
113 lines (107 loc) · 4.02 KB
/
merge.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
use crate::OutputFormat;
use anyhow::{bail, Context};
use gix::bstr::BString;
use gix::bstr::ByteSlice;
use gix::merge::blob::builtin_driver::binary;
use gix::merge::blob::builtin_driver::text::Conflict;
use gix::merge::blob::pipeline::WorktreeRoots;
use gix::merge::blob::{Resolution, ResourceKind};
use gix::object::tree::EntryKind;
use gix::Id;
use std::path::Path;
pub fn file(
repo: gix::Repository,
out: &mut dyn std::io::Write,
format: OutputFormat,
conflict: Option<gix::merge::blob::builtin_driver::text::Conflict>,
base: BString,
ours: BString,
theirs: BString,
) -> anyhow::Result<()> {
if format != OutputFormat::Human {
bail!("JSON output isn't implemented yet");
}
let index = &repo.index_or_load_from_head()?;
let specs = repo.pathspec(
false,
[base, ours, theirs],
true,
index,
gix::worktree::stack::state::attributes::Source::WorktreeThenIdMapping.adjust_for_bare(repo.is_bare()),
)?;
// TODO: there should be a way to normalize paths without going through patterns, at least in this case maybe?
// `Search` actually sorts patterns by excluding or not, all that can lead to strange results.
let mut patterns = specs.search().patterns().map(|p| p.path().to_owned());
let base = patterns.next().unwrap();
let ours = patterns.next().unwrap();
let theirs = patterns.next().unwrap();
let base_id = repo.rev_parse_single(base.as_bstr()).ok();
let ours_id = repo.rev_parse_single(ours.as_bstr()).ok();
let theirs_id = repo.rev_parse_single(theirs.as_bstr()).ok();
let roots = worktree_roots(base_id, ours_id, theirs_id, repo.work_dir())?;
let mut cache = repo.merge_resource_cache(roots)?;
let null = repo.object_hash().null();
cache.set_resource(
base_id.map_or(null, Id::detach),
EntryKind::Blob,
base.as_bstr(),
ResourceKind::CommonAncestorOrBase,
&repo.objects,
)?;
cache.set_resource(
ours_id.map_or(null, Id::detach),
EntryKind::Blob,
ours.as_bstr(),
ResourceKind::CurrentOrOurs,
&repo.objects,
)?;
cache.set_resource(
theirs_id.map_or(null, Id::detach),
EntryKind::Blob,
theirs.as_bstr(),
ResourceKind::OtherOrTheirs,
&repo.objects,
)?;
let mut options = repo.blob_merge_options()?;
if let Some(conflict) = conflict {
options.text.conflict = conflict;
options.resolve_binary_with = match conflict {
Conflict::Keep { .. } => None,
Conflict::ResolveWithOurs => Some(binary::ResolveWith::Ours),
Conflict::ResolveWithTheirs => Some(binary::ResolveWith::Theirs),
Conflict::ResolveWithUnion => None,
};
}
let platform = cache.prepare_merge(&repo.objects, options)?;
let labels = gix::merge::blob::builtin_driver::text::Labels {
ancestor: Some(base.as_bstr()),
current: Some(ours.as_bstr()),
other: Some(theirs.as_bstr()),
};
let mut buf = repo.empty_reusable_buffer();
let (pick, resolution) = platform.merge(&mut buf, labels, repo.command_context()?)?;
let buf = platform.buffer_by_pick(pick).unwrap_or(&buf);
out.write_all(buf)?;
if resolution == Resolution::Conflict {
bail!("File conflicted")
}
Ok(())
}
fn worktree_roots(
base: Option<gix::Id<'_>>,
ours: Option<gix::Id<'_>>,
theirs: Option<gix::Id<'_>>,
workdir: Option<&Path>,
) -> anyhow::Result<gix::merge::blob::pipeline::WorktreeRoots> {
let roots = if base.is_none() || ours.is_none() || theirs.is_none() {
let workdir = workdir.context("A workdir is required if one of the bases are provided as path.")?;
gix::merge::blob::pipeline::WorktreeRoots {
current_root: ours.is_none().then(|| workdir.to_owned()),
other_root: theirs.is_none().then(|| workdir.to_owned()),
common_ancestor_root: base.is_none().then(|| workdir.to_owned()),
}
} else {
WorktreeRoots::default()
};
Ok(roots)
}