-
Notifications
You must be signed in to change notification settings - Fork 414
New issue
Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? # to your account
WIP: Add wrappers for git_index_conflict_{add,remove,get,cleanup} #780
base: master
Are you sure you want to change the base?
Conversation
/// may be `None` to indicate that that file was not present in the trees during | ||
/// the merge. For example, ancestor_entry may be `None` to indicate that a file | ||
/// was added in both branches and must be resolved. | ||
pub fn conflict_add( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The general intention of this crate is to be a lightweight binding over libgit2 itself, but this is a pretty heavyweight method. Would it be possible to avoid the movement around of all the data internally?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I based this implementation on all the other methods taking a IndexEntry
(e.g. Index::add()
). This one is particularly long because it takes three IndexEntry
parameters. I don't know if the handling of IndexEntry
s can be more concise, my intention was more to handle them exactly the same as in the existing methods.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe std::mem::transmute
can be used here, but I have no clue if git_index_entry
is even remotely binary compatible with an IndexEntry
especially regarding the path: Vec<u8>
. Also, this would be incredibly easy to mess up and hard to debug 🙈
Maybe someone has a better idea 😄
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah ok, well in that case can the conversion into a git_index_entry
be refactored into a helper function instead of being duplicated?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure, will do so. Should I also refactor all conversions in other methods using IndexEntry
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess? Sort of depends, I'm not intimately familiar with all this code.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Alright, my solution is to use a helper function which supplies pointers to raw::git_index_entry
instances via a callback. This is needed as the construction of a git_index_entry
requires the construction of a CString
to which the git_index_entry
points to. I went for a callback solution as it is (in my opinion) the easiest approach to make sure that the CString
is available while using the raw git_index_entry
. The procedure is implemented in try_raw_entries()@src/index.rs:92
. The implementation is a bit messy and requires the construction of three Vec
s. Suggestions welcome!
src/index.rs
Outdated
// a CString which is owned by the function. To make the pointer to the CString | ||
// valid during usage of raw::git_index_entry, we supply the index entry in a | ||
// callback where pointers to the CString are valid. | ||
fn try_raw_entries( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this function would be easier to use as:
impl IndexEntry {
fn to_raw(&self, storage: &mut Vec<CString>) -> raw::git_index_entry {
// ...
}
}
which would avoid the vector, allocations internally, and callback style
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In my approach the Vec
s are necessary as I need multiple entries available in the callback, therefore try_raw_entries
can handle multiple entries.
With your approach I think something like this would work:
impl IndexEntry {
fn to_raw(&self, cpath: &CString) -> raw::git_index_entry {
// ...
}
}
Although, I think this would still not ensure that cpath
lives longer than the returned value. The usage would look like this:
let cpath = CString::new(&self.path[..])?;
let raw = self.to_raw(&cpath);
My last commit replaces the Vec
s with arrays but still contains the callback. If it is safe to have a pointer pointing to the cpath
parameter in the return value, I would convert the implementation to that approach. But I'm unsure whether cpath
is actually always freed after the return value as it has no lifetime attached to it. I am not that familiar with the internals of memory management in Rust.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I implemented your callback-less approach and checked if the tests are passing. Looks good so far, but I'm still not sure whether this API is safe: the return value references data of a parameter but does not carry the lifetime of the parameter to indicate that c_path
must live longer than the returned value. Maybe this is not necessary as the c_path
binding is always created before the return value in assigned to a variable, I don't know.
c1cea81
to
6e1ec16
Compare
8894101
to
9e7b2fb
Compare
9e7b2fb
to
b548535
Compare
☔ The latest upstream changes (possibly af81314) made this pull request unmergeable. Please resolve the merge conflicts. |
In response to #778 I added wrapper functions for
git_index_conflict_add
git_index_conflict_remove
git_index_conflict_cleanup
git_index_conflict_get
However, I do not know if this trivial addition is actually safe, as conflicts are already exposed via an iterator. I do not know whether it is safe to modify the index while iterating over the conflicts.
Suggestions welcome!