@@ -18,6 +18,7 @@ use crate::string_array::StringArray;
18
18
use crate :: tagforeach:: { tag_foreach_cb, TagForeachCB , TagForeachData } ;
19
19
use crate :: util:: { self , path_to_repo_path, Binding } ;
20
20
use crate :: worktree:: { Worktree , WorktreeAddOptions } ;
21
+ use crate :: CherrypickOptions ;
21
22
use crate :: RevertOptions ;
22
23
use crate :: {
23
24
raw, AttrCheckFlags , Buf , Error , Object , Remote , RepositoryOpenFlags , RepositoryState , Revspec ,
@@ -29,7 +30,6 @@ use crate::{
29
30
use crate :: { ApplyLocation , ApplyOptions , Rebase , RebaseOptions } ;
30
31
use crate :: { Blame , BlameOptions , Reference , References , ResetType , Signature , Submodule } ;
31
32
use crate :: { Blob , BlobWriter , Branch , BranchType , Branches , Commit , Config , Index , Oid , Tree } ;
32
- use crate :: { CherrypickOptions , IndexEntry } ;
33
33
use crate :: { Describe , IntoCString , Reflog , RepositoryInitMode , RevparseMode } ;
34
34
use crate :: { DescribeOptions , Diff , DiffOptions , Odb , PackBuilder , TreeBuilder } ;
35
35
use crate :: { Note , Notes , ObjectType , Revwalk , Status , StatusOptions , Statuses , Tag } ;
@@ -1848,47 +1848,13 @@ impl Repository {
1848
1848
///
1849
1849
/// Note that this function does not reference a repository and any
1850
1850
/// configuration must be passed as `git_merge_file_options`.
1851
- ///
1852
- /// @param out The git_merge_file_result to be filled in
1853
- /// @param ancestor The contents of the ancestor file
1854
- /// @param ours The contents of the file in "our" side
1855
- /// @param theirs The contents of the file in "their" side
1856
- /// @param opts The merge file options or `NULL` for defaults
1857
- /// @return 0 on success or error code
1858
1851
pub fn merge_file (
1859
1852
& self ,
1860
- ancestor : Option < & IndexEntry > ,
1861
- ours : Option < & IndexEntry > ,
1862
- theirs : Option < & IndexEntry > ,
1853
+ ancestor : Option < & MergeFileInput < ' _ > > ,
1854
+ ours : Option < & MergeFileInput < ' _ > > ,
1855
+ theirs : Option < & MergeFileInput < ' _ > > ,
1863
1856
options : Option < & MergeFileOptions > ,
1864
1857
) -> Result < MergeFileResult , Error > {
1865
- let ancestor_input;
1866
- let ours_input;
1867
- let theirs_input;
1868
-
1869
- let ancestor_raw;
1870
- let ours_raw;
1871
- let theirs_raw;
1872
-
1873
- if let Some ( ancestor) = ancestor {
1874
- ancestor_input = MergeFileInput :: from ( & self , ancestor) ;
1875
- ancestor_raw = ancestor_input. raw ( ) ;
1876
- } else {
1877
- ancestor_raw = ptr:: null ( ) ;
1878
- }
1879
- if let Some ( ours) = ours {
1880
- ours_input = MergeFileInput :: from ( & self , ours) ;
1881
- ours_raw = ours_input. raw ( ) ;
1882
- } else {
1883
- ours_raw = ptr:: null ( ) ;
1884
- }
1885
- if let Some ( theirs) = theirs {
1886
- theirs_input = MergeFileInput :: from ( & self , theirs) ;
1887
- theirs_raw = theirs_input. raw ( ) ;
1888
- } else {
1889
- theirs_raw = ptr:: null ( ) ;
1890
- }
1891
-
1892
1858
let mut ret = raw:: git_merge_file_result {
1893
1859
automergeable : 0 ,
1894
1860
path : ptr:: null ( ) ,
@@ -1900,9 +1866,9 @@ impl Repository {
1900
1866
unsafe {
1901
1867
try_call ! ( raw:: git_merge_file(
1902
1868
& mut ret,
1903
- ancestor_raw ,
1904
- ours_raw ,
1905
- theirs_raw ,
1869
+ ancestor . map ( |a| a . raw ( ) ) . unwrap_or ( ptr :: null ( ) ) ,
1870
+ ours . map ( |a| a . raw ( ) ) . unwrap_or ( ptr :: null ( ) ) ,
1871
+ theirs . map ( |a| a . raw ( ) ) . unwrap_or ( ptr :: null ( ) ) ,
1906
1872
options. map( |o| o. raw( ) )
1907
1873
) ) ;
1908
1874
@@ -3104,8 +3070,9 @@ impl RepositoryInitOptions {
3104
3070
#[ cfg( test) ]
3105
3071
mod tests {
3106
3072
use crate :: build:: CheckoutBuilder ;
3107
- use crate :: { CherrypickOptions , FileMode } ;
3073
+ use crate :: { CherrypickOptions , FileMode , MergeFileInput } ;
3108
3074
use crate :: { ObjectType , Oid , Repository , ResetType } ;
3075
+ use std:: convert:: TryInto ;
3109
3076
use std:: ffi:: OsStr ;
3110
3077
use std:: fs;
3111
3078
use std:: io:: Write ;
@@ -3357,6 +3324,7 @@ mod tests {
3357
3324
file_a
3358
3325
. write_all ( file_on_branch_a_content_1. as_bytes ( ) )
3359
3326
. unwrap ( ) ;
3327
+ drop ( file_a) ;
3360
3328
index. add_path ( Path :: new ( "file_a" ) ) . unwrap ( ) ;
3361
3329
let id_a = index. write_tree ( ) . unwrap ( ) ;
3362
3330
let tree_a = repo. find_tree ( id_a) . unwrap ( ) ;
@@ -3378,10 +3346,11 @@ mod tests {
3378
3346
// create commit oid3 on branchB
3379
3347
let mut index = repo. index ( ) . unwrap ( ) ;
3380
3348
let p = Path :: new ( repo. workdir ( ) . unwrap ( ) ) . join ( "file_a" ) ;
3381
- let mut file_b = fs:: File :: create ( & p) . unwrap ( ) ;
3382
- file_b
3349
+ let mut file_a = fs:: File :: create ( & p) . unwrap ( ) ;
3350
+ file_a
3383
3351
. write_all ( file_on_branch_b_content_1. as_bytes ( ) )
3384
3352
. unwrap ( ) ;
3353
+ drop ( file_a) ;
3385
3354
index. add_path ( Path :: new ( "file_a" ) ) . unwrap ( ) ;
3386
3355
let id_b = index. write_tree ( ) . unwrap ( ) ;
3387
3356
let tree_b = repo. find_tree ( id_b) . unwrap ( ) ;
@@ -3405,6 +3374,7 @@ mod tests {
3405
3374
let p = Path :: new ( repo. workdir ( ) . unwrap ( ) ) . join ( "file_a" ) ;
3406
3375
let mut file_a = fs:: OpenOptions :: new ( ) . append ( true ) . open ( & p) . unwrap ( ) ;
3407
3376
file_a. write ( file_on_branch_a_content_2. as_bytes ( ) ) . unwrap ( ) ;
3377
+ drop ( file_a) ;
3408
3378
index. add_path ( Path :: new ( "file_a" ) ) . unwrap ( ) ;
3409
3379
let id_a_2 = index. write_tree ( ) . unwrap ( ) ;
3410
3380
let tree_a_2 = repo. find_tree ( id_a_2) . unwrap ( ) ;
@@ -3423,11 +3393,12 @@ mod tests {
3423
3393
3424
3394
t ! ( repo. reset( commit3. as_object( ) , ResetType :: Hard , None ) ) ;
3425
3395
3426
- // create commit oid4 on branchB
3396
+ // create commit oid5 on branchB
3427
3397
let mut index = repo. index ( ) . unwrap ( ) ;
3428
3398
let p = Path :: new ( repo. workdir ( ) . unwrap ( ) ) . join ( "file_a" ) ;
3429
3399
let mut file_a = fs:: OpenOptions :: new ( ) . append ( true ) . open ( & p) . unwrap ( ) ;
3430
3400
file_a. write ( file_on_branch_b_content_2. as_bytes ( ) ) . unwrap ( ) ;
3401
+ drop ( file_a) ;
3431
3402
index. add_path ( Path :: new ( "file_a" ) ) . unwrap ( ) ;
3432
3403
let id_b_2 = index. write_tree ( ) . unwrap ( ) ;
3433
3404
let tree_b_2 = repo. find_tree ( id_b_2) . unwrap ( ) ;
@@ -3457,11 +3428,63 @@ mod tests {
3457
3428
for conflict in index_conflicts {
3458
3429
let conflict = conflict. unwrap ( ) ;
3459
3430
3431
+ let ancestor_input;
3432
+ let ours_input;
3433
+ let theirs_input;
3434
+
3435
+ let ancestor_blob;
3436
+ let ours_blob;
3437
+ let theirs_blob;
3438
+
3439
+ let ancestor_content;
3440
+ let ours_content;
3441
+ let theirs_content;
3442
+
3443
+ if let Some ( ancestor) = conflict. ancestor {
3444
+ ancestor_blob = repo
3445
+ . find_blob ( ancestor. id . clone ( ) )
3446
+ . expect ( "failed to find blob of index entry to make MergeFileInput" ) ;
3447
+ ancestor_content = ancestor_blob. content ( ) ;
3448
+ let mut input = MergeFileInput :: new ( ) ;
3449
+ input. path ( String :: from_utf8 ( ancestor. path ) . unwrap ( ) ) ;
3450
+ input. mode ( Some ( FileMode :: from ( ancestor. mode . try_into ( ) . unwrap ( ) ) ) ) ;
3451
+ input. content ( Some ( & ancestor_content) ) ;
3452
+ ancestor_input = Some ( input) ;
3453
+ } else {
3454
+ ancestor_input = None ;
3455
+ }
3456
+ if let Some ( ours) = conflict. our {
3457
+ ours_blob = repo
3458
+ . find_blob ( ours. id . clone ( ) )
3459
+ . expect ( "failed to find blob of index entry to make MergeFileInput" ) ;
3460
+ ours_content = ours_blob. content ( ) ;
3461
+ let mut input = MergeFileInput :: new ( ) ;
3462
+ input. path ( String :: from_utf8 ( ours. path ) . unwrap ( ) ) ;
3463
+ input. mode ( Some ( FileMode :: from ( ours. mode . try_into ( ) . unwrap ( ) ) ) ) ;
3464
+ input. content ( Some ( & ours_content) ) ;
3465
+ ours_input = Some ( input) ;
3466
+ } else {
3467
+ ours_input = None ;
3468
+ }
3469
+ if let Some ( theirs) = conflict. their {
3470
+ theirs_blob = repo
3471
+ . find_blob ( theirs. id . clone ( ) )
3472
+ . expect ( "failed to find blob of index entry to make MergeFileInput" ) ;
3473
+ theirs_content = theirs_blob. content ( ) ;
3474
+ let mut input = MergeFileInput :: new ( ) ;
3475
+ input. path ( String :: from_utf8 ( theirs. path ) . unwrap ( ) ) ;
3476
+ input. mode ( Some ( FileMode :: from ( theirs. mode . try_into ( ) . unwrap ( ) ) ) ) ;
3477
+ input. content ( Some ( & theirs_content) ) ;
3478
+ theirs_input = Some ( input) ;
3479
+ } else {
3480
+ theirs_input = None ;
3481
+ }
3482
+
3460
3483
let merge_file_result = repo
3461
3484
. merge_file (
3462
- conflict . ancestor . as_ref ( ) ,
3463
- conflict . our . as_ref ( ) ,
3464
- conflict . their . as_ref ( ) ,
3485
+ ancestor_input . as_ref ( ) ,
3486
+ ours_input . as_ref ( ) ,
3487
+ theirs_input . as_ref ( ) ,
3465
3488
None ,
3466
3489
)
3467
3490
. unwrap ( ) ;
0 commit comments