Merge "Implement REST endpoint to list files in a commit"
This commit is contained in:
@@ -2525,6 +2525,51 @@ describes the resulting cherry-picked change.
|
|||||||
}
|
}
|
||||||
----
|
----
|
||||||
|
|
||||||
|
[[list-files]]
|
||||||
|
=== List Files
|
||||||
|
--
|
||||||
|
'GET /projects/link:#project-name[\{project-name\}]/commits/link:#commit-id[\{commit-id\}]/files/'
|
||||||
|
--
|
||||||
|
|
||||||
|
Lists the files that were modified, added or deleted in a commit.
|
||||||
|
|
||||||
|
.Request
|
||||||
|
----
|
||||||
|
GET /projects/work%2Fmy-project/commits/a8a477efffbbf3b44169bb9a1d3a334cbbd9aa96/files/ HTTP/1.0
|
||||||
|
----
|
||||||
|
|
||||||
|
As result a map is returned that maps the link:rest-api-changes.html#file-id[file path] to a
|
||||||
|
link:rest-api-changes.html#file-info[FileInfo] entry. The entries in the map are
|
||||||
|
sorted by file path.
|
||||||
|
|
||||||
|
.Response
|
||||||
|
----
|
||||||
|
HTTP/1.1 200 OK
|
||||||
|
Content-Disposition: attachment
|
||||||
|
Content-Type: application/json; charset=UTF-8
|
||||||
|
|
||||||
|
)]}'
|
||||||
|
{
|
||||||
|
"/COMMIT_MSG": {
|
||||||
|
"status": "A",
|
||||||
|
"lines_inserted": 7,
|
||||||
|
"size_delta": 551,
|
||||||
|
"size": 551
|
||||||
|
},
|
||||||
|
"gerrit-server/src/main/java/com/google/gerrit/server/project/RefControl.java": {
|
||||||
|
"lines_inserted": 5,
|
||||||
|
"lines_deleted": 3,
|
||||||
|
"size_delta": 98,
|
||||||
|
"size": 23348
|
||||||
|
}
|
||||||
|
}
|
||||||
|
----
|
||||||
|
|
||||||
|
The integer-valued request parameter `parent` changes the response to return a
|
||||||
|
list of the files which are different in this commit compared to the given
|
||||||
|
parent commit. This is useful for supporting review of merge commits. The value
|
||||||
|
is the 1-based index of the parent's position in the commit object.
|
||||||
|
|
||||||
[[dashboard-endpoints]]
|
[[dashboard-endpoints]]
|
||||||
== Dashboard Endpoints
|
== Dashboard Endpoints
|
||||||
|
|
||||||
|
@@ -14,6 +14,8 @@
|
|||||||
|
|
||||||
package com.google.gerrit.extensions.common;
|
package com.google.gerrit.extensions.common;
|
||||||
|
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
public class FileInfo {
|
public class FileInfo {
|
||||||
public Character status;
|
public Character status;
|
||||||
public Boolean binary;
|
public Boolean binary;
|
||||||
@@ -22,4 +24,18 @@ public class FileInfo {
|
|||||||
public Integer linesDeleted;
|
public Integer linesDeleted;
|
||||||
public long sizeDelta;
|
public long sizeDelta;
|
||||||
public long size;
|
public long size;
|
||||||
|
|
||||||
|
public boolean equals(Object o) {
|
||||||
|
if (o instanceof FileInfo) {
|
||||||
|
FileInfo fileInfo = (FileInfo) o;
|
||||||
|
return Objects.equals(status, fileInfo.status)
|
||||||
|
&& Objects.equals(binary, fileInfo.binary)
|
||||||
|
&& Objects.equals(oldPath, fileInfo.oldPath)
|
||||||
|
&& Objects.equals(linesInserted, fileInfo.linesInserted)
|
||||||
|
&& Objects.equals(linesDeleted, fileInfo.linesDeleted)
|
||||||
|
&& Objects.equals(sizeDelta, fileInfo.sizeDelta)
|
||||||
|
&& Objects.equals(size, fileInfo.size);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -20,6 +20,7 @@ import com.google.gerrit.extensions.common.FileInfo;
|
|||||||
import com.google.gerrit.reviewdb.client.Change;
|
import com.google.gerrit.reviewdb.client.Change;
|
||||||
import com.google.gerrit.reviewdb.client.Patch;
|
import com.google.gerrit.reviewdb.client.Patch;
|
||||||
import com.google.gerrit.reviewdb.client.PatchSet;
|
import com.google.gerrit.reviewdb.client.PatchSet;
|
||||||
|
import com.google.gerrit.reviewdb.client.Project;
|
||||||
import com.google.gerrit.reviewdb.client.RevId;
|
import com.google.gerrit.reviewdb.client.RevId;
|
||||||
import com.google.gerrit.server.patch.PatchList;
|
import com.google.gerrit.server.patch.PatchList;
|
||||||
import com.google.gerrit.server.patch.PatchListCache;
|
import com.google.gerrit.server.patch.PatchListCache;
|
||||||
@@ -68,7 +69,12 @@ public class FileInfoJson {
|
|||||||
|
|
||||||
private Map<String, FileInfo> toFileInfoMap(Change change, PatchListKey key)
|
private Map<String, FileInfo> toFileInfoMap(Change change, PatchListKey key)
|
||||||
throws PatchListNotAvailableException {
|
throws PatchListNotAvailableException {
|
||||||
PatchList list = patchListCache.get(key, change.getProject());
|
return toFileInfoMap(change.getProject(), key);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<String, FileInfo> toFileInfoMap(Project.NameKey project, PatchListKey key)
|
||||||
|
throws PatchListNotAvailableException {
|
||||||
|
PatchList list = patchListCache.get(key, project);
|
||||||
|
|
||||||
Map<String, FileInfo> files = new TreeMap<>();
|
Map<String, FileInfo> files = new TreeMap<>();
|
||||||
for (PatchListEntry e : list.getPatches()) {
|
for (PatchListEntry e : list.getPatches()) {
|
||||||
|
@@ -14,34 +14,46 @@
|
|||||||
|
|
||||||
package com.google.gerrit.server.restapi.project;
|
package com.google.gerrit.server.restapi.project;
|
||||||
|
|
||||||
|
import com.google.gerrit.extensions.client.DiffPreferencesInfo;
|
||||||
import com.google.gerrit.extensions.registration.DynamicMap;
|
import com.google.gerrit.extensions.registration.DynamicMap;
|
||||||
import com.google.gerrit.extensions.restapi.ChildCollection;
|
import com.google.gerrit.extensions.restapi.ChildCollection;
|
||||||
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.RestReadView;
|
||||||
import com.google.gerrit.extensions.restapi.RestView;
|
import com.google.gerrit.extensions.restapi.RestView;
|
||||||
import com.google.gerrit.reviewdb.client.Patch;
|
import com.google.gerrit.reviewdb.client.Patch;
|
||||||
|
import com.google.gerrit.server.change.FileInfoJson;
|
||||||
import com.google.gerrit.server.git.GitRepositoryManager;
|
import com.google.gerrit.server.git.GitRepositoryManager;
|
||||||
|
import com.google.gerrit.server.patch.PatchListKey;
|
||||||
|
import com.google.gerrit.server.patch.PatchListNotAvailableException;
|
||||||
import com.google.gerrit.server.project.CommitResource;
|
import com.google.gerrit.server.project.CommitResource;
|
||||||
import com.google.gerrit.server.project.FileResource;
|
import com.google.gerrit.server.project.FileResource;
|
||||||
import com.google.inject.Inject;
|
import com.google.inject.Inject;
|
||||||
|
import com.google.inject.Provider;
|
||||||
import com.google.inject.Singleton;
|
import com.google.inject.Singleton;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import org.eclipse.jgit.revwalk.RevCommit;
|
||||||
|
import org.kohsuke.args4j.Option;
|
||||||
|
|
||||||
@Singleton
|
@Singleton
|
||||||
public class FilesInCommitCollection implements ChildCollection<CommitResource, FileResource> {
|
public class FilesInCommitCollection implements ChildCollection<CommitResource, FileResource> {
|
||||||
private final DynamicMap<RestView<FileResource>> views;
|
private final DynamicMap<RestView<FileResource>> views;
|
||||||
|
private final Provider<ListFiles> list;
|
||||||
private final GitRepositoryManager repoManager;
|
private final GitRepositoryManager repoManager;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
FilesInCommitCollection(
|
FilesInCommitCollection(
|
||||||
DynamicMap<RestView<FileResource>> views, GitRepositoryManager repoManager) {
|
DynamicMap<RestView<FileResource>> views,
|
||||||
|
Provider<ListFiles> list,
|
||||||
|
GitRepositoryManager repoManager) {
|
||||||
this.views = views;
|
this.views = views;
|
||||||
|
this.list = list;
|
||||||
this.repoManager = repoManager;
|
this.repoManager = repoManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public RestView<CommitResource> list() throws ResourceNotFoundException {
|
public RestView<CommitResource> list() throws ResourceNotFoundException {
|
||||||
throw new ResourceNotFoundException();
|
return list.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -57,4 +69,32 @@ public class FilesInCommitCollection implements ChildCollection<CommitResource,
|
|||||||
public DynamicMap<RestView<FileResource>> views() {
|
public DynamicMap<RestView<FileResource>> views() {
|
||||||
return views;
|
return views;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static final class ListFiles implements RestReadView<CommitResource> {
|
||||||
|
@Option(name = "--parent", metaVar = "parent-number")
|
||||||
|
int parentNum;
|
||||||
|
|
||||||
|
private final FileInfoJson fileInfoJson;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public ListFiles(FileInfoJson fileInfoJson) {
|
||||||
|
this.fileInfoJson = fileInfoJson;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object apply(CommitResource resource) throws PatchListNotAvailableException {
|
||||||
|
RevCommit commit = resource.getCommit();
|
||||||
|
PatchListKey key;
|
||||||
|
|
||||||
|
if (parentNum > 0) {
|
||||||
|
key =
|
||||||
|
PatchListKey.againstParentNum(
|
||||||
|
parentNum, commit, DiffPreferencesInfo.Whitespace.IGNORE_NONE);
|
||||||
|
} else {
|
||||||
|
key = PatchListKey.againstCommit(null, commit, DiffPreferencesInfo.Whitespace.IGNORE_NONE);
|
||||||
|
}
|
||||||
|
|
||||||
|
return fileInfoJson.toFileInfoMap(resource.getProjectState().getNameKey(), key);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -144,10 +144,7 @@ public class ProjectsRestApiBindingsIT extends AbstractRestApiBindingsTest {
|
|||||||
ImmutableList.of(
|
ImmutableList.of(
|
||||||
RestCall.get("/projects/%s/commits/%s"),
|
RestCall.get("/projects/%s/commits/%s"),
|
||||||
RestCall.get("/projects/%s/commits/%s/in"),
|
RestCall.get("/projects/%s/commits/%s/in"),
|
||||||
RestCall.builder(GET, "/projects/%s/commits/%s/files")
|
RestCall.get("/projects/%s/commits/%s/files"),
|
||||||
// GET /projects/<project>/branches/<branch>/files is not implemented
|
|
||||||
.expectedResponseCode(SC_NOT_FOUND)
|
|
||||||
.build(),
|
|
||||||
RestCall.post("/projects/%s/commits/%s/cherrypick"));
|
RestCall.post("/projects/%s/commits/%s/cherrypick"));
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@@ -0,0 +1,90 @@
|
|||||||
|
// Copyright (C) 2018 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.acceptance.rest.project;
|
||||||
|
|
||||||
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
import static com.google.gerrit.acceptance.GitUtil.getChangeId;
|
||||||
|
import static com.google.gerrit.acceptance.GitUtil.pushHead;
|
||||||
|
|
||||||
|
import com.google.gerrit.acceptance.AbstractDaemonTest;
|
||||||
|
import com.google.gerrit.acceptance.PushOneCommit;
|
||||||
|
import com.google.gerrit.acceptance.RestResponse;
|
||||||
|
import com.google.gerrit.extensions.common.FileInfo;
|
||||||
|
import com.google.gson.reflect.TypeToken;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.util.Map;
|
||||||
|
import org.eclipse.jgit.revwalk.RevCommit;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class ListCommitFilesIT extends AbstractDaemonTest {
|
||||||
|
private static String SUBJECT_1 = "subject 1";
|
||||||
|
private static String SUBJECT_2 = "subject 2";
|
||||||
|
private static String FILE_A = "a.txt";
|
||||||
|
private static String FILE_B = "b.txt";
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void listCommitFiles() throws Exception {
|
||||||
|
commitBuilder().add(FILE_B, "2").message(SUBJECT_1).create();
|
||||||
|
pushHead(testRepo, "refs/heads/master", false);
|
||||||
|
|
||||||
|
RevCommit a = commitBuilder().add(FILE_A, "1").rm(FILE_B).message(SUBJECT_2).create();
|
||||||
|
String id = getChangeId(testRepo, a).get();
|
||||||
|
pushHead(testRepo, "refs/for/master", false);
|
||||||
|
|
||||||
|
RestResponse r =
|
||||||
|
userRestSession.get("/projects/" + project.get() + "/commits/" + a.name() + "/files/");
|
||||||
|
r.assertOK();
|
||||||
|
Type type = new TypeToken<Map<String, FileInfo>>() {}.getType();
|
||||||
|
Map<String, FileInfo> files1 = newGson().fromJson(r.getReader(), type);
|
||||||
|
r.consume();
|
||||||
|
|
||||||
|
r = userRestSession.get("/changes/" + id + "/revisions/" + a.name() + "/files");
|
||||||
|
r.assertOK();
|
||||||
|
Map<String, FileInfo> files2 = newGson().fromJson(r.getReader(), type);
|
||||||
|
r.consume();
|
||||||
|
|
||||||
|
assertThat(files1).isEqualTo(files2);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void listMergeCommitFiles() throws Exception {
|
||||||
|
PushOneCommit.Result result = createMergeCommitChange("refs/for/master");
|
||||||
|
|
||||||
|
RestResponse r =
|
||||||
|
userRestSession.get(
|
||||||
|
"/projects/"
|
||||||
|
+ project.get()
|
||||||
|
+ "/commits/"
|
||||||
|
+ result.getCommit().name()
|
||||||
|
+ "/files/?parent=2");
|
||||||
|
r.assertOK();
|
||||||
|
Type type = new TypeToken<Map<String, FileInfo>>() {}.getType();
|
||||||
|
Map<String, FileInfo> files1 = newGson().fromJson(r.getReader(), type);
|
||||||
|
r.consume();
|
||||||
|
|
||||||
|
r =
|
||||||
|
userRestSession.get(
|
||||||
|
"/changes/"
|
||||||
|
+ result.getChangeId()
|
||||||
|
+ "/revisions/"
|
||||||
|
+ result.getCommit().name()
|
||||||
|
+ "/files/?parent=2");
|
||||||
|
r.assertOK();
|
||||||
|
Map<String, FileInfo> files2 = newGson().fromJson(r.getReader(), type);
|
||||||
|
r.consume();
|
||||||
|
|
||||||
|
assertThat(files1).isEqualTo(files2);
|
||||||
|
}
|
||||||
|
}
|
Reference in New Issue
Block a user