Add change messages to Get Change Detail REST api call

Include change-level messages to the payload returned from
the existing Get Change Detail REST api endpoint.

Also available as an option in the Query Changes endpoint.

Add associated unit test, update REST api docs.

Bug: Issue 1819
Change-Id: Iddd50b4054f2ab9113208a1ce618db3e34b5de8b
This commit is contained in:
John Spurlock
2013-03-23 16:41:50 -04:00
parent 26bac6f723
commit 74a70ccb14
6 changed files with 342 additions and 4 deletions

View File

@@ -208,6 +208,11 @@ default. Optional fields are:
referencing accounts. referencing accounts.
-- --
[[messages]]
--
* `MESSAGES`: include messages associated with the change.
--
.Request .Request
---- ----
GET /changes/?q=97&o=CURRENT_REVISION&o=CURRENT_COMMIT&o=CURRENT_FILES HTTP/1.0 GET /changes/?q=97&o=CURRENT_REVISION&o=CURRENT_COMMIT&o=CURRENT_FILES HTTP/1.0
@@ -356,7 +361,8 @@ Get Change Detail
'GET /changes/link:#change-id[\{change-id\}]/detail' 'GET /changes/link:#change-id[\{change-id\}]/detail'
Retrieves a change with link:#labels[labels], link:#detailed-labels[ Retrieves a change with link:#labels[labels], link:#detailed-labels[
detailed labels] and link:#detailed-accounts[detailed accounts]. detailed labels], link:#detailed-accounts[detailed accounts], and
link:#messages[messages].
.Request .Request
---- ----
@@ -473,6 +479,30 @@ describes the change.
"name": "Jane Roe", "name": "Jane Roe",
"email": "jane.roe@example.com" "email": "jane.roe@example.com"
} }
],
"messages": [
{
"id": "YH-egE",
"author": {
"_account_id": 1000096,
"name": "John Doe",
"email": "john.doe@example.com"
},
"updated": "2013-03-23 21:34:02.419000000",
"message": "Patch Set 1:\n\nThis is the first message.",
"revision_number": 1
},
{
"id": "WEEdhU",
"author": {
"_account_id": 1000097,
"name": "Jane Roe",
"email": "jane.roe@example.com"
},
"updated": "2013-03-23 21:36:52.332000000",
"message": "Patch Set 1:\n\nThis is the second message.\n\nWith a line break.",
"revision_number": 1
}
] ]
} }
---- ----
@@ -1815,6 +1845,10 @@ Only set if link:#detailed-labels[detailed labels] are requested.
The reviewers that can be removed by the calling user as a list of The reviewers that can be removed by the calling user as a list of
link:rest-api-accounts.html#account-info[AccountInfo] entities. + link:rest-api-accounts.html#account-info[AccountInfo] entities. +
Only set if link:#detailed-labels[detailed labels] are requested. Only set if link:#detailed-labels[detailed labels] are requested.
|`messages`|optional|
Messages associated with the change as a list of
link:#change-message-info[ChangeMessageInfo] entities. +
Only set if link:#messages[messages] are requested.
|`current_revision` |optional| |`current_revision` |optional|
The commit ID of the current patch set of this change. + The commit ID of the current patch set of this change. +
Only set if link:#current-revision[the current revision] is requested Only set if link:#current-revision[the current revision] is requested
@@ -1828,6 +1862,27 @@ Whether the query would deliver more results if not limited. +
Only set on either the last or the first change that is returned. Only set on either the last or the first change that is returned.
|================================== |==================================
[[change-message-info]]
ChangeMessageInfo
~~~~~~~~~~~~~~~~~
The `ChangeMessageInfo` entity contains information about a message
attached to a change.
[options="header",width="50%",cols="1,^1,5"]
|==================================
|Field Name ||Description
|`id` ||The ID of the message.
|`author` |optional|
Author of the message as an
link:rest-api-accounts.html#account-info[AccountInfo] entity. +
Unset if written by the Gerrit system.
|`date` ||
The link:rest-api.html#timestamp[timestamp] this message was posted.
|`message` ||The text left by the user.
|`_revision_number` |optional|
Which patchset (if any) generated this message.
|==================================
[[comment-info]] [[comment-info]]
CommentInfo CommentInfo
~~~~~~~~~~~ ~~~~~~~~~~~

View File

@@ -34,7 +34,10 @@ public enum ListChangesOption {
ALL_FILES(6), ALL_FILES(6),
/** If accounts are included, include detailed account info. */ /** If accounts are included, include detailed account info. */
DETAILED_ACCOUNTS(7); DETAILED_ACCOUNTS(7),
/** Include messages associated with the change. */
MESSAGES(9);
private final int value; private final int value;

View File

@@ -23,6 +23,7 @@ import static com.google.gerrit.common.changes.ListChangesOption.CURRENT_REVISIO
import static com.google.gerrit.common.changes.ListChangesOption.DETAILED_ACCOUNTS; import static com.google.gerrit.common.changes.ListChangesOption.DETAILED_ACCOUNTS;
import static com.google.gerrit.common.changes.ListChangesOption.DETAILED_LABELS; import static com.google.gerrit.common.changes.ListChangesOption.DETAILED_LABELS;
import static com.google.gerrit.common.changes.ListChangesOption.LABELS; import static com.google.gerrit.common.changes.ListChangesOption.LABELS;
import static com.google.gerrit.common.changes.ListChangesOption.MESSAGES;
import com.google.common.base.Joiner; import com.google.common.base.Joiner;
import com.google.common.base.Objects; import com.google.common.base.Objects;
@@ -251,6 +252,9 @@ public class ChangeJson {
out.permitted_labels = permittedLabels(cd); out.permitted_labels = permittedLabels(cd);
out.removable_reviewers = removableReviewers(cd, out.labels.values()); out.removable_reviewers = removableReviewers(cd, out.labels.values());
} }
if (options.contains(MESSAGES)) {
out.messages = messages(cd);
}
out.finish(); out.finish();
if (options.contains(ALL_REVISIONS) || options.contains(CURRENT_REVISION) if (options.contains(ALL_REVISIONS) || options.contains(CURRENT_REVISION)
@@ -611,6 +615,38 @@ public class ChangeJson {
return permitted.asMap(); return permitted.asMap();
} }
private Collection<ChangeMessageInfo> messages(ChangeData cd)
throws OrmException {
List<ChangeMessage> messages =
db.get().changeMessages().byChange(cd.getId()).toList();
if (messages.isEmpty()) {
return Collections.emptyList();
}
// chronological order
Collections.sort(messages, new Comparator<ChangeMessage>() {
@Override
public int compare(ChangeMessage a, ChangeMessage b) {
return a.getWrittenOn().compareTo(b.getWrittenOn());
}
});
List<ChangeMessageInfo> result =
Lists.newArrayListWithCapacity(messages.size());
for (ChangeMessage message : messages) {
PatchSet.Id patchNum = message.getPatchSetId();
ChangeMessageInfo cmi = new ChangeMessageInfo();
cmi.id = message.getKey().get();
cmi.author = accountLoader.get(message.getAuthor());
cmi.date = message.getWrittenOn();
cmi.message = message.getMessage();
cmi._revisionNumber = patchNum != null ? patchNum.get() : null;
result.add(cmi);
}
return result;
}
private Collection<AccountInfo> removableReviewers(ChangeData cd, private Collection<AccountInfo> removableReviewers(ChangeData cd,
Collection<LabelInfo> labels) throws OrmException { Collection<LabelInfo> labels) throws OrmException {
ChangeControl ctl = control(cd); ChangeControl ctl = control(cd);
@@ -841,6 +877,7 @@ public class ChangeJson {
Map<String, LabelInfo> labels; Map<String, LabelInfo> labels;
Map<String, Collection<String>> permitted_labels; Map<String, Collection<String>> permitted_labels;
Collection<AccountInfo> removable_reviewers; Collection<AccountInfo> removable_reviewers;
Collection<ChangeMessageInfo> messages;
String current_revision; String current_revision;
Map<String, RevisionInfo> revisions; Map<String, RevisionInfo> revisions;
@@ -926,4 +963,12 @@ public class ChangeJson {
super(id); super(id);
} }
} }
static class ChangeMessageInfo {
String id;
AccountInfo author;
Timestamp date;
String message;
Integer _revisionNumber;
}
} }

View File

@@ -27,7 +27,8 @@ public class GetDetail implements RestReadView<ChangeResource> {
this.json = json this.json = json
.addOption(ListChangesOption.LABELS) .addOption(ListChangesOption.LABELS)
.addOption(ListChangesOption.DETAILED_LABELS) .addOption(ListChangesOption.DETAILED_LABELS)
.addOption(ListChangesOption.DETAILED_ACCOUNTS); .addOption(ListChangesOption.DETAILED_ACCOUNTS)
.addOption(ListChangesOption.MESSAGES);
} }
@Override @Override

View File

@@ -24,7 +24,8 @@ public class GetReview implements RestReadView<RevisionResource> {
@Inject @Inject
GetReview(ChangeJson json) { GetReview(ChangeJson json) {
this.json = json.addOption(ListChangesOption.DETAILED_LABELS) this.json = json
.addOption(ListChangesOption.DETAILED_LABELS)
.addOption(ListChangesOption.DETAILED_ACCOUNTS); .addOption(ListChangesOption.DETAILED_ACCOUNTS);
} }

View File

@@ -0,0 +1,233 @@
// Copyright (C) 2013 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.change;
import static org.easymock.EasyMock.anyBoolean;
import static org.easymock.EasyMock.anyObject;
import static org.easymock.EasyMock.createMock;
import static org.easymock.EasyMock.expect;
import static org.easymock.EasyMock.expectLastCall;
import static org.easymock.EasyMock.replay;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import com.google.gerrit.common.changes.ListChangesOption;
import com.google.gerrit.reviewdb.client.Account;
import com.google.gerrit.reviewdb.client.Branch;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.client.ChangeMessage;
import com.google.gerrit.reviewdb.client.PatchSet;
import com.google.gerrit.reviewdb.client.PatchSetApproval;
import com.google.gerrit.reviewdb.client.Project;
import com.google.gerrit.reviewdb.server.ChangeAccess;
import com.google.gerrit.reviewdb.server.ChangeMessageAccess;
import com.google.gerrit.reviewdb.server.PatchSetAccess;
import com.google.gerrit.reviewdb.server.PatchSetApprovalAccess;
import com.google.gerrit.reviewdb.server.ReviewDb;
import com.google.gerrit.server.CurrentUser;
import com.google.gerrit.server.account.AccountByEmailCache;
import com.google.gerrit.server.account.AccountCache;
import com.google.gerrit.server.account.AccountInfo;
import com.google.gerrit.server.account.CapabilityControl;
import com.google.gerrit.server.account.GroupBackend;
import com.google.gerrit.server.account.Realm;
import com.google.gerrit.server.change.ChangeJson.ChangeInfo;
import com.google.gerrit.server.change.ChangeJson.ChangeMessageInfo;
import com.google.gerrit.server.config.AnonymousCowardName;
import com.google.gerrit.server.config.CanonicalWebUrl;
import com.google.gerrit.server.config.GerritServerConfig;
import com.google.gerrit.server.git.GitRepositoryManager;
import com.google.gerrit.server.patch.PatchListCache;
import com.google.gerrit.server.project.ProjectCache;
import com.google.gerrit.server.query.change.ChangeData;
import com.google.gwtorm.server.ListResultSet;
import com.google.gwtorm.server.OrmException;
import com.google.gwtorm.server.ResultSet;
import com.google.inject.Binder;
import com.google.inject.Guice;
import com.google.inject.Module;
import junit.framework.TestCase;
import org.easymock.EasyMock;
import org.easymock.IAnswer;
import org.eclipse.jgit.lib.Config;
import java.sql.Timestamp;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
public class ChangeJsonTest extends TestCase {
public void testFormatChangeMessages() throws OrmException {
// create mocks
final CurrentUser currentUser = createMock(CurrentUser.class);
final GitRepositoryManager grm = createMock(GitRepositoryManager.class);
final AccountByEmailCache abec = createMock(AccountByEmailCache.class);
final AccountCache ac = createMock(AccountCache.class);
final AccountInfo.Loader.Factory alf =
createMock(AccountInfo.Loader.Factory.class);
final CapabilityControl.Factory ccf =
createMock(CapabilityControl.Factory.class);
final GroupBackend gb = createMock(GroupBackend.class);
final Realm r = createMock(Realm.class);
final PatchListCache plc = createMock(PatchListCache.class);
final ProjectCache pc = createMock(ProjectCache.class);
final Config config = new Config(); // unable to mock
final ReviewDb rdb = createMock(ReviewDb.class);
final ChangeAccess ca = createMock(ChangeAccess.class);
final PatchSetAccess psa = createMock(PatchSetAccess.class);
final PatchSetApprovalAccess psaa =
createMock(PatchSetApprovalAccess.class);
final ChangeMessageAccess cma = createMock(ChangeMessageAccess.class);
AccountInfo.Loader accountLoader = createMock(AccountInfo.Loader.class);
// create ChangeJson instance
Module mod = new Module() {
@Override
public void configure(Binder binder) {
binder.bind(CurrentUser.class).toInstance(currentUser);
binder.bind(GitRepositoryManager.class).toInstance(grm);
binder.bind(AccountByEmailCache.class).toInstance(abec);
binder.bind(AccountCache.class).toInstance(ac);
binder.bind(AccountInfo.Loader.Factory.class).toInstance(alf);
binder.bind(CapabilityControl.Factory.class).toInstance(ccf);
binder.bind(GroupBackend.class).toInstance(gb);
binder.bind(Realm.class).toInstance(r);
binder.bind(PatchListCache.class).toInstance(plc);
binder.bind(ProjectCache.class).toInstance(pc);
binder.bind(ReviewDb.class).toInstance(rdb);
binder.bind(Config.class).annotatedWith(GerritServerConfig.class)
.toInstance(config);
binder.bind(String.class).annotatedWith(CanonicalWebUrl.class)
.toInstance("");
binder.bind(String.class).annotatedWith(AnonymousCowardName.class)
.toInstance("");
}
};
ChangeJson json = Guice.createInjector(mod).getInstance(ChangeJson.class);
// define mock behavior for tests
expect(alf.create(anyBoolean())).andReturn(accountLoader).anyTimes();
Project.NameKey proj = new Project.NameKey("ProjectNameKey");
Branch.NameKey forBranch = new Branch.NameKey(proj, "BranchNameKey");
Change.Key changeKey123 = new Change.Key("ChangeKey123");
Change.Id changeId123 = new Change.Id(123);
Change change123 = new Change(changeKey123, changeId123, null, forBranch);
Change.Key changeKey234 = new Change.Key("ChangeKey234");
Change.Id changeId234 = new Change.Id(234);
Change change234 = new Change(changeKey234, changeId234, null, forBranch);
expect(ca.get(Sets.newHashSet(changeId123)))
.andAnswer(results(Change.class, change123)).anyTimes();
expect(ca.get(changeId123)).andReturn(change123).anyTimes();
expect(ca.get(Sets.newHashSet(changeId234)))
.andAnswer(results(Change.class, change234));
expect(ca.get(changeId234)).andReturn(change234);
expect(rdb.changes()).andReturn(ca).anyTimes();
expect(psa.get(EasyMock.<Iterable<PatchSet.Id>>anyObject()))
.andAnswer(results(PatchSet.class)).anyTimes();
expect(rdb.patchSets()).andReturn(psa).anyTimes();
expect(psaa.byPatchSet(anyObject(PatchSet.Id.class)))
.andAnswer(results(PatchSetApproval.class)).anyTimes();
expect(rdb.patchSetApprovals()).andReturn(psaa).anyTimes();
expect(currentUser.getStarredChanges())
.andReturn(Collections.<Change.Id>emptySet()).anyTimes();
long timeBase = System.currentTimeMillis();
ChangeMessage changeMessage1 =changeMessage(
changeId123, "cm1", 111, timeBase, 1111, "first message");
ChangeMessage changeMessage2 = changeMessage(
changeId123, "cm2", 222, timeBase + 1000, 1111, "second message");
expect(cma.byChange(changeId123))
.andAnswer(results(ChangeMessage.class, changeMessage2, changeMessage1))
.anyTimes();
expect(cma.byChange(changeId234)).andAnswer(results(ChangeMessage.class));
expect(rdb.changeMessages()).andReturn(cma).anyTimes();
expect(accountLoader.get(anyObject(Account.Id.class)))
.andAnswer(accountForId()).anyTimes();
accountLoader.fill();
expectLastCall().anyTimes();
replay(rdb, ca, psa, psaa, alf, currentUser, cma, accountLoader);
// test 1: messages not returned by default
ChangeInfo ci = json.format(new ChangeData(changeId123));
assertNull(ci.messages);
json.addOption(ListChangesOption.MESSAGES);
// test 2: two change messages, in chronological order
ci = json.format(new ChangeData(changeId123));
assertNotNull(ci.messages);
assertEquals(2, ci.messages.size());
Iterator<ChangeMessageInfo> cmis = ci.messages.iterator();
assertEquals(changeMessage1, cmis.next());
assertEquals(changeMessage2, cmis.next());
// test 3: no change messages
ci = json.format(new ChangeData(changeId234));
assertNotNull(ci.messages);
assertEquals(0, ci.messages.size());
}
private static IAnswer<AccountInfo> accountForId() {
return new IAnswer<AccountInfo>() {
@Override
public AccountInfo answer() throws Throwable {
Account.Id id = (Account.Id) EasyMock.getCurrentArguments()[0];
AccountInfo ai = new AccountInfo(id);
return ai;
}};
}
private static <T> IAnswer<ResultSet<T>> results(Class<T> type, T... items) {
final List<T> list = Lists.newArrayList(items);
return new IAnswer<ResultSet<T>>() {
@Override
public ResultSet<T> answer() throws Throwable {
return new ListResultSet<T>(list);
}};
}
private static void assertEquals(ChangeMessage cm, ChangeMessageInfo cmi) {
assertEquals(cm.getPatchSetId().get(), (int) cmi._revisionNumber);
assertEquals(cm.getMessage(), cmi.message);
assertEquals(cm.getKey().get(), cmi.id);
assertEquals(cm.getWrittenOn(), cmi.date);
assertNotNull(cmi.author);
assertEquals(cm.getAuthor(), cmi.author._id);
}
private static ChangeMessage changeMessage(Change.Id changeId,
String uuid, int accountId, long time, int psId, String message) {
ChangeMessage.Key key = new ChangeMessage.Key(changeId, uuid);
Account.Id author = new Account.Id(accountId);
Timestamp updated = new Timestamp(time);
PatchSet.Id ps = new PatchSet.Id(changeId, psId);
ChangeMessage changeMessage = new ChangeMessage(key, author, updated, ps);
changeMessage.setMessage(message);
return changeMessage;
}
}