gix_merge/blob/platform/
mod.rs

1use crate::blob::{pipeline, BuiltinDriver, Pipeline, Platform};
2use bstr::{BStr, BString};
3use gix_filter::attributes;
4
5/// A stored value representing a resource that participates in a merge.
6#[derive(Clone, Eq, PartialEq, Ord, PartialOrd, Debug)]
7pub(super) struct Resource {
8    /// The `id` of the value, or `null` if it's only living in a worktree.
9    id: gix_hash::ObjectId,
10    /// The repository-relative path where the resource lives in the tree.
11    rela_path: BString,
12    /// The outcome of converting a resource into a mergable format using [Pipeline::convert_to_mergeable()].
13    data: Option<pipeline::Data>,
14    /// The kind of the resource we are looking at. Only possible values are `Blob` and `BlobExecutable`.
15    mode: gix_object::tree::EntryKind,
16    /// A possibly empty buffer, depending on `conversion.data` which may indicate the data is considered binary
17    /// or the resource doesn't exist.
18    buffer: Vec<u8>,
19}
20
21/// A blob or executable ready to be merged in one way or another.
22#[derive(Copy, Clone, Ord, PartialOrd, Eq, PartialEq, Hash)]
23pub struct ResourceRef<'a> {
24    /// The data itself, suitable for merging, and if the object or worktree item is present at all.
25    pub data: resource::Data<'a>,
26    /// The location of the resource, relative to the working tree.
27    pub rela_path: &'a BStr,
28    /// The id of the content as it would be stored in `git`, or `null` if the content doesn't exist anymore at
29    /// `rela_path` or if it was never computed. This can happen with content read from the worktree, which
30    /// after its 'to-git' conversion never had its hash computed.
31    pub id: &'a gix_hash::oid,
32}
33
34/// Options for use in [`Platform::new()`].
35#[derive(Default, Clone, PartialEq, Eq, Debug, Hash, Ord, PartialOrd)]
36pub struct Options {
37    /// Define which driver to use by name if the `merge` attribute for a resource is unspecified.
38    ///
39    /// This is the value of the `merge.default` git configuration.
40    pub default_driver: Option<BString>,
41}
42
43/// The selection of the driver to use by a resource obtained with [`Platform::prepare_merge()`].
44///
45/// If available, an index into the `drivers` field to access more diff-related information of the driver for items
46/// at the given path, as previously determined by git-attributes.
47///
48/// * `merge` is set
49///     - Use the [`BuiltinDriver::Text`]
50/// * `-merge` is unset
51///     - Use the [`BuiltinDriver::Binary`]
52/// * `!merge` is unspecified
53///     - Use [`Options::default_driver`] or [`BuiltinDriver::Text`].
54/// * `merge=name`
55///     - Search for a user-configured or built-in driver called `name`.
56///     - If not found, silently default to [`BuiltinDriver::Text`]
57///
58/// Note that drivers are queried even if there is no object available.
59#[derive(Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Debug, Hash)]
60pub enum DriverChoice {
61    /// Use the given built-in driver to perform the merge.
62    BuiltIn(BuiltinDriver),
63    /// Use the user-provided driver program using the index into [the platform drivers array](Platform::drivers()).
64    Index(usize),
65}
66
67impl Default for DriverChoice {
68    fn default() -> Self {
69        DriverChoice::BuiltIn(Default::default())
70    }
71}
72
73/// Lifecycle
74impl Platform {
75    /// Create a new instance with a way to `filter` data from the object database and turn it into something that is merge-able.
76    /// `filter_mode` decides how to do that specifically.
77    /// Use `attr_stack` to access attributes pertaining worktree filters and merge settings.
78    /// `drivers` are the list of available merge drivers that individual paths can refer to by means of git attributes.
79    /// `options` further configure the operation.
80    pub fn new(
81        filter: Pipeline,
82        filter_mode: pipeline::Mode,
83        attr_stack: gix_worktree::Stack,
84        mut drivers: Vec<super::Driver>,
85        options: Options,
86    ) -> Self {
87        drivers.sort_by(|a, b| a.name.cmp(&b.name));
88        Platform {
89            drivers,
90            current: None,
91            ancestor: None,
92            other: None,
93            filter,
94            filter_mode,
95            attr_stack,
96            attrs: {
97                let mut out = attributes::search::Outcome::default();
98                out.initialize_with_selection(&Default::default(), ["merge", "conflict-marker-size"]);
99                out
100            },
101            options,
102        }
103    }
104}
105
106/// Access
107impl Platform {
108    /// Return all drivers that this instance was initialized with.
109    ///
110    /// They are sorted by [`name`](super::Driver::name) to support binary searches.
111    pub fn drivers(&self) -> &[super::Driver] {
112        &self.drivers
113    }
114}
115
116///
117pub mod set_resource;
118
119///
120pub mod resource;
121
122///
123pub mod merge;
124pub use merge::inner::{builtin_merge, prepare_external_driver};
125
126///
127pub mod prepare_merge;