What's the jgit equivalent for “git merge-base --fork-point branchA branchB”

前端 未结 1 1046
悲&欢浪女
悲&欢浪女 2021-01-22 07:38

What\'s the jgit equivalent codeo for \"git merge-base --fork-point branchA branchB\"?

I tried the code below but I\'m not getting the correct answer. I\'m using this t

1条回答
  •  太阳男子
    2021-01-22 07:41

    here's a way to solve it. note, I'm writing in Gradle/Groovy so the code may look a little funky.

    • getAll Branches
    • for each branch get merge base commit - since all come from master, all will find one
    • iterate through commits and stop on first one found within the merge base list

    `

    /**
     * Obtain GitWorkingCopyLog for all changes on branch with (svn --stop-on-copy equivalent) 
     * @param repoDir
     * @return GitWorkingCopyLog
       */
    private GitWorkingCopyLog getLogs(File repoDir) {
      List commits = []
      Git git = openExistingRepository(repoDir)
    
      ObjectId head = getHead(git.repository)  // tip of current branch
    
      // find common merge ancestor branch points for all branches
      List forkCandidates = mergeBaseForAllBranches(git, head)
    
      for (RevCommit r in git.log().call()) {
        if (r.name in forkCandidates) {
          break // stop looping on first rev in common merge base
        }
        commits.add(r)
      }
      return new GitWorkingCopyLog(git.repository, commits)
    }
    
    /**
     * Generate list of commit ids for merge base with all candidates.  All branches come from master
     * so all branches share a common point of origin even if unrelated.
     * @param git jgit controller
     * @param head head revision for target branch
     * @return list of commit ids
     */
    private ArrayList mergeBaseForAllBranches(Git git, ObjectId head) {
      def baseCandidates = []
      getBranches(git).each { Ref other ->
        if (other.objectId != head) {
          String base = getMergeBase(git.repository, head, other.objectId)
          baseCandidates.add(base)
        }
      }
      baseCandidates
    }
    
    /**
     *
     * @param repository
     * @param commitIdA
     * @param commitIdB
     * @return divergence point between the two branches (even if seemingly unrelated all must come back to master)
     */
    private String getMergeBase(Repository repository, ObjectId commitIdA, ObjectId commitIdB) {
      RevWalk walk = new RevWalk(repository)
      try {
        RevCommit revA = walk.lookupCommit(commitIdA)
        RevCommit revB = walk.lookupCommit(commitIdB)
    
        walk.setRevFilter(RevFilter.MERGE_BASE)
    
    
        walk.markStart(revA)
        walk.markStart(revB)
        RevCommit mergeBase = walk.next()
        println "\tA: ${revA.name}\n\tB: ${revB.name}\n\tM: ${mergeBase.name}"
        if (! mergeBase) { return null }
        return mergeBase.name
      } catch(Exception e) {
        project.logger.error("GetMergeBase Failed: ${commitIdA}, ${commitIdB} because ${e.message}")
      }
      return null
    }
    
    /**
     * Get Refs for all branches
     * @param git
     * @return Ref list
     */
    private List getBranches(Git git) {
      List branchRefs = git.branchList().call()
      return branchRefs
    }    `
    

    0 讨论(0)
提交回复
热议问题