1
1
use crate :: util:: { self , Binding } ;
2
- use crate :: { raw, signature, Oid , Repository , Signature } ;
2
+ use crate :: { raw, signature, Error , Oid , Repository , Signature } ;
3
+ use libc:: c_char;
3
4
use std:: iter:: FusedIterator ;
4
- use std:: marker;
5
5
use std:: mem;
6
6
use std:: ops:: Range ;
7
7
use std:: path:: Path ;
8
+ use std:: { marker, ptr} ;
8
9
9
10
/// Opaque structure to hold blame results.
10
11
pub struct Blame < ' repo > {
@@ -30,6 +31,24 @@ pub struct BlameIter<'blame> {
30
31
}
31
32
32
33
impl < ' repo > Blame < ' repo > {
34
+ /// Get blame data for a file that has been modified in memory.
35
+ ///
36
+ /// Lines that differ between the buffer and the committed version are
37
+ /// marked as having a zero OID for their final_commit_id.
38
+ pub fn blame_buffer ( & self , buffer : & [ u8 ] ) -> Result < Blame < ' _ > , Error > {
39
+ let mut raw = ptr:: null_mut ( ) ;
40
+
41
+ unsafe {
42
+ try_call ! ( raw:: git_blame_buffer(
43
+ & mut raw,
44
+ self . raw,
45
+ buffer. as_ptr( ) as * const c_char,
46
+ buffer. len( )
47
+ ) ) ;
48
+ Ok ( Binding :: from_raw ( raw) )
49
+ }
50
+ }
51
+
33
52
/// Gets the number of hunks that exist in the blame structure.
34
53
pub fn len ( & self ) -> usize {
35
54
unsafe { raw:: git_blame_get_hunk_count ( self . raw ) as usize }
@@ -348,6 +367,13 @@ mod tests {
348
367
assert_eq ! ( hunk. final_start_line( ) , 1 ) ;
349
368
assert_eq ! ( hunk. path( ) , Some ( Path :: new( "foo/bar" ) ) ) ;
350
369
assert_eq ! ( hunk. lines_in_hunk( ) , 0 ) ;
351
- assert ! ( !hunk. is_boundary( ) )
370
+ assert ! ( !hunk. is_boundary( ) ) ;
371
+
372
+ let blame_buffer = blame. blame_buffer ( "\n " . as_bytes ( ) ) . unwrap ( ) ;
373
+ let line = blame_buffer. get_line ( 1 ) . unwrap ( ) ;
374
+
375
+ assert_eq ! ( blame_buffer. len( ) , 2 ) ;
376
+ assert_eq ! ( blame_buffer. iter( ) . count( ) , 2 ) ;
377
+ assert ! ( line. final_commit_id( ) . is_zero( ) ) ;
352
378
}
353
379
}
0 commit comments