Dissolve gerrit-server top-level directory
Change-Id: I538512dfe0f1bea774c01fdd45fa410a45634011
This commit is contained in:
		
				
					committed by
					
						
						Dave Borowitz
					
				
			
			
				
	
			
			
			
						parent
						
							472396c797
						
					
				
				
					commit
					376a7bbb64
				
			
							
								
								
									
										226
									
								
								java/com/google/gerrit/server/git/NotesBranchUtil.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										226
									
								
								java/com/google/gerrit/server/git/NotesBranchUtil.java
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,226 @@
 | 
			
		||||
// Copyright (C) 2012 The Android Open Source Project
 | 
			
		||||
//
 | 
			
		||||
// Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
// you may not use this file except in compliance with the License.
 | 
			
		||||
// You may obtain a copy of the License at
 | 
			
		||||
//
 | 
			
		||||
// http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
//
 | 
			
		||||
// Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
// distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
// See the License for the specific language governing permissions and
 | 
			
		||||
// limitations under the License.
 | 
			
		||||
 | 
			
		||||
package com.google.gerrit.server.git;
 | 
			
		||||
 | 
			
		||||
import static com.google.common.base.MoreObjects.firstNonNull;
 | 
			
		||||
 | 
			
		||||
import com.google.gerrit.reviewdb.client.Project;
 | 
			
		||||
import com.google.gerrit.server.GerritPersonIdent;
 | 
			
		||||
import com.google.gerrit.server.extensions.events.GitReferenceUpdated;
 | 
			
		||||
import com.google.gerrit.server.update.RefUpdateUtil;
 | 
			
		||||
import com.google.inject.Inject;
 | 
			
		||||
import com.google.inject.assistedinject.Assisted;
 | 
			
		||||
import java.io.IOException;
 | 
			
		||||
import org.eclipse.jgit.lib.BatchRefUpdate;
 | 
			
		||||
import org.eclipse.jgit.lib.CommitBuilder;
 | 
			
		||||
import org.eclipse.jgit.lib.ObjectId;
 | 
			
		||||
import org.eclipse.jgit.lib.ObjectInserter;
 | 
			
		||||
import org.eclipse.jgit.lib.ObjectReader;
 | 
			
		||||
import org.eclipse.jgit.lib.PersonIdent;
 | 
			
		||||
import org.eclipse.jgit.lib.Ref;
 | 
			
		||||
import org.eclipse.jgit.lib.Repository;
 | 
			
		||||
import org.eclipse.jgit.notes.Note;
 | 
			
		||||
import org.eclipse.jgit.notes.NoteMap;
 | 
			
		||||
import org.eclipse.jgit.notes.NoteMerger;
 | 
			
		||||
import org.eclipse.jgit.revwalk.RevCommit;
 | 
			
		||||
import org.eclipse.jgit.revwalk.RevWalk;
 | 
			
		||||
import org.eclipse.jgit.transport.ReceiveCommand;
 | 
			
		||||
 | 
			
		||||
/** A utility class for updating a notes branch with automatic merge of note trees. */
 | 
			
		||||
public class NotesBranchUtil {
 | 
			
		||||
  public interface Factory {
 | 
			
		||||
    NotesBranchUtil create(Project.NameKey project, Repository db, ObjectInserter inserter);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private final PersonIdent gerritIdent;
 | 
			
		||||
  private final GitReferenceUpdated gitRefUpdated;
 | 
			
		||||
  private final Project.NameKey project;
 | 
			
		||||
  private final Repository db;
 | 
			
		||||
  private final ObjectInserter inserter;
 | 
			
		||||
 | 
			
		||||
  private RevCommit baseCommit;
 | 
			
		||||
  private NoteMap base;
 | 
			
		||||
 | 
			
		||||
  private RevCommit oursCommit;
 | 
			
		||||
  private NoteMap ours;
 | 
			
		||||
 | 
			
		||||
  private RevWalk revWalk;
 | 
			
		||||
  private ObjectReader reader;
 | 
			
		||||
  private boolean overwrite;
 | 
			
		||||
 | 
			
		||||
  private ReviewNoteMerger noteMerger;
 | 
			
		||||
 | 
			
		||||
  @Inject
 | 
			
		||||
  public NotesBranchUtil(
 | 
			
		||||
      @GerritPersonIdent PersonIdent gerritIdent,
 | 
			
		||||
      GitReferenceUpdated gitRefUpdated,
 | 
			
		||||
      @Assisted Project.NameKey project,
 | 
			
		||||
      @Assisted Repository db,
 | 
			
		||||
      @Assisted ObjectInserter inserter) {
 | 
			
		||||
    this.gerritIdent = gerritIdent;
 | 
			
		||||
    this.gitRefUpdated = gitRefUpdated;
 | 
			
		||||
    this.project = project;
 | 
			
		||||
    this.db = db;
 | 
			
		||||
    this.inserter = inserter;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * Create a new commit in the {@code notesBranch} by updating existing or creating new notes from
 | 
			
		||||
   * the {@code notes} map.
 | 
			
		||||
   *
 | 
			
		||||
   * <p>Does not retry in the case of lock failure; callers may use {@link
 | 
			
		||||
   * com.google.gerrit.server.update.RetryHelper}.
 | 
			
		||||
   *
 | 
			
		||||
   * @param notes map of notes
 | 
			
		||||
   * @param notesBranch notes branch to update
 | 
			
		||||
   * @param commitAuthor author of the commit in the notes branch
 | 
			
		||||
   * @param commitMessage for the commit in the notes branch
 | 
			
		||||
   * @throws LockFailureException if committing the notes failed due to a lock failure on the notes
 | 
			
		||||
   *     branch
 | 
			
		||||
   * @throws IOException if committing the notes failed for any other reason
 | 
			
		||||
   */
 | 
			
		||||
  public final void commitAllNotes(
 | 
			
		||||
      NoteMap notes, String notesBranch, PersonIdent commitAuthor, String commitMessage)
 | 
			
		||||
      throws IOException {
 | 
			
		||||
    this.overwrite = true;
 | 
			
		||||
    commitNotes(notes, notesBranch, commitAuthor, commitMessage);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * Create a new commit in the {@code notesBranch} by creating not yet existing notes from the
 | 
			
		||||
   * {@code notes} map. The notes from the {@code notes} map which already exist in the note-tree of
 | 
			
		||||
   * the tip of the {@code notesBranch} will not be updated.
 | 
			
		||||
   *
 | 
			
		||||
   * <p>Does not retry in the case of lock failure; callers may use {@link
 | 
			
		||||
   * com.google.gerrit.server.update.RetryHelper}.
 | 
			
		||||
   *
 | 
			
		||||
   * @param notes map of notes
 | 
			
		||||
   * @param notesBranch notes branch to update
 | 
			
		||||
   * @param commitAuthor author of the commit in the notes branch
 | 
			
		||||
   * @param commitMessage for the commit in the notes branch
 | 
			
		||||
   * @return map with those notes from the {@code notes} that were newly created
 | 
			
		||||
   * @throws LockFailureException if committing the notes failed due to a lock failure on the notes
 | 
			
		||||
   *     branch
 | 
			
		||||
   * @throws IOException if committing the notes failed for any other reason
 | 
			
		||||
   */
 | 
			
		||||
  public final NoteMap commitNewNotes(
 | 
			
		||||
      NoteMap notes, String notesBranch, PersonIdent commitAuthor, String commitMessage)
 | 
			
		||||
      throws IOException {
 | 
			
		||||
    this.overwrite = false;
 | 
			
		||||
    commitNotes(notes, notesBranch, commitAuthor, commitMessage);
 | 
			
		||||
    NoteMap newlyCreated = NoteMap.newEmptyMap();
 | 
			
		||||
    for (Note n : notes) {
 | 
			
		||||
      if (base == null || !base.contains(n)) {
 | 
			
		||||
        newlyCreated.set(n, n.getData());
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    return newlyCreated;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private void commitNotes(
 | 
			
		||||
      NoteMap notes, String notesBranch, PersonIdent commitAuthor, String commitMessage)
 | 
			
		||||
      throws LockFailureException, IOException {
 | 
			
		||||
    try {
 | 
			
		||||
      revWalk = new RevWalk(db);
 | 
			
		||||
      reader = db.newObjectReader();
 | 
			
		||||
      loadBase(notesBranch);
 | 
			
		||||
      if (overwrite) {
 | 
			
		||||
        addAllNotes(notes);
 | 
			
		||||
      } else {
 | 
			
		||||
        addNewNotes(notes);
 | 
			
		||||
      }
 | 
			
		||||
      if (base != null) {
 | 
			
		||||
        oursCommit = createCommit(ours, commitAuthor, commitMessage, baseCommit);
 | 
			
		||||
      } else {
 | 
			
		||||
        oursCommit = createCommit(ours, commitAuthor, commitMessage);
 | 
			
		||||
      }
 | 
			
		||||
      updateRef(notesBranch);
 | 
			
		||||
    } finally {
 | 
			
		||||
      revWalk.close();
 | 
			
		||||
      reader.close();
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private void addNewNotes(NoteMap notes) throws IOException {
 | 
			
		||||
    for (Note n : notes) {
 | 
			
		||||
      if (!ours.contains(n)) {
 | 
			
		||||
        ours.set(n, n.getData());
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private void addAllNotes(NoteMap notes) throws IOException {
 | 
			
		||||
    for (Note n : notes) {
 | 
			
		||||
      if (ours.contains(n)) {
 | 
			
		||||
        // Merge the existing and the new note as if they are both new,
 | 
			
		||||
        // means: base == null
 | 
			
		||||
        // There is no really a common ancestry for these two note revisions
 | 
			
		||||
        ObjectId noteContent =
 | 
			
		||||
            getNoteMerger().merge(null, n, ours.getNote(n), reader, inserter).getData();
 | 
			
		||||
        ours.set(n, noteContent);
 | 
			
		||||
      } else {
 | 
			
		||||
        ours.set(n, n.getData());
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private NoteMerger getNoteMerger() {
 | 
			
		||||
    if (noteMerger == null) {
 | 
			
		||||
      noteMerger = new ReviewNoteMerger();
 | 
			
		||||
    }
 | 
			
		||||
    return noteMerger;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private void loadBase(String notesBranch) throws IOException {
 | 
			
		||||
    Ref branch = db.getRefDatabase().exactRef(notesBranch);
 | 
			
		||||
    if (branch != null) {
 | 
			
		||||
      baseCommit = revWalk.parseCommit(branch.getObjectId());
 | 
			
		||||
      base = NoteMap.read(revWalk.getObjectReader(), baseCommit);
 | 
			
		||||
    }
 | 
			
		||||
    if (baseCommit != null) {
 | 
			
		||||
      ours = NoteMap.read(revWalk.getObjectReader(), baseCommit);
 | 
			
		||||
    } else {
 | 
			
		||||
      ours = NoteMap.newEmptyMap();
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private RevCommit createCommit(
 | 
			
		||||
      NoteMap map, PersonIdent author, String message, RevCommit... parents) throws IOException {
 | 
			
		||||
    CommitBuilder b = new CommitBuilder();
 | 
			
		||||
    b.setTreeId(map.writeTree(inserter));
 | 
			
		||||
    b.setAuthor(author != null ? author : gerritIdent);
 | 
			
		||||
    b.setCommitter(gerritIdent);
 | 
			
		||||
    if (parents.length > 0) {
 | 
			
		||||
      b.setParentIds(parents);
 | 
			
		||||
    }
 | 
			
		||||
    b.setMessage(message);
 | 
			
		||||
    ObjectId commitId = inserter.insert(b);
 | 
			
		||||
    inserter.flush();
 | 
			
		||||
    return revWalk.parseCommit(commitId);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private void updateRef(String notesBranch) throws LockFailureException, IOException {
 | 
			
		||||
    if (baseCommit != null && oursCommit.getTree().equals(baseCommit.getTree())) {
 | 
			
		||||
      // If the trees are identical, there is no change in the notes.
 | 
			
		||||
      // Avoid saving this commit as it has no new information.
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    BatchRefUpdate bru = db.getRefDatabase().newBatchUpdate();
 | 
			
		||||
    bru.addCommand(
 | 
			
		||||
        new ReceiveCommand(firstNonNull(baseCommit, ObjectId.zeroId()), oursCommit, notesBranch));
 | 
			
		||||
    RefUpdateUtil.executeChecked(bru, revWalk);
 | 
			
		||||
    gitRefUpdated.fire(project, bru, null);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user