@ -44,6 +44,7 @@
package org.eclipse.jgit.blame ;
package org.eclipse.jgit.blame ;
import static org.eclipse.jgit.lib.Constants.OBJ_BLOB ;
import static org.eclipse.jgit.lib.Constants.OBJ_BLOB ;
import static org.eclipse.jgit.lib.FileMode.TYPE_FILE ;
import java.io.IOException ;
import java.io.IOException ;
import java.util.Collection ;
import java.util.Collection ;
@ -72,6 +73,7 @@ import org.eclipse.jgit.revwalk.RevCommit;
import org.eclipse.jgit.revwalk.RevFlag ;
import org.eclipse.jgit.revwalk.RevFlag ;
import org.eclipse.jgit.revwalk.RevWalk ;
import org.eclipse.jgit.revwalk.RevWalk ;
import org.eclipse.jgit.treewalk.TreeWalk ;
import org.eclipse.jgit.treewalk.TreeWalk ;
import org.eclipse.jgit.treewalk.filter.AndTreeFilter ;
import org.eclipse.jgit.treewalk.filter.PathFilter ;
import org.eclipse.jgit.treewalk.filter.PathFilter ;
import org.eclipse.jgit.treewalk.filter.TreeFilter ;
import org.eclipse.jgit.treewalk.filter.TreeFilter ;
@ -121,7 +123,7 @@ public class BlameGenerator {
/** Revision pool used to acquire commits from. */
/** Revision pool used to acquire commits from. */
private RevWalk revPool ;
private RevWalk revPool ;
/** Indicates the commit has already been processed . */
/** Indicates the commit was put into the queue at least once . */
private RevFlag SEEN ;
private RevFlag SEEN ;
private ObjectReader reader ;
private ObjectReader reader ;
@ -422,6 +424,18 @@ public class BlameGenerator {
return this ;
return this ;
}
}
/ * *
* Allocate a new RevFlag for use by the caller .
*
* @param name
* unique name of the flag in the blame context .
* @return the newly allocated flag .
* @since 3 . 4
* /
public RevFlag newFlag ( String name ) {
return revPool . newFlag ( name ) ;
}
/ * *
/ * *
* Execute the generator in a blocking fashion until all data is ready .
* Execute the generator in a blocking fashion until all data is ready .
*
*
@ -532,6 +546,7 @@ public class BlameGenerator {
private void push ( BlobCandidate toInsert ) {
private void push ( BlobCandidate toInsert ) {
Candidate c = queue ;
Candidate c = queue ;
if ( c ! = null ) {
if ( c ! = null ) {
c . remove ( SEEN ) ; // will be pushed by toInsert
c . regionList = null ;
c . regionList = null ;
toInsert . parent = c ;
toInsert . parent = c ;
}
}
@ -539,8 +554,24 @@ public class BlameGenerator {
}
}
private void push ( Candidate toInsert ) {
private void push ( Candidate toInsert ) {
// Mark sources to ensure they get discarded (above) if
if ( toInsert . has ( SEEN ) ) {
// another path to the same commit.
// We have already added a Candidate for this commit to the queue,
// this can happen if the commit is a merge base for two or more
// parallel branches that were merged together.
//
// It is likely the candidate was not yet processed. The queue
// sorts descending by commit time and usually descendant commits
// have higher timestamps than the ancestors.
//
// Find the existing candidate and merge the new candidate's
// region list into it.
for ( Candidate p = queue ; p ! = null ; p = p . queueNext ) {
if ( p . canMergeRegions ( toInsert ) ) {
p . mergeRegions ( toInsert ) ;
return ;
}
}
}
toInsert . add ( SEEN ) ;
toInsert . add ( SEEN ) ;
// Insert into the queue using descending commit time, so
// Insert into the queue using descending commit time, so
@ -567,23 +598,21 @@ public class BlameGenerator {
RevCommit parent = n . getParent ( 0 ) ;
RevCommit parent = n . getParent ( 0 ) ;
if ( parent = = null )
if ( parent = = null )
return split ( n . getNextCandidate ( 0 ) , n ) ;
return split ( n . getNextCandidate ( 0 ) , n ) ;
if ( parent . has ( SEEN ) )
return false ;
revPool . parseHeaders ( parent ) ;
revPool . parseHeaders ( parent ) ;
if ( find ( parent , n . sourcePath ) ) {
if ( n . sourceCommit ! = null & & n . recursivePath ) {
if ( idBuf . equals ( n . sourceBlob ) ) {
treeWalk . setFilter ( AndTreeFilter . create ( n . sourcePath , ID_DIFF ) ) ;
// The common case of the file not being modified in
treeWalk . reset ( n . sourceCommit . getTree ( ) , parent . getTree ( ) ) ;
// a simple string-of-pearls history. Blame parent.
if ( ! treeWalk . next ( ) )
n . sourceCommit = parent ;
return blameEntireRegionOnParent ( n , parent ) ;
push ( n ) ;
if ( isFile ( treeWalk . getRawMode ( 1 ) ) ) {
return false ;
treeWalk . getObjectId ( idBuf , 1 ) ;
return splitBlameWithParent ( n , parent ) ;
}
}
} else if ( find ( parent , n . sourcePath ) ) {
Candidate next = n . create ( parent , n . sourcePath ) ;
if ( idBuf . equals ( n . sourceBlob ) )
next . sourceBlob = idBuf . toObjectId ( ) ;
return blameEntireRegionOnParent ( n , parent ) ;
next . loadText ( reader ) ;
return splitBlameWithParent ( n , parent ) ;
return split ( next , n ) ;
}
}
if ( n . sourceCommit = = null )
if ( n . sourceCommit = = null )
@ -597,7 +626,7 @@ public class BlameGenerator {
// A 100% rename without any content change can also
// A 100% rename without any content change can also
// skip directly to the parent.
// skip directly to the parent.
n . sourceCommit = parent ;
n . sourceCommit = parent ;
n . sourcePath = PathFilter . create ( r . getOldPath ( ) ) ;
n . setSourcePath ( PathFilter . create ( r . getOldPath ( ) ) ) ;
push ( n ) ;
push ( n ) ;
return false ;
return false ;
}
}
@ -609,6 +638,21 @@ public class BlameGenerator {
return split ( next , n ) ;
return split ( next , n ) ;
}
}
private boolean blameEntireRegionOnParent ( Candidate n , RevCommit parent ) {
// File was not modified, blame parent.
n . sourceCommit = parent ;
push ( n ) ;
return false ;
}
private boolean splitBlameWithParent ( Candidate n , RevCommit parent )
throws IOException {
Candidate next = n . create ( parent , n . sourcePath ) ;
next . sourceBlob = idBuf . toObjectId ( ) ;
next . loadText ( reader ) ;
return split ( next , n ) ;
}
private boolean split ( Candidate parent , Candidate source )
private boolean split ( Candidate parent , Candidate source )
throws IOException {
throws IOException {
EditList editList = diffAlgorithm . diff ( textComparator ,
EditList editList = diffAlgorithm . diff ( textComparator ,
@ -636,26 +680,16 @@ public class BlameGenerator {
private boolean processMerge ( Candidate n ) throws IOException {
private boolean processMerge ( Candidate n ) throws IOException {
int pCnt = n . getParentCount ( ) ;
int pCnt = n . getParentCount ( ) ;
for ( int pIdx = 0 ; pIdx < pCnt ; pIdx + + ) {
RevCommit parent = n . getParent ( pIdx ) ;
if ( parent . has ( SEEN ) )
continue ;
revPool . parseHeaders ( parent ) ;
}
// If any single parent exactly matches the merge, follow only
// If any single parent exactly matches the merge, follow only
// that one parent through history.
// that one parent through history.
ObjectId [ ] ids = null ;
ObjectId [ ] ids = null ;
for ( int pIdx = 0 ; pIdx < pCnt ; pIdx + + ) {
for ( int pIdx = 0 ; pIdx < pCnt ; pIdx + + ) {
RevCommit parent = n . getParent ( pIdx ) ;
RevCommit parent = n . getParent ( pIdx ) ;
if ( parent . has ( SEEN ) )
revPool . parseHeaders ( parent ) ;
continue ;
if ( ! find ( parent , n . sourcePath ) )
if ( ! find ( parent , n . sourcePath ) )
continue ;
continue ;
if ( ! ( n instanceof ReverseCandidate ) & & idBuf . equals ( n . sourceBlob ) ) {
if ( ! ( n instanceof ReverseCandidate ) & & idBuf . equals ( n . sourceBlob ) ) {
n . sourceCommit = parent ;
return blameEntireRegionOnParent ( n , parent ) ;
push ( n ) ;
return false ;
}
}
if ( ids = = null )
if ( ids = = null )
ids = new ObjectId [ pCnt ] ;
ids = new ObjectId [ pCnt ] ;
@ -668,8 +702,6 @@ public class BlameGenerator {
renames = new DiffEntry [ pCnt ] ;
renames = new DiffEntry [ pCnt ] ;
for ( int pIdx = 0 ; pIdx < pCnt ; pIdx + + ) {
for ( int pIdx = 0 ; pIdx < pCnt ; pIdx + + ) {
RevCommit parent = n . getParent ( pIdx ) ;
RevCommit parent = n . getParent ( pIdx ) ;
if ( parent . has ( SEEN ) )
continue ;
if ( ids ! = null & & ids [ pIdx ] ! = null )
if ( ids ! = null & & ids [ pIdx ] ! = null )
continue ;
continue ;
@ -689,7 +721,7 @@ public class BlameGenerator {
// we choose to follow the one parent over trying to do
// we choose to follow the one parent over trying to do
// possibly both parents.
// possibly both parents.
n . sourceCommit = parent ;
n . sourceCommit = parent ;
n . sourcePath = PathFilter . create ( r . getOldPath ( ) ) ;
n . setSourcePath ( PathFilter . create ( r . getOldPath ( ) ) ) ;
push ( n ) ;
push ( n ) ;
return false ;
return false ;
}
}
@ -702,8 +734,6 @@ public class BlameGenerator {
Candidate [ ] parents = new Candidate [ pCnt ] ;
Candidate [ ] parents = new Candidate [ pCnt ] ;
for ( int pIdx = 0 ; pIdx < pCnt ; pIdx + + ) {
for ( int pIdx = 0 ; pIdx < pCnt ; pIdx + + ) {
RevCommit parent = n . getParent ( pIdx ) ;
RevCommit parent = n . getParent ( pIdx ) ;
if ( parent . has ( SEEN ) )
continue ;
Candidate p ;
Candidate p ;
if ( renames ! = null & & renames [ pIdx ] ! = null ) {
if ( renames ! = null & & renames [ pIdx ] ! = null ) {
@ -927,20 +957,17 @@ public class BlameGenerator {
private boolean find ( RevCommit commit , PathFilter path ) throws IOException {
private boolean find ( RevCommit commit , PathFilter path ) throws IOException {
treeWalk . setFilter ( path ) ;
treeWalk . setFilter ( path ) ;
treeWalk . reset ( commit . getTree ( ) ) ;
treeWalk . reset ( commit . getTree ( ) ) ;
while ( treeWalk . next ( ) ) {
if ( treeWalk . next ( ) & & isFile ( treeWalk . getRawMode ( 0 ) ) ) {
if ( path . isDone ( treeWalk ) ) {
if ( treeWalk . getFileMode ( 0 ) . getObjectType ( ) ! = OBJ_BLOB )
return false ;
treeWalk . getObjectId ( idBuf , 0 ) ;
treeWalk . getObjectId ( idBuf , 0 ) ;
return true ;
return true ;
}
}
if ( treeWalk . isSubtree ( ) )
treeWalk . enterSubtree ( ) ;
}
return false ;
return false ;
}
}
private static final boolean isFile ( int rawMode ) {
return ( rawMode & TYPE_FILE ) = = TYPE_FILE ;
}
private DiffEntry findRename ( RevCommit parent , RevCommit commit ,
private DiffEntry findRename ( RevCommit parent , RevCommit commit ,
PathFilter path ) throws IOException {
PathFilter path ) throws IOException {
if ( renameDetector = = null )
if ( renameDetector = = null )
@ -961,4 +988,26 @@ public class BlameGenerator {
return ent . getChangeType ( ) = = ChangeType . RENAME
return ent . getChangeType ( ) = = ChangeType . RENAME
| | ent . getChangeType ( ) = = ChangeType . COPY ;
| | ent . getChangeType ( ) = = ChangeType . COPY ;
}
}
private static final TreeFilter ID_DIFF = new TreeFilter ( ) {
@Override
public boolean include ( TreeWalk tw ) {
return ! tw . idEqual ( 0 , 1 ) ;
}
@Override
public boolean shouldBeRecursive ( ) {
return false ;
}
@Override
public TreeFilter clone ( ) {
return this ;
}
@Override
public String toString ( ) {
return "ID_DIFF" ; //$NON-NLS-1$
}
} ;
}
}