1
- use std:: ffi:: CString ;
1
+ use std:: ffi:: { CStr , CString } ;
2
2
use std:: { marker, mem, ptr, str} ;
3
3
4
+ use libc:: { c_char, c_int, c_void} ;
5
+ use raw:: { git_commit, git_oid, git_signature, git_tree, GIT_PASSTHROUGH } ;
6
+
4
7
use crate :: build:: CheckoutBuilder ;
5
8
use crate :: util:: Binding ;
6
- use crate :: { raw, Error , Index , MergeOptions , Oid , Signature } ;
9
+ use crate :: { panic , raw, Commit , Error , Index , MergeOptions , Oid , Signature , Tree } ;
7
10
11
+ pub type CommitCreate < ' a > = dyn FnMut ( & Oid , & Signature < ' _ > , & Signature < ' _ > , Option < & str > , Option < & str > , & Tree < ' _ > , usize , & Commit < ' _ > ) -> i32 + ' a ;
8
12
/// Rebase options
9
13
///
10
14
/// Use to tell the rebase machinery how to operate.
@@ -13,6 +17,7 @@ pub struct RebaseOptions<'cb> {
13
17
rewrite_notes_ref : Option < CString > ,
14
18
merge_options : Option < MergeOptions > ,
15
19
checkout_options : Option < CheckoutBuilder < ' cb > > ,
20
+ commit_create_cb : Option < Box < CommitCreate < ' cb > > > ,
16
21
}
17
22
18
23
impl < ' cb > Default for RebaseOptions < ' cb > {
@@ -29,11 +34,25 @@ impl<'cb> RebaseOptions<'cb> {
29
34
rewrite_notes_ref : None ,
30
35
merge_options : None ,
31
36
checkout_options : None ,
37
+ commit_create_cb : None ,
32
38
} ;
33
39
assert_eq ! ( unsafe { raw:: git_rebase_init_options( & mut opts. raw, 1 ) } , 0 ) ;
34
40
opts
35
41
}
36
42
43
+ /// Used by `Repository::rebase`, this will instruct other clients working on this
44
+ /// rebase that you want a quiet rebase experience, which they may choose to
45
+ /// provide in an application-specific manner. This has no effect upon
46
+ /// libgit2 directly, but is provided for interoperability between Git
47
+ /// tools.
48
+ pub fn commit_create_cb < F > ( & mut self , cb : F ) -> & mut RebaseOptions < ' cb >
49
+ where
50
+ F : FnMut ( & Oid , & Signature < ' _ > , & Signature < ' _ > , Option < & str > , Option < & str > , & Tree < ' _ > , usize , & Commit < ' _ > ) -> i32 + ' cb ,
51
+ {
52
+ self . commit_create_cb = Some ( Box :: new ( cb) as Box < CommitCreate < ' cb > > ) ;
53
+ self
54
+ }
55
+
37
56
/// Used by `Repository::rebase`, this will instruct other clients working on this
38
57
/// rebase that you want a quiet rebase experience, which they may choose to
39
58
/// provide in an application-specific manner. This has no effect upon
@@ -90,6 +109,23 @@ impl<'cb> RebaseOptions<'cb> {
90
109
if let Some ( opts) = self . checkout_options . as_mut ( ) {
91
110
opts. configure ( & mut self . raw . checkout_options ) ;
92
111
}
112
+
113
+ if self . commit_create_cb . is_some ( ) {
114
+ let f: extern "C" fn (
115
+ * mut git_oid ,
116
+ * const git_signature ,
117
+ * const git_signature ,
118
+ * const c_char ,
119
+ * const c_char ,
120
+ * const git_tree ,
121
+ usize ,
122
+ * const git_commit ,
123
+ * mut c_void ,
124
+ ) -> c_int = commit_create_cb;
125
+ self . raw . commit_create_cb = Some ( f) ;
126
+ }
127
+
128
+ self . raw . payload = self as * mut _ as * mut c_void ;
93
129
self . raw . rewrite_notes_ref = self
94
130
. rewrite_notes_ref
95
131
. as_ref ( )
@@ -333,6 +369,53 @@ impl<'rebase> Binding for RebaseOperation<'rebase> {
333
369
}
334
370
}
335
371
372
+ extern "C" fn commit_create_cb (
373
+ out : * mut git_oid ,
374
+ author : * const git_signature ,
375
+ committer : * const git_signature ,
376
+ message_encoding : * const c_char ,
377
+ message : * const c_char ,
378
+ tree : * const git_tree ,
379
+ parent_count : usize ,
380
+ parents : * const git_commit ,
381
+ data : * mut c_void ,
382
+ ) -> c_int {
383
+ panic:: wrap ( || unsafe {
384
+ let payload = & mut * ( data as * mut RebaseOptions < ' _ > ) ;
385
+
386
+ let callback = match payload. commit_create_cb {
387
+ Some ( ref mut c) => {
388
+ c
389
+ } ,
390
+ None => {
391
+ return GIT_PASSTHROUGH ;
392
+ } ,
393
+ } ;
394
+ let message_encoding= if message_encoding. is_null ( ) {
395
+ None
396
+ } else {
397
+ Some ( str:: from_utf8 ( CStr :: from_ptr ( message_encoding) . to_bytes ( ) ) . unwrap ( ) )
398
+ } ;
399
+ let message= if message. is_null ( ) {
400
+ None
401
+ } else {
402
+ Some ( str:: from_utf8 ( CStr :: from_ptr ( message) . to_bytes ( ) ) . unwrap ( ) )
403
+ } ;
404
+ let outt = callback (
405
+ & Oid :: from_raw ( out) ,
406
+ & Signature :: from_raw ( author as * mut git_signature ) ,
407
+ & Signature :: from_raw ( committer as * mut git_signature ) ,
408
+ message_encoding,
409
+ message,
410
+ & Tree :: from_raw ( tree as * mut git_tree ) ,
411
+ parent_count,
412
+ & Commit :: from_raw ( parents as * mut git_commit )
413
+ ) ;
414
+ return outt;
415
+ } )
416
+ . unwrap_or ( GIT_PASSTHROUGH )
417
+ }
418
+
336
419
#[ cfg( test) ]
337
420
mod tests {
338
421
use crate :: { RebaseOperationType , RebaseOptions , Signature } ;
0 commit comments