// Copyright (C) 2012 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.common.util.concurrent.CheckedFuture; import com.google.gerrit.common.ChangeHooks; import com.google.gerrit.extensions.api.changes.RestoreInput; import com.google.gerrit.extensions.restapi.AuthException; import com.google.gerrit.extensions.restapi.ResourceConflictException; import com.google.gerrit.extensions.restapi.RestModifyView; import com.google.gerrit.extensions.webui.UiAction; import com.google.gerrit.reviewdb.client.Change; import com.google.gerrit.reviewdb.client.Change.Status; import com.google.gerrit.reviewdb.client.ChangeMessage; import com.google.gerrit.reviewdb.server.ReviewDb; import com.google.gerrit.server.ChangeMessagesUtil; import com.google.gerrit.server.ChangeUtil; import com.google.gerrit.server.IdentifiedUser; import com.google.gerrit.server.change.ChangeJson.ChangeInfo; import com.google.gerrit.server.mail.ReplyToChangeSender; import com.google.gerrit.server.mail.RestoredSender; import com.google.gerrit.server.notedb.ChangeUpdate; import com.google.gerrit.server.project.ChangeControl; import com.google.gwtorm.server.AtomicUpdate; import com.google.gwtorm.server.OrmException; import com.google.inject.Inject; import com.google.inject.Provider; import com.google.inject.Singleton; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.io.IOException; @Singleton public class Restore implements RestModifyView<ChangeResource, RestoreInput>, UiAction<ChangeResource> { private static final Logger log = LoggerFactory.getLogger(Restore.class); private final ChangeHooks hooks; private final RestoredSender.Factory restoredSenderFactory; private final Provider<ReviewDb> dbProvider; private final ChangeJson json; private final MergeabilityChecker mergeabilityChecker; private final ChangeMessagesUtil cmUtil; private final ChangeUpdate.Factory updateFactory; @Inject Restore(ChangeHooks hooks, RestoredSender.Factory restoredSenderFactory, Provider<ReviewDb> dbProvider, ChangeJson json, MergeabilityChecker mergeabilityChecker, ChangeMessagesUtil cmUtil, ChangeUpdate.Factory updateFactory) { this.hooks = hooks; this.restoredSenderFactory = restoredSenderFactory; this.dbProvider = dbProvider; this.json = json; this.mergeabilityChecker = mergeabilityChecker; this.cmUtil = cmUtil; this.updateFactory = updateFactory; } @Override public ChangeInfo apply(ChangeResource req, RestoreInput input) throws AuthException, ResourceConflictException, OrmException, IOException { ChangeControl control = req.getControl(); IdentifiedUser caller = (IdentifiedUser) control.getCurrentUser(); Change change = req.getChange(); if (!control.canRestore()) { throw new AuthException("restore not permitted"); } else if (change.getStatus() != Status.ABANDONED) { throw new ResourceConflictException("change is " + status(change)); } ChangeMessage message; ChangeUpdate update; ReviewDb db = dbProvider.get(); db.changes().beginTransaction(change.getId()); try { change = db.changes().atomicUpdate( change.getId(), new AtomicUpdate<Change>() { @Override public Change update(Change change) { if (change.getStatus() == Status.ABANDONED) { change.setStatus(Status.NEW); ChangeUtil.updated(change); return change; } return null; } }); if (change == null) { throw new ResourceConflictException("change is " + status(db.changes().get(req.getChange().getId()))); } //TODO(yyonas): atomic update was not propagated update = updateFactory.create(control); message = newMessage(input, caller, change); cmUtil.addChangeMessage(db, update, message); db.commit(); } finally { db.rollback(); } update.commit(); CheckedFuture<?, IOException> f = mergeabilityChecker.newCheck() .addChange(change) .reindex() .runAsync(); try { ReplyToChangeSender cm = restoredSenderFactory.create(change); cm.setFrom(caller.getAccountId()); cm.setChangeMessage(message); cm.send(); } catch (Exception e) { log.error("Cannot email update for change " + change.getChangeId(), e); } hooks.doChangeRestoredHook(change, caller.getAccount(), db.patchSets().get(change.currentPatchSetId()), Strings.emptyToNull(input.message), dbProvider.get()); ChangeInfo result = json.format(change); f.checkedGet(); return result; } @Override public UiAction.Description getDescription(ChangeResource resource) { return new UiAction.Description() .setLabel("Restore") .setTitle("Restore the change") .setVisible(resource.getChange().getStatus() == Status.ABANDONED && resource.getControl().canRestore()); } private ChangeMessage newMessage(RestoreInput input, IdentifiedUser caller, Change change) throws OrmException { StringBuilder msg = new StringBuilder(); msg.append("Restored"); if (!Strings.nullToEmpty(input.message).trim().isEmpty()) { msg.append("\n\n"); msg.append(input.message.trim()); } ChangeMessage message = new ChangeMessage( new ChangeMessage.Key( change.getId(), ChangeUtil.messageUUID(dbProvider.get())), caller.getAccountId(), change.getLastUpdatedOn(), change.currentPatchSetId()); message.setMessage(msg.toString()); return message; } private static String status(Change change) { return change != null ? change.getStatus().name().toLowerCase() : "deleted"; } }