blob: af23ba7131a6ffb2bd5797fe3304b8b199560d61 [file] [log] [blame]
// Copyright (C) 2021 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.restapi.change;
import static com.google.gerrit.entities.RefNames.changeMetaRef;
import com.google.gerrit.common.Nullable;
import com.google.gerrit.extensions.client.ListChangesOption;
import com.google.gerrit.extensions.client.ListOption;
import com.google.gerrit.extensions.common.ChangeInfo;
import com.google.gerrit.extensions.common.ChangeInfoDiffer;
import com.google.gerrit.extensions.common.ChangeInfoDifference;
import com.google.gerrit.extensions.restapi.BadRequestException;
import com.google.gerrit.extensions.restapi.PreconditionFailedException;
import com.google.gerrit.extensions.restapi.Response;
import com.google.gerrit.extensions.restapi.RestReadView;
import com.google.gerrit.server.DynamicOptions;
import com.google.gerrit.server.DynamicOptions.DynamicBean;
import com.google.gerrit.server.change.ChangeResource;
import com.google.gerrit.server.git.GitRepositoryManager;
import com.google.inject.Inject;
import com.google.inject.Provider;
import java.io.IOException;
import java.util.EnumSet;
import java.util.HashMap;
import java.util.Map;
import org.eclipse.jgit.errors.InvalidObjectIdException;
import org.eclipse.jgit.errors.MissingObjectException;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.Repository;
import org.eclipse.jgit.revwalk.RevCommit;
import org.eclipse.jgit.revwalk.RevWalk;
import org.kohsuke.args4j.Option;
/** Gets the diff for a change at two NoteDb meta SHA-1s. */
public class GetMetaDiff
implements RestReadView<ChangeResource>,
DynamicOptions.BeanReceiver,
DynamicOptions.BeanProvider {
private final EnumSet<ListChangesOption> options = EnumSet.noneOf(ListChangesOption.class);
private final Map<String, DynamicBean> dynamicBeans = new HashMap<>();
private final Provider<GetChange> getChangeProvider;
private final GitRepositoryManager repoManager;
@Option(name = "-o", usage = "Output options")
public void addOption(ListChangesOption o) {
options.add(o);
}
@Option(name = "-O", usage = "Output option flags, in hex")
void setOptionFlagsHex(String hex) {
options.addAll(ListOption.fromBits(ListChangesOption.class, Integer.parseInt(hex, 16)));
}
@Option(name = "--old", usage = "old NoteDb meta SHA-1")
String oldMetaRevId = "";
public void setOldMetaRevId(@Nullable String oldMetaRevId) {
this.oldMetaRevId = oldMetaRevId == null ? "" : oldMetaRevId;
}
@Option(name = "--meta", usage = "new NoteDb meta SHA-1")
String metaRevId = "";
public void setNewMetaRevId(@Nullable String metaRevId) {
this.metaRevId = metaRevId == null ? "" : metaRevId;
}
@Inject
GetMetaDiff(Provider<GetChange> getChangeProvider, GitRepositoryManager repoManager) {
this.getChangeProvider = getChangeProvider;
this.repoManager = repoManager;
}
@Override
public void setDynamicBean(String plugin, DynamicOptions.DynamicBean dynamicBean) {
dynamicBeans.put(plugin, dynamicBean);
}
@Override
public DynamicBean getDynamicBean(String plugin) {
return dynamicBeans.get(plugin);
}
@Override
public Response<ChangeInfoDifference> apply(ChangeResource resource)
throws BadRequestException, PreconditionFailedException, IOException {
return Response.ok(
ChangeInfoDiffer.getDifference(getOldChangeInfo(resource), getNewChangeInfo(resource)));
}
private ChangeInfo getOldChangeInfo(ChangeResource resource)
throws BadRequestException, IOException, PreconditionFailedException {
GetChange getChange = createGetChange();
getChange.setMetaRevId(getOldMetaRevId(resource));
ChangeInfo oldChangeInfo;
try {
oldChangeInfo = getChange.apply(resource).value();
} catch (PreconditionFailedException e) {
oldChangeInfo = new ChangeInfo();
}
return oldChangeInfo;
}
private String getOldMetaRevId(ChangeResource resource)
throws IOException, BadRequestException, PreconditionFailedException {
if (!oldMetaRevId.isEmpty()) {
return oldMetaRevId;
}
String newMetaRevId = getNewMetaRevId(resource);
try (Repository repo = repoManager.openRepository(resource.getProject());
RevWalk rw = new RevWalk(repo)) {
ObjectId resourceId = ObjectId.fromString(newMetaRevId);
RevCommit commit = rw.parseCommit(resourceId);
return commit.getParentCount() == 0
? resourceId.getName()
: commit.getParent(0).getId().getName();
} catch (InvalidObjectIdException e) {
throw new BadRequestException("invalid meta SHA1: " + newMetaRevId, e);
} catch (MissingObjectException e) {
throw new PreconditionFailedException(e.getMessage());
}
}
private ChangeInfo getNewChangeInfo(ChangeResource resource)
throws BadRequestException, PreconditionFailedException, IOException {
GetChange getChange = createGetChange();
getChange.setMetaRevId(getNewMetaRevId(resource));
return getChange.apply(resource).value();
}
private String getNewMetaRevId(ChangeResource resource) throws IOException {
if (!metaRevId.isEmpty()) {
return metaRevId;
}
try (Repository repo = repoManager.openRepository(resource.getProject())) {
return repo.exactRef(changeMetaRef(resource.getId())).getObjectId().getName();
}
}
private GetChange createGetChange() {
GetChange getChange = getChangeProvider.get();
options.forEach(getChange::addOption);
dynamicBeans.forEach(getChange::setDynamicBean);
return getChange;
}
}