Merge "Add REST endpoint to create empty change"
This commit is contained in:
commit
438e99402e
@ -7,6 +7,63 @@ link:rest-api.html[REST API].
|
|||||||
[[change-endpoints]]
|
[[change-endpoints]]
|
||||||
== Change Endpoints
|
== Change Endpoints
|
||||||
|
|
||||||
|
[[create-change]]
|
||||||
|
=== Create Change
|
||||||
|
--
|
||||||
|
'POST /changes'
|
||||||
|
--
|
||||||
|
|
||||||
|
The change info link:#change-info[ChangeInfo] entity must be provided in the
|
||||||
|
request body. Only the following attributes are honored: `project`,
|
||||||
|
`branch`, `subject`, `status` and `topic`. The first three attributes are
|
||||||
|
mandatory. Valid values for status are: `DRAFT` and `NEW`.
|
||||||
|
|
||||||
|
.Request
|
||||||
|
----
|
||||||
|
POST /changes HTTP/1.0
|
||||||
|
Content-Type: application/json;charset=UTF-8
|
||||||
|
|
||||||
|
{
|
||||||
|
"project" : "myProject",
|
||||||
|
"subject" : "Let's support 100% Gerrit workflow direct in browser",
|
||||||
|
"branch" : "master",
|
||||||
|
"topic" : "create-change-in-browser",
|
||||||
|
"status" : "DRAFT"
|
||||||
|
}
|
||||||
|
----
|
||||||
|
|
||||||
|
As response a link:#change-info[ChangeInfo] entity is returned that describes
|
||||||
|
the resulting change.
|
||||||
|
|
||||||
|
.Response
|
||||||
|
----
|
||||||
|
HTTP/1.1 200 OK
|
||||||
|
Content-Disposition: attachment
|
||||||
|
Content-Type: application/json;charset=UTF-8
|
||||||
|
|
||||||
|
)]}'
|
||||||
|
{
|
||||||
|
"kind": "gerritcodereview#change",
|
||||||
|
"id": "myProject~master~I8473b95934b5732ac55d26311a706c9c2bde9941",
|
||||||
|
"project": "myProject",
|
||||||
|
"branch": "master",
|
||||||
|
"topic": "create-change-in-browser",
|
||||||
|
"change_id": "I8473b95934b5732ac55d26311a706c9c2bde9941",
|
||||||
|
"subject": "Let's support 100% Gerrit workflow direct in browser",
|
||||||
|
"status": "DRAFT",
|
||||||
|
"created": "2014-05-05 07:15:44.639000000",
|
||||||
|
"updated": "2014-05-05 07:15:44.639000000",
|
||||||
|
"mergeable": true,
|
||||||
|
"insertions": 0,
|
||||||
|
"deletions": 0,
|
||||||
|
"_sortkey": "002cbc25000004e5",
|
||||||
|
"_number": 4711,
|
||||||
|
"owner": {
|
||||||
|
"name": "John Doe"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
----
|
||||||
|
|
||||||
[[list-changes]]
|
[[list-changes]]
|
||||||
=== Query Changes
|
=== Query Changes
|
||||||
--
|
--
|
||||||
@ -3439,7 +3496,6 @@ The `WebLinkInfo` entity describes a link to an external site.
|
|||||||
|`url` |The link URL.
|
|`url` |The link URL.
|
||||||
|======================
|
|======================
|
||||||
|
|
||||||
|
|
||||||
GERRIT
|
GERRIT
|
||||||
------
|
------
|
||||||
Part of link:index.html[Gerrit Code Review]
|
Part of link:index.html[Gerrit Code Review]
|
||||||
|
@ -0,0 +1,90 @@
|
|||||||
|
// Copyright (C) 2014 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.change;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
import com.google.gerrit.acceptance.AbstractDaemonTest;
|
||||||
|
import com.google.gerrit.acceptance.RestResponse;
|
||||||
|
import com.google.gerrit.extensions.common.ChangeInfo;
|
||||||
|
import com.google.gerrit.extensions.common.ChangeStatus;
|
||||||
|
import com.google.gerrit.server.change.ChangeJson;
|
||||||
|
|
||||||
|
import org.apache.http.HttpStatus;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class CreateChangeIT extends AbstractDaemonTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void createEmptyChange_MissingBranch() throws Exception {
|
||||||
|
ChangeInfo ci = new ChangeInfo();
|
||||||
|
ci.project = project.get();
|
||||||
|
RestResponse r = adminSession.post("/changes/", ci);
|
||||||
|
assertEquals(HttpStatus.SC_BAD_REQUEST, r.getStatusCode());
|
||||||
|
r.getEntityContent().contains("branch must be non-empty");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void createEmptyChange_MissingMessage() throws Exception {
|
||||||
|
ChangeInfo ci = new ChangeInfo();
|
||||||
|
ci.project = project.get();
|
||||||
|
ci.branch = "master";
|
||||||
|
RestResponse r = adminSession.post("/changes/", ci);
|
||||||
|
assertEquals(HttpStatus.SC_BAD_REQUEST, r.getStatusCode());
|
||||||
|
r.getEntityContent().contains("commit message must be non-empty");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void createEmptyChange_InvalidStatus() throws Exception {
|
||||||
|
ChangeInfo ci = newChangeInfo(ChangeStatus.SUBMITTED);
|
||||||
|
RestResponse r = adminSession.post("/changes/", ci);
|
||||||
|
assertEquals(HttpStatus.SC_BAD_REQUEST, r.getStatusCode());
|
||||||
|
r.getEntityContent().contains("unsupported change status");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void createNewChange() throws Exception {
|
||||||
|
assertChange(newChangeInfo(ChangeStatus.NEW));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void createDraftChange() throws Exception {
|
||||||
|
assertChange(newChangeInfo(ChangeStatus.DRAFT));
|
||||||
|
}
|
||||||
|
|
||||||
|
private ChangeInfo newChangeInfo(ChangeStatus status) {
|
||||||
|
ChangeInfo in = new ChangeInfo();
|
||||||
|
in.project = project.get();
|
||||||
|
in.branch = "master";
|
||||||
|
in.subject = "Empty change";
|
||||||
|
in.topic = "support-gerrit-workflow-in-browser";
|
||||||
|
in.status = status;
|
||||||
|
return in;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertChange(ChangeInfo in) throws Exception {
|
||||||
|
RestResponse r = adminSession.post("/changes/", in);
|
||||||
|
assertEquals(HttpStatus.SC_CREATED, r.getStatusCode());
|
||||||
|
|
||||||
|
ChangeJson.ChangeInfo info = newGson().fromJson(r.getReader(),
|
||||||
|
ChangeJson.ChangeInfo.class);
|
||||||
|
ChangeInfo out = get(info.changeId);
|
||||||
|
|
||||||
|
assertEquals(in.branch, out.branch);
|
||||||
|
assertEquals(in.subject, out.subject);
|
||||||
|
assertEquals(in.topic, out.topic);
|
||||||
|
assertEquals(in.status, out.status);
|
||||||
|
}
|
||||||
|
}
|
@ -21,7 +21,7 @@ import static com.google.gerrit.extensions.common.ListChangesOption.DETAILED_LAB
|
|||||||
import static com.google.gerrit.extensions.common.ListChangesOption.LABELS;
|
import static com.google.gerrit.extensions.common.ListChangesOption.LABELS;
|
||||||
import static com.google.gerrit.extensions.common.ListChangesOption.MESSAGES;
|
import static com.google.gerrit.extensions.common.ListChangesOption.MESSAGES;
|
||||||
|
|
||||||
import com.google.common.collect.ImmutableMap;
|
import com.google.common.collect.EnumBiMap;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
import com.google.common.collect.Maps;
|
import com.google.common.collect.Maps;
|
||||||
import com.google.gerrit.extensions.common.ApprovalInfo;
|
import com.google.gerrit.extensions.common.ApprovalInfo;
|
||||||
@ -39,14 +39,16 @@ import java.util.EnumSet;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
class ChangeInfoMapper {
|
public class ChangeInfoMapper {
|
||||||
private final static ImmutableMap<Change.Status, ChangeStatus> MAP =
|
private final static EnumBiMap<Change.Status, ChangeStatus> MAP =
|
||||||
Maps.immutableEnumMap(ImmutableMap.of(
|
EnumBiMap.create(Change.Status.class, ChangeStatus.class);
|
||||||
Status.DRAFT, ChangeStatus.DRAFT,
|
static {
|
||||||
Status.NEW, ChangeStatus.NEW,
|
MAP.put(Status.DRAFT, ChangeStatus.DRAFT);
|
||||||
Status.SUBMITTED, ChangeStatus.SUBMITTED,
|
MAP.put(Status.NEW, ChangeStatus.NEW);
|
||||||
Status.MERGED, ChangeStatus.MERGED,
|
MAP.put(Status.SUBMITTED, ChangeStatus.SUBMITTED);
|
||||||
Status.ABANDONED, ChangeStatus.ABANDONED));
|
MAP.put(Status.MERGED, ChangeStatus.MERGED);
|
||||||
|
MAP.put(Status.ABANDONED, ChangeStatus.ABANDONED);
|
||||||
|
}
|
||||||
|
|
||||||
private final EnumSet<ListChangesOption> s;
|
private final EnumSet<ListChangesOption> s;
|
||||||
|
|
||||||
@ -136,6 +138,13 @@ class ChangeInfoMapper {
|
|||||||
return s.contains(o);
|
return s.contains(o);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static Status changeStatus2Status(ChangeStatus status) {
|
||||||
|
if (status != null) {
|
||||||
|
return MAP.inverse().get(status);
|
||||||
|
}
|
||||||
|
return Change.Status.NEW;
|
||||||
|
}
|
||||||
|
|
||||||
private static ApprovalInfo fromApprovalInfo(ChangeJson.ApprovalInfo ai) {
|
private static ApprovalInfo fromApprovalInfo(ChangeJson.ApprovalInfo ai) {
|
||||||
ApprovalInfo ao = new ApprovalInfo();
|
ApprovalInfo ao = new ApprovalInfo();
|
||||||
ao.value = ai.value;
|
ao.value = ai.value;
|
||||||
|
@ -16,8 +16,10 @@ package com.google.gerrit.server.change;
|
|||||||
|
|
||||||
import com.google.common.collect.ImmutableList;
|
import com.google.common.collect.ImmutableList;
|
||||||
import com.google.gerrit.extensions.registration.DynamicMap;
|
import com.google.gerrit.extensions.registration.DynamicMap;
|
||||||
|
import com.google.gerrit.extensions.restapi.AcceptsPost;
|
||||||
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.RestApiException;
|
||||||
import com.google.gerrit.extensions.restapi.RestCollection;
|
import com.google.gerrit.extensions.restapi.RestCollection;
|
||||||
import com.google.gerrit.extensions.restapi.RestView;
|
import com.google.gerrit.extensions.restapi.RestView;
|
||||||
import com.google.gerrit.extensions.restapi.TopLevelResource;
|
import com.google.gerrit.extensions.restapi.TopLevelResource;
|
||||||
@ -35,12 +37,14 @@ import java.util.Collections;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public class ChangesCollection implements
|
public class ChangesCollection implements
|
||||||
RestCollection<TopLevelResource, ChangeResource> {
|
RestCollection<TopLevelResource, ChangeResource>,
|
||||||
|
AcceptsPost<TopLevelResource> {
|
||||||
private final Provider<ReviewDb> db;
|
private final Provider<ReviewDb> db;
|
||||||
private final Provider<CurrentUser> user;
|
private final Provider<CurrentUser> user;
|
||||||
private final ChangeControl.GenericFactory changeControlFactory;
|
private final ChangeControl.GenericFactory changeControlFactory;
|
||||||
private final Provider<QueryChanges> queryFactory;
|
private final Provider<QueryChanges> queryFactory;
|
||||||
private final DynamicMap<RestView<ChangeResource>> views;
|
private final DynamicMap<RestView<ChangeResource>> views;
|
||||||
|
private final CreateChange.Factory createChangeFactory;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
ChangesCollection(
|
ChangesCollection(
|
||||||
@ -48,12 +52,14 @@ public class ChangesCollection implements
|
|||||||
Provider<CurrentUser> user,
|
Provider<CurrentUser> user,
|
||||||
ChangeControl.GenericFactory changeControlFactory,
|
ChangeControl.GenericFactory changeControlFactory,
|
||||||
Provider<QueryChanges> queryFactory,
|
Provider<QueryChanges> queryFactory,
|
||||||
DynamicMap<RestView<ChangeResource>> views) {
|
DynamicMap<RestView<ChangeResource>> views,
|
||||||
|
CreateChange.Factory createChangeFactory) {
|
||||||
this.db = dbProvider;
|
this.db = dbProvider;
|
||||||
this.user = user;
|
this.user = user;
|
||||||
this.changeControlFactory = changeControlFactory;
|
this.changeControlFactory = changeControlFactory;
|
||||||
this.queryFactory = queryFactory;
|
this.queryFactory = queryFactory;
|
||||||
this.views = views;
|
this.views = views;
|
||||||
|
this.createChangeFactory = createChangeFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -125,4 +131,10 @@ public class ChangesCollection implements
|
|||||||
triplet.getBranchNameKey(),
|
triplet.getBranchNameKey(),
|
||||||
triplet.getChangeKey()).toList();
|
triplet.getChangeKey()).toList();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
@Override
|
||||||
|
public CreateChange post(TopLevelResource parent) throws RestApiException {
|
||||||
|
return createChangeFactory.create();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,265 @@
|
|||||||
|
// Copyright (C) 2014 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 com.google.common.base.Strings;
|
||||||
|
import com.google.gerrit.common.data.Capable;
|
||||||
|
import com.google.gerrit.extensions.common.ChangeInfo;
|
||||||
|
import com.google.gerrit.extensions.common.ChangeStatus;
|
||||||
|
import com.google.gerrit.extensions.restapi.AuthException;
|
||||||
|
import com.google.gerrit.extensions.restapi.BadRequestException;
|
||||||
|
import com.google.gerrit.extensions.restapi.Response;
|
||||||
|
import com.google.gerrit.extensions.restapi.RestModifyView;
|
||||||
|
import com.google.gerrit.extensions.restapi.TopLevelResource;
|
||||||
|
import com.google.gerrit.extensions.restapi.UnprocessableEntityException;
|
||||||
|
import com.google.gerrit.reviewdb.client.Branch;
|
||||||
|
import com.google.gerrit.reviewdb.client.Change;
|
||||||
|
import com.google.gerrit.reviewdb.client.PatchSet;
|
||||||
|
import com.google.gerrit.reviewdb.client.Project;
|
||||||
|
import com.google.gerrit.reviewdb.server.ReviewDb;
|
||||||
|
import com.google.gerrit.server.CurrentUser;
|
||||||
|
import com.google.gerrit.server.GerritPersonIdent;
|
||||||
|
import com.google.gerrit.server.IdentifiedUser;
|
||||||
|
import com.google.gerrit.server.api.changes.ChangeInfoMapper;
|
||||||
|
import com.google.gerrit.server.events.CommitReceivedEvent;
|
||||||
|
import com.google.gerrit.server.git.GitRepositoryManager;
|
||||||
|
import com.google.gerrit.server.git.MergeUtil;
|
||||||
|
import com.google.gerrit.server.git.validators.CommitValidationException;
|
||||||
|
import com.google.gerrit.server.git.validators.CommitValidators;
|
||||||
|
import com.google.gerrit.server.project.InvalidChangeOperationException;
|
||||||
|
import com.google.gerrit.server.project.ProjectResource;
|
||||||
|
import com.google.gerrit.server.project.ProjectsCollection;
|
||||||
|
import com.google.gerrit.server.project.RefControl;
|
||||||
|
import com.google.gerrit.server.ssh.NoSshInfo;
|
||||||
|
import com.google.gerrit.server.util.TimeUtil;
|
||||||
|
import com.google.gwtorm.server.OrmException;
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
import com.google.inject.Provider;
|
||||||
|
|
||||||
|
import org.eclipse.jgit.lib.CommitBuilder;
|
||||||
|
import org.eclipse.jgit.lib.Constants;
|
||||||
|
import org.eclipse.jgit.lib.ObjectId;
|
||||||
|
import org.eclipse.jgit.lib.ObjectInserter;
|
||||||
|
import org.eclipse.jgit.lib.PersonIdent;
|
||||||
|
import org.eclipse.jgit.lib.Ref;
|
||||||
|
import org.eclipse.jgit.lib.RefUpdate;
|
||||||
|
import org.eclipse.jgit.lib.Repository;
|
||||||
|
import org.eclipse.jgit.revwalk.RevCommit;
|
||||||
|
import org.eclipse.jgit.revwalk.RevWalk;
|
||||||
|
import org.eclipse.jgit.transport.ReceiveCommand;
|
||||||
|
import org.eclipse.jgit.util.ChangeIdUtil;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.UnsupportedEncodingException;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class CreateChange implements
|
||||||
|
RestModifyView<TopLevelResource, ChangeInfo> {
|
||||||
|
|
||||||
|
public static interface Factory {
|
||||||
|
CreateChange create();
|
||||||
|
}
|
||||||
|
|
||||||
|
private final ReviewDb db;
|
||||||
|
private final GitRepositoryManager gitManager;
|
||||||
|
private final PersonIdent myIdent;
|
||||||
|
private final Provider<CurrentUser> userProvider;
|
||||||
|
private final Provider<ProjectsCollection> projectsCollection;
|
||||||
|
private final CommitValidators.Factory commitValidatorsFactory;
|
||||||
|
private final ChangeInserter.Factory changeInserterFactory;
|
||||||
|
private final ChangeJson json;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
CreateChange(ReviewDb db,
|
||||||
|
GitRepositoryManager gitManager,
|
||||||
|
@GerritPersonIdent PersonIdent myIdent,
|
||||||
|
Provider<CurrentUser> userProvider,
|
||||||
|
Provider<ProjectsCollection> projectsCollection,
|
||||||
|
CommitValidators.Factory commitValidatorsFactory,
|
||||||
|
ChangeInserter.Factory changeInserterFactory,
|
||||||
|
ChangeJson json) {
|
||||||
|
this.db = db;
|
||||||
|
this.gitManager = gitManager;
|
||||||
|
this.myIdent = myIdent;
|
||||||
|
this.userProvider = userProvider;
|
||||||
|
this.projectsCollection = projectsCollection;
|
||||||
|
this.commitValidatorsFactory = commitValidatorsFactory;
|
||||||
|
this.changeInserterFactory = changeInserterFactory;
|
||||||
|
this.json = json;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Response<ChangeJson.ChangeInfo> apply(TopLevelResource parent,
|
||||||
|
ChangeInfo input) throws AuthException, OrmException,
|
||||||
|
BadRequestException, UnprocessableEntityException, IOException,
|
||||||
|
InvalidChangeOperationException {
|
||||||
|
|
||||||
|
if (Strings.isNullOrEmpty(input.project)) {
|
||||||
|
throw new BadRequestException("project must be non-empty");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Strings.isNullOrEmpty(input.branch)) {
|
||||||
|
throw new BadRequestException("branch must be non-empty");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Strings.isNullOrEmpty(input.subject)) {
|
||||||
|
throw new BadRequestException("commit message must be non-empty");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (input.status != null) {
|
||||||
|
if (input.status != ChangeStatus.NEW
|
||||||
|
&& input.status != ChangeStatus.DRAFT) {
|
||||||
|
throw new BadRequestException("unsupported change status");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
String refName = input.branch;
|
||||||
|
if (!refName.startsWith(Constants.R_REFS)) {
|
||||||
|
refName = Constants.R_HEADS + input.branch;
|
||||||
|
}
|
||||||
|
|
||||||
|
ProjectResource rsrc = projectsCollection.get().parse(input.project);
|
||||||
|
|
||||||
|
Capable r = rsrc.getControl().canPushToAtLeastOneRef();
|
||||||
|
if (r != Capable.OK) {
|
||||||
|
throw new AuthException(r.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
RefControl refControl = rsrc.getControl().controlForRef(refName);
|
||||||
|
if (!refControl.canUpload() || !refControl.canRead()) {
|
||||||
|
throw new AuthException("cannot upload review");
|
||||||
|
}
|
||||||
|
|
||||||
|
Project.NameKey project = rsrc.getNameKey();
|
||||||
|
Repository git = gitManager.openRepository(project);
|
||||||
|
|
||||||
|
try {
|
||||||
|
RevWalk rw = new RevWalk(git);
|
||||||
|
try {
|
||||||
|
Ref destRef = git.getRef(refName);
|
||||||
|
if (destRef == null) {
|
||||||
|
throw new UnprocessableEntityException(String.format(
|
||||||
|
"Branch %s does not exist.", refName));
|
||||||
|
}
|
||||||
|
|
||||||
|
IdentifiedUser me = (IdentifiedUser)userProvider.get();
|
||||||
|
PersonIdent author =
|
||||||
|
me.newCommitterIdent(myIdent.getWhen(), myIdent.getTimeZone());
|
||||||
|
|
||||||
|
RevCommit mergeTip = rw.parseCommit(destRef.getObjectId());
|
||||||
|
ObjectId id = ChangeIdUtil.computeChangeId(mergeTip.getTree(),
|
||||||
|
mergeTip, author, author, input.subject);
|
||||||
|
String commitMessage = ChangeIdUtil.insertId(input.subject, id);
|
||||||
|
|
||||||
|
RevCommit c = newCommit(git, rw, author, mergeTip, commitMessage);
|
||||||
|
|
||||||
|
Change change = new Change(
|
||||||
|
getChangeId(id, c),
|
||||||
|
new Change.Id(db.nextChangeId()),
|
||||||
|
me.getAccountId(),
|
||||||
|
new Branch.NameKey(project, destRef.getName()),
|
||||||
|
TimeUtil.nowTs());
|
||||||
|
|
||||||
|
ChangeInserter ins =
|
||||||
|
changeInserterFactory.create(refControl, change, c);
|
||||||
|
|
||||||
|
validateCommit(git, refControl, c, me, ins);
|
||||||
|
updateRef(git, rw, c, change, ins.getPatchSet());
|
||||||
|
|
||||||
|
change.setTopic(input.topic);
|
||||||
|
change.setStatus(ChangeInfoMapper.changeStatus2Status(input.status));
|
||||||
|
ins.insert();
|
||||||
|
|
||||||
|
return Response.created(json.format(change.getId()));
|
||||||
|
} finally {
|
||||||
|
rw.release();
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
git.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void validateCommit(Repository git, RefControl refControl,
|
||||||
|
RevCommit c, IdentifiedUser me, ChangeInserter ins)
|
||||||
|
throws InvalidChangeOperationException {
|
||||||
|
PatchSet newPatchSet = ins.getPatchSet();
|
||||||
|
CommitValidators commitValidators =
|
||||||
|
commitValidatorsFactory.create(refControl, new NoSshInfo(), git);
|
||||||
|
CommitReceivedEvent commitReceivedEvent =
|
||||||
|
new CommitReceivedEvent(new ReceiveCommand(
|
||||||
|
ObjectId.zeroId(),
|
||||||
|
c.getId(),
|
||||||
|
newPatchSet.getRefName()),
|
||||||
|
refControl.getProjectControl().getProject(),
|
||||||
|
refControl.getRefName(),
|
||||||
|
c,
|
||||||
|
me);
|
||||||
|
|
||||||
|
try {
|
||||||
|
commitValidators.validateForGerritCommits(commitReceivedEvent);
|
||||||
|
} catch (CommitValidationException e) {
|
||||||
|
throw new InvalidChangeOperationException(e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void updateRef(Repository git, RevWalk rw, RevCommit c,
|
||||||
|
Change change, PatchSet newPatchSet) throws IOException {
|
||||||
|
RefUpdate ru = git.updateRef(newPatchSet.getRefName());
|
||||||
|
ru.setExpectedOldObjectId(ObjectId.zeroId());
|
||||||
|
ru.setNewObjectId(c);
|
||||||
|
ru.disableRefLog();
|
||||||
|
if (ru.update(rw) != RefUpdate.Result.NEW) {
|
||||||
|
throw new IOException(String.format(
|
||||||
|
"Failed to create ref %s in %s: %s", newPatchSet.getRefName(),
|
||||||
|
change.getDest().getParentKey().get(), ru.getResult()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Change.Key getChangeId(ObjectId id, RevCommit emptyCommit) {
|
||||||
|
List<String> idList = emptyCommit.getFooterLines(
|
||||||
|
MergeUtil.CHANGE_ID);
|
||||||
|
Change.Key changeKey = !idList.isEmpty()
|
||||||
|
? new Change.Key(idList.get(idList.size() - 1).trim())
|
||||||
|
: new Change.Key("I" + id.name());
|
||||||
|
return changeKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static RevCommit newCommit(Repository git, RevWalk rw,
|
||||||
|
PersonIdent authorIdent, RevCommit mergeTip, String commitMessage)
|
||||||
|
throws IOException {
|
||||||
|
RevCommit emptyCommit;
|
||||||
|
ObjectInserter oi = git.newObjectInserter();
|
||||||
|
try {
|
||||||
|
CommitBuilder commit = new CommitBuilder();
|
||||||
|
commit.setTreeId(mergeTip.getTree().getId());
|
||||||
|
commit.setParentId(mergeTip);
|
||||||
|
commit.setAuthor(authorIdent);
|
||||||
|
commit.setCommitter(authorIdent);
|
||||||
|
commit.setMessage(commitMessage);
|
||||||
|
emptyCommit = rw.parseCommit(insert(oi, commit));
|
||||||
|
} finally {
|
||||||
|
oi.release();
|
||||||
|
}
|
||||||
|
return emptyCommit;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static ObjectId insert(ObjectInserter inserter,
|
||||||
|
CommitBuilder commit) throws IOException,
|
||||||
|
UnsupportedEncodingException {
|
||||||
|
ObjectId id = inserter.insert(commit);
|
||||||
|
inserter.flush();
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
}
|
@ -107,6 +107,7 @@ public class Module extends RestApiModule {
|
|||||||
factory(EmailReviewComments.Factory.class);
|
factory(EmailReviewComments.Factory.class);
|
||||||
factory(ChangeInserter.Factory.class);
|
factory(ChangeInserter.Factory.class);
|
||||||
factory(PatchSetInserter.Factory.class);
|
factory(PatchSetInserter.Factory.class);
|
||||||
|
factory(CreateChange.Factory.class);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -78,7 +78,11 @@ import java.util.Set;
|
|||||||
import java.util.TimeZone;
|
import java.util.TimeZone;
|
||||||
|
|
||||||
public class MergeUtil {
|
public class MergeUtil {
|
||||||
|
public static final FooterKey CHANGE_ID = new FooterKey("Change-Id");
|
||||||
|
private static final FooterKey REVIEWED_ON = new FooterKey("Reviewed-on");
|
||||||
private static final Logger log = LoggerFactory.getLogger(MergeUtil.class);
|
private static final Logger log = LoggerFactory.getLogger(MergeUtil.class);
|
||||||
|
private static final String R_HEADS_MASTER =
|
||||||
|
Constants.R_HEADS + Constants.MASTER;
|
||||||
|
|
||||||
public static boolean useRecursiveMerge(Config cfg) {
|
public static boolean useRecursiveMerge(Config cfg) {
|
||||||
return cfg.getBoolean("core", null, "useRecursiveMerge", false);
|
return cfg.getBoolean("core", null, "useRecursiveMerge", false);
|
||||||
@ -89,12 +93,6 @@ public class MergeUtil {
|
|||||||
MergeUtil create(ProjectState project, boolean useContentMerge);
|
MergeUtil create(ProjectState project, boolean useContentMerge);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static final String R_HEADS_MASTER =
|
|
||||||
Constants.R_HEADS + Constants.MASTER;
|
|
||||||
|
|
||||||
private static final FooterKey REVIEWED_ON = new FooterKey("Reviewed-on");
|
|
||||||
private static final FooterKey CHANGE_ID = new FooterKey("Change-Id");
|
|
||||||
|
|
||||||
private final Provider<ReviewDb> db;
|
private final Provider<ReviewDb> db;
|
||||||
private final IdentifiedUser.GenericFactory identifiedUserFactory;
|
private final IdentifiedUser.GenericFactory identifiedUserFactory;
|
||||||
private final Provider<String> urlProvider;
|
private final Provider<String> urlProvider;
|
||||||
|
Loading…
Reference in New Issue
Block a user