| // Copyright (C) 2009 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.httpd.rpc.changedetail; |
| |
| import com.google.gerrit.common.data.LabelType; |
| import com.google.gerrit.common.data.PatchSetPublishDetail; |
| import com.google.gerrit.common.data.PermissionRange; |
| import com.google.gerrit.common.data.SubmitRecord; |
| import com.google.gerrit.httpd.rpc.Handler; |
| import com.google.gerrit.reviewdb.client.Change; |
| import com.google.gerrit.reviewdb.client.PatchLineComment; |
| import com.google.gerrit.reviewdb.client.PatchSet; |
| import com.google.gerrit.reviewdb.client.PatchSetInfo; |
| import com.google.gerrit.reviewdb.server.ReviewDb; |
| import com.google.gerrit.server.IdentifiedUser; |
| import com.google.gerrit.server.account.AccountInfoCacheFactory; |
| import com.google.gerrit.server.patch.PatchSetInfoFactory; |
| import com.google.gerrit.server.patch.PatchSetInfoNotAvailableException; |
| import com.google.gerrit.server.project.ChangeControl; |
| import com.google.gerrit.server.project.NoSuchChangeException; |
| import com.google.gwtorm.server.OrmException; |
| import com.google.inject.Inject; |
| import com.google.inject.assistedinject.Assisted; |
| |
| import java.util.HashMap; |
| import java.util.List; |
| import java.util.Map; |
| |
| final class PatchSetPublishDetailFactory extends Handler<PatchSetPublishDetail> { |
| interface Factory { |
| PatchSetPublishDetailFactory create(PatchSet.Id patchSetId); |
| } |
| |
| private final PatchSetInfoFactory infoFactory; |
| private final ReviewDb db; |
| private final ChangeControl.Factory changeControlFactory; |
| private final AccountInfoCacheFactory aic; |
| private final IdentifiedUser user; |
| |
| private final PatchSet.Id patchSetId; |
| |
| private PatchSetInfo patchSetInfo; |
| private Change change; |
| private List<PatchLineComment> drafts; |
| |
| @Inject |
| PatchSetPublishDetailFactory(final PatchSetInfoFactory infoFactory, |
| final ReviewDb db, |
| final AccountInfoCacheFactory.Factory accountInfoCacheFactory, |
| final ChangeControl.Factory changeControlFactory, |
| final IdentifiedUser user, @Assisted final PatchSet.Id patchSetId) { |
| this.infoFactory = infoFactory; |
| this.db = db; |
| this.changeControlFactory = changeControlFactory; |
| this.aic = accountInfoCacheFactory.create(); |
| this.user = user; |
| |
| this.patchSetId = patchSetId; |
| } |
| |
| @Override |
| public PatchSetPublishDetail call() throws OrmException, |
| PatchSetInfoNotAvailableException, NoSuchChangeException { |
| final Change.Id changeId = patchSetId.getParentKey(); |
| final ChangeControl control = changeControlFactory.validateFor(changeId); |
| change = control.getChange(); |
| PatchSet patchSet = db.patchSets().get(patchSetId); |
| patchSetInfo = infoFactory.get(change, patchSet); |
| drafts = db.patchComments().draftByPatchSetAuthor(patchSetId, user.getAccountId()).toList(); |
| |
| aic.want(change.getOwner()); |
| |
| PatchSetPublishDetail detail = new PatchSetPublishDetail(); |
| detail.setPatchSetInfo(patchSetInfo); |
| detail.setChange(change); |
| detail.setDrafts(drafts); |
| |
| if (change.getStatus().isOpen() |
| && patchSetId.equals(change.currentPatchSetId())) { |
| // TODO Push this selection of labels down into the Prolog interpreter. |
| // Ideally we discover the labels the user can apply here based on doing |
| // a findall() over the space of labels they can apply combined against |
| // the submit rule, thereby skipping any mutually exclusive cases. However |
| // those are not common, so it might just be reasonable to take this |
| // simple approach. |
| |
| Map<String, PermissionRange> rangeByName = |
| new HashMap<String, PermissionRange>(); |
| for (PermissionRange r : control.getLabelRanges()) { |
| if (r.isLabel()) { |
| rangeByName.put(r.getLabel(), r); |
| } |
| } |
| |
| boolean couldSubmit = false; |
| List<SubmitRecord> submitRecords = control.canSubmit(db, patchSet); |
| for (SubmitRecord rec : submitRecords) { |
| if (rec.status == SubmitRecord.Status.OK) { |
| couldSubmit = true; |
| } |
| |
| if (rec.labels != null) { |
| int ok = 0; |
| |
| for (SubmitRecord.Label lbl : rec.labels) { |
| aic.want(lbl.appliedBy); |
| |
| boolean canMakeOk = false; |
| PermissionRange range = rangeByName.get(lbl.label); |
| if (range != null) { |
| LabelType lt = control.getLabelTypes().byLabel(lbl.label); |
| if (lt != null && lt.getMax().getValue() == range.getMax()) { |
| canMakeOk = true; |
| } |
| } |
| |
| switch (lbl.status) { |
| case OK: |
| case MAY: |
| ok++; |
| break; |
| |
| case NEED: |
| if (canMakeOk) { |
| ok++; |
| } |
| break; |
| |
| case IMPOSSIBLE: |
| case REJECT: |
| break; |
| } |
| } |
| |
| if (rec.status == SubmitRecord.Status.NOT_READY |
| && ok == rec.labels.size()) { |
| couldSubmit = true; |
| } |
| } |
| } |
| |
| if (couldSubmit && control.getRefControl().canSubmit()) { |
| detail.setCanSubmit(true); |
| } |
| } |
| |
| detail.setSubmitTypeRecord(control.getSubmitTypeRecord(db, patchSet)); |
| detail.setAccounts(aic.create()); |
| |
| return detail; |
| } |
| } |