Rebuild: Include bundle diff in output
Generally speaking, the list of diffs should be empty, but printing it out is helpful to debug when something goes wrong. Change-Id: I2dc279fc315424382b088e47a98477ccfcf7ca41
This commit is contained in:
@@ -14,12 +14,19 @@
|
|||||||
|
|
||||||
package com.google.gerrit.server.change;
|
package com.google.gerrit.server.change;
|
||||||
|
|
||||||
|
import static java.util.stream.Collectors.joining;
|
||||||
|
|
||||||
|
import com.google.gerrit.extensions.restapi.BinaryResult;
|
||||||
import com.google.gerrit.extensions.restapi.IdString;
|
import com.google.gerrit.extensions.restapi.IdString;
|
||||||
import com.google.gerrit.extensions.restapi.ResourceNotFoundException;
|
import com.google.gerrit.extensions.restapi.ResourceNotFoundException;
|
||||||
import com.google.gerrit.extensions.restapi.Response;
|
|
||||||
import com.google.gerrit.extensions.restapi.RestModifyView;
|
import com.google.gerrit.extensions.restapi.RestModifyView;
|
||||||
import com.google.gerrit.reviewdb.server.ReviewDb;
|
import com.google.gerrit.reviewdb.server.ReviewDb;
|
||||||
|
import com.google.gerrit.reviewdb.server.ReviewDbUtil;
|
||||||
|
import com.google.gerrit.server.CommentsUtil;
|
||||||
import com.google.gerrit.server.change.Rebuild.Input;
|
import com.google.gerrit.server.change.Rebuild.Input;
|
||||||
|
import com.google.gerrit.server.notedb.ChangeBundle;
|
||||||
|
import com.google.gerrit.server.notedb.ChangeBundleReader;
|
||||||
|
import com.google.gerrit.server.notedb.ChangeNotes;
|
||||||
import com.google.gerrit.server.notedb.NotesMigration;
|
import com.google.gerrit.server.notedb.NotesMigration;
|
||||||
import com.google.gerrit.server.notedb.rebuild.ChangeRebuilder;
|
import com.google.gerrit.server.notedb.rebuild.ChangeRebuilder;
|
||||||
import com.google.gerrit.server.project.NoSuchChangeException;
|
import com.google.gerrit.server.project.NoSuchChangeException;
|
||||||
@@ -31,6 +38,7 @@ import com.google.inject.Singleton;
|
|||||||
import org.eclipse.jgit.errors.ConfigInvalidException;
|
import org.eclipse.jgit.errors.ConfigInvalidException;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
@Singleton
|
@Singleton
|
||||||
public class Rebuild implements RestModifyView<ChangeResource, Input> {
|
public class Rebuild implements RestModifyView<ChangeResource, Input> {
|
||||||
@@ -40,29 +48,63 @@ public class Rebuild implements RestModifyView<ChangeResource, Input> {
|
|||||||
private final Provider<ReviewDb> db;
|
private final Provider<ReviewDb> db;
|
||||||
private final NotesMigration migration;
|
private final NotesMigration migration;
|
||||||
private final ChangeRebuilder rebuilder;
|
private final ChangeRebuilder rebuilder;
|
||||||
|
private final ChangeBundleReader bundleReader;
|
||||||
|
private final CommentsUtil commentsUtil;
|
||||||
|
private final ChangeNotes.Factory notesFactory;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
Rebuild(Provider<ReviewDb> db,
|
Rebuild(Provider<ReviewDb> db,
|
||||||
NotesMigration migration,
|
NotesMigration migration,
|
||||||
ChangeRebuilder rebuilder) {
|
ChangeRebuilder rebuilder,
|
||||||
|
ChangeBundleReader bundleReader,
|
||||||
|
CommentsUtil commentsUtil,
|
||||||
|
ChangeNotes.Factory notesFactory) {
|
||||||
this.db = db;
|
this.db = db;
|
||||||
this.migration = migration;
|
this.migration = migration;
|
||||||
this.rebuilder = rebuilder;
|
this.rebuilder = rebuilder;
|
||||||
|
this.bundleReader = bundleReader;
|
||||||
|
this.commentsUtil = commentsUtil;
|
||||||
|
this.notesFactory = notesFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Response<?> apply(ChangeResource rsrc, Input input)
|
public BinaryResult apply(ChangeResource rsrc, Input input)
|
||||||
throws ResourceNotFoundException, IOException, OrmException,
|
throws ResourceNotFoundException, IOException, OrmException,
|
||||||
ConfigInvalidException {
|
ConfigInvalidException {
|
||||||
if (!migration.commitChangeWrites()) {
|
if (!migration.commitChangeWrites()) {
|
||||||
throw new ResourceNotFoundException();
|
throw new ResourceNotFoundException();
|
||||||
|
} if (!migration.readChanges()) {
|
||||||
|
// ChangeBundle#fromNotes currently doesn't work if reading isn't enabled,
|
||||||
|
// so don't attempt a diff.
|
||||||
|
rebuild(rsrc);
|
||||||
|
return BinaryResult.create("Rebuilt change successfully");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Not the same transaction as the rebuild, so may result in spurious diffs
|
||||||
|
// in the case of races. This should be easy enough to detect by rerunning.
|
||||||
|
ChangeBundle reviewDbBundle = bundleReader.fromReviewDb(
|
||||||
|
ReviewDbUtil.unwrapDb(db.get()), rsrc.getId());
|
||||||
|
rebuild(rsrc);
|
||||||
|
ChangeNotes notes = notesFactory.create(
|
||||||
|
db.get(), rsrc.getChange().getProject(), rsrc.getId());
|
||||||
|
ChangeBundle noteDbBundle = ChangeBundle.fromNotes(commentsUtil, notes);
|
||||||
|
List<String> diffs = reviewDbBundle.differencesFrom(noteDbBundle);
|
||||||
|
if (diffs.isEmpty()) {
|
||||||
|
return BinaryResult.create("No differences between ReviewDb and NoteDb");
|
||||||
|
}
|
||||||
|
return BinaryResult.create(
|
||||||
|
diffs.stream()
|
||||||
|
.collect(joining(
|
||||||
|
"\n", "Differences between ReviewDb and NoteDb:\n", "\n")));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void rebuild(ChangeResource rsrc) throws ResourceNotFoundException,
|
||||||
|
ConfigInvalidException, OrmException, IOException {
|
||||||
try {
|
try {
|
||||||
rebuilder.rebuild(db.get(), rsrc.getId());
|
rebuilder.rebuild(db.get(), rsrc.getId());
|
||||||
} catch (NoSuchChangeException e) {
|
} catch (NoSuchChangeException e) {
|
||||||
throw new ResourceNotFoundException(
|
throw new ResourceNotFoundException(
|
||||||
IdString.fromDecoded(rsrc.getId().toString()));
|
IdString.fromDecoded(rsrc.getId().toString()));
|
||||||
}
|
}
|
||||||
return Response.none();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user