All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.gerrit.server.change.EmailReviewComments Maven / Gradle / Ivy

There is a newer version: 3.11.1
Show newest version
// 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 static com.google.gerrit.server.CommentsUtil.COMMENT_ORDER;
import static com.google.gerrit.server.mail.EmailFactories.COMMENTS_ADDED;

import com.google.common.collect.ImmutableList;
import com.google.common.flogger.FluentLogger;
import com.google.gerrit.common.Nullable;
import com.google.gerrit.entities.Change;
import com.google.gerrit.entities.Comment;
import com.google.gerrit.entities.PatchSet;
import com.google.gerrit.entities.Project;
import com.google.gerrit.entities.SubmitRequirement;
import com.google.gerrit.entities.SubmitRequirementResult;
import com.google.gerrit.server.CurrentUser;
import com.google.gerrit.server.IdentifiedUser;
import com.google.gerrit.server.config.SendEmailExecutor;
import com.google.gerrit.server.mail.EmailFactories;
import com.google.gerrit.server.mail.send.ChangeEmail;
import com.google.gerrit.server.mail.send.CommentChangeEmailDecorator;
import com.google.gerrit.server.mail.send.MessageIdGenerator;
import com.google.gerrit.server.mail.send.MessageIdGenerator.MessageId;
import com.google.gerrit.server.mail.send.OutgoingEmail;
import com.google.gerrit.server.patch.PatchSetInfoFactory;
import com.google.gerrit.server.update.PostUpdateContext;
import com.google.gerrit.server.util.LabelVote;
import com.google.gerrit.server.util.RequestContext;
import com.google.gerrit.server.util.ThreadLocalRequestContext;
import com.google.inject.Inject;
import com.google.inject.assistedinject.Assisted;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.time.Instant;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;
import org.eclipse.jgit.lib.ObjectId;

public class EmailReviewComments {
  private static final FluentLogger logger = FluentLogger.forEnclosingClass();

  public interface Factory {
    // TODO(dborowitz/wyatta): Rationalize these arguments so HTML and text templates are operating
    // on the same set of inputs.
    /**
     * Creates handle for sending email
     *
     * @param postUpdateContext the post update context from the calling BatchUpdateOp
     * @param patchSet patch set corresponding to the top-level op
     * @param preUpdateMetaId the SHA1 to which the notes branch pointed before the update
     * @param message used by text template only. The contents of this message typically include the
     *     "Patch set N" header and "(M comments)".
     * @param comments inline comments.
     * @param patchSetComment used by HTML template only: some quasi-human-generated text. The
     *     contents should *not* include a "Patch set N" header or "(M comments)" footer, as these
     *     will be added automatically in soy in a structured way.
     * @param labels labels applied as part of this review operation.
     */
    EmailReviewComments create(
        PostUpdateContext postUpdateContext,
        PatchSet patchSet,
        ObjectId preUpdateMetaId,
        @Assisted("message") String message,
        List comments,
        @Nullable @Assisted("patchSetComment") String patchSetComment,
        List labels);
  }

  private final ExecutorService sendEmailsExecutor;
  private final AsyncSender asyncSender;

  @Inject
  EmailReviewComments(
      @SendEmailExecutor ExecutorService executor,
      PatchSetInfoFactory patchSetInfoFactory,
      EmailFactories emailFactories,
      ThreadLocalRequestContext requestContext,
      MessageIdGenerator messageIdGenerator,
      @Assisted PostUpdateContext postUpdateContext,
      @Assisted PatchSet patchSet,
      @Assisted ObjectId preUpdateMetaId,
      @Assisted("message") String message,
      @Assisted List comments,
      @Nullable @Assisted("patchSetComment") String patchSetComment,
      @Assisted List labels) {
    this.sendEmailsExecutor = executor;

    MessageId messageId;
    try {
      messageId =
          messageIdGenerator.fromChangeUpdateAndReason(
              postUpdateContext.getRepoView(), patchSet.id(), "EmailReviewComments");
    } catch (IOException e) {
      throw new UncheckedIOException(e);
    }

    Change.Id changeId = patchSet.id().changeId();

    // Getting the change data from PostUpdateContext retrieves a cached ChangeData
    // instance. This ChangeData instance has been created when the change was (re)indexed
    // due to the update, and hence has submit requirement results already cached (since
    // (re)indexing triggers the evaluation of the submit requirements).
    Map postUpdateSubmitRequirementResults =
        postUpdateContext
            .getChangeData(postUpdateContext.getProject(), changeId)
            .submitRequirementsIncludingLegacy();
    this.asyncSender =
        new AsyncSender(
            requestContext,
            emailFactories,
            patchSetInfoFactory,
            postUpdateContext.getUser().asIdentifiedUser(),
            messageId,
            postUpdateContext.getNotify(changeId),
            postUpdateContext.getProject(),
            changeId,
            patchSet,
            preUpdateMetaId,
            message,
            postUpdateContext.getWhen(),
            ImmutableList.copyOf(COMMENT_ORDER.sortedCopy(comments)),
            patchSetComment,
            ImmutableList.copyOf(labels),
            postUpdateSubmitRequirementResults);
  }

  public void sendAsync() {
    @SuppressWarnings("unused")
    Future possiblyIgnoredError = sendEmailsExecutor.submit(asyncSender);
  }

  /**
   * {@link Runnable} that sends the email asynchonously.
   *
   * 

Only pass objects into this class that are thread-safe (e.g. immutable) so that they can be * safely accessed from the background thread. */ // TODO: The passed in Comment class is not thread-safe, replace it with an AutoValue type. private static class AsyncSender implements Runnable, RequestContext { private final ThreadLocalRequestContext requestContext; private final EmailFactories emailFactories; private final PatchSetInfoFactory patchSetInfoFactory; private final IdentifiedUser user; private final MessageId messageId; private final NotifyResolver.Result notify; private final Project.NameKey projectName; private final Change.Id changeId; private final PatchSet patchSet; private final ObjectId preUpdateMetaId; private final String message; private final Instant timestamp; private final ImmutableList comments; @Nullable private final String patchSetComment; private final ImmutableList labels; private final Map postUpdateSubmitRequirementResults; AsyncSender( ThreadLocalRequestContext requestContext, EmailFactories emailFactories, PatchSetInfoFactory patchSetInfoFactory, IdentifiedUser user, MessageId messageId, NotifyResolver.Result notify, Project.NameKey projectName, Change.Id changeId, PatchSet patchSet, ObjectId preUpdateMetaId, String message, Instant timestamp, ImmutableList comments, @Nullable String patchSetComment, ImmutableList labels, Map postUpdateSubmitRequirementResults) { this.requestContext = requestContext; this.emailFactories = emailFactories; this.patchSetInfoFactory = patchSetInfoFactory; this.user = user; this.messageId = messageId; this.notify = notify; this.projectName = projectName; this.changeId = changeId; this.patchSet = patchSet; this.preUpdateMetaId = preUpdateMetaId; this.message = message; this.timestamp = timestamp; this.comments = comments; this.patchSetComment = patchSetComment; this.labels = labels; this.postUpdateSubmitRequirementResults = postUpdateSubmitRequirementResults; } @Override public void run() { RequestContext old = requestContext.setContext(this); try { CommentChangeEmailDecorator commentChangeEmail = emailFactories.createCommentChangeEmail( projectName, changeId, preUpdateMetaId, postUpdateSubmitRequirementResults); commentChangeEmail.setComments(comments); commentChangeEmail.setPatchSetComment(patchSetComment); commentChangeEmail.setLabels(labels); ChangeEmail changeEmail = emailFactories.createChangeEmail(projectName, changeId, commentChangeEmail); changeEmail.setPatchSet(patchSet, patchSetInfoFactory.get(projectName, patchSet)); changeEmail.setChangeMessage(message, timestamp); OutgoingEmail outgoingEmail = emailFactories.createOutgoingEmail(COMMENTS_ADDED, changeEmail); outgoingEmail.setFrom(user.getAccountId()); outgoingEmail.setNotify(notify); outgoingEmail.setMessageId(messageId); outgoingEmail.send(); } catch (Exception e) { logger.atSevere().withCause(e).log("Cannot email comments for %s", patchSet.id()); } finally { @SuppressWarnings("unused") var unused = requestContext.setContext(old); } } @Override public String toString() { return "send-email comments"; } @Override public CurrentUser getUser() { return user.getRealUser(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy