Merge "Send email notification to user that is assigned to a change"
This commit is contained in:
@@ -120,6 +120,12 @@ The Reverted templates will determine the contents of the email related to a
|
|||||||
change being reverted. It is a `ChangeEmail`: see `ChangeSubject.soy` and
|
change being reverted. It is a `ChangeEmail`: see `ChangeSubject.soy` and
|
||||||
ChangeFooter.
|
ChangeFooter.
|
||||||
|
|
||||||
|
=== SetAssignee.soy and SetAssigneeHtml.soy
|
||||||
|
|
||||||
|
The SetAssignee templates will determine the contents of the email related to a
|
||||||
|
user being assigned to a change. It is a `ChangeEmail`: see `ChangeSubject.soy`
|
||||||
|
and ChangeFooter.
|
||||||
|
|
||||||
|
|
||||||
== Mail Variables and Methods
|
== Mail Variables and Methods
|
||||||
|
|
||||||
|
@@ -25,6 +25,7 @@ import com.google.gerrit.acceptance.PushOneCommit;
|
|||||||
import com.google.gerrit.extensions.api.changes.AssigneeInput;
|
import com.google.gerrit.extensions.api.changes.AssigneeInput;
|
||||||
import com.google.gerrit.extensions.client.ReviewerState;
|
import com.google.gerrit.extensions.client.ReviewerState;
|
||||||
import com.google.gerrit.extensions.common.AccountInfo;
|
import com.google.gerrit.extensions.common.AccountInfo;
|
||||||
|
import com.google.gerrit.testutil.FakeEmailSender.Message;
|
||||||
import com.google.gerrit.testutil.TestTimeUtil;
|
import com.google.gerrit.testutil.TestTimeUtil;
|
||||||
|
|
||||||
import org.junit.AfterClass;
|
import org.junit.AfterClass;
|
||||||
@@ -59,6 +60,10 @@ public class AssigneeIT extends AbstractDaemonTest {
|
|||||||
assertThat(setAssignee(r, user.email)._accountId)
|
assertThat(setAssignee(r, user.email)._accountId)
|
||||||
.isEqualTo(user.getId().get());
|
.isEqualTo(user.getId().get());
|
||||||
assertThat(getAssignee(r)._accountId).isEqualTo(user.getId().get());
|
assertThat(getAssignee(r)._accountId).isEqualTo(user.getId().get());
|
||||||
|
|
||||||
|
assertThat(sender.getMessages()).hasSize(1);
|
||||||
|
Message m = sender.getMessages().get(0);
|
||||||
|
assertThat(m.rcpt()).containsExactly(user.emailAddress);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@@ -66,7 +71,7 @@ public class AssigneeIT extends AbstractDaemonTest {
|
|||||||
PushOneCommit.Result r = createChange();
|
PushOneCommit.Result r = createChange();
|
||||||
setAssignee(r, user.email);
|
setAssignee(r, user.email);
|
||||||
assertThat(setAssignee(r, user.email)._accountId)
|
assertThat(setAssignee(r, user.email)._accountId)
|
||||||
.isEqualTo(user.getId().get());
|
.isEqualTo(user.getId().get());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@@ -127,6 +127,8 @@ public class SitePathInitializer {
|
|||||||
extractMailExample("RestoredHtml.soy");
|
extractMailExample("RestoredHtml.soy");
|
||||||
extractMailExample("Reverted.soy");
|
extractMailExample("Reverted.soy");
|
||||||
extractMailExample("RevertedHtml.soy");
|
extractMailExample("RevertedHtml.soy");
|
||||||
|
extractMailExample("SetAssignee.soy");
|
||||||
|
extractMailExample("SetAssigneeHtml.soy");
|
||||||
|
|
||||||
if (!ui.isBatch()) {
|
if (!ui.isBatch()) {
|
||||||
System.err.println();
|
System.err.println();
|
||||||
|
@@ -32,16 +32,24 @@ import com.google.gerrit.server.config.AnonymousCowardName;
|
|||||||
import com.google.gerrit.server.extensions.events.AssigneeChanged;
|
import com.google.gerrit.server.extensions.events.AssigneeChanged;
|
||||||
import com.google.gerrit.server.git.BatchUpdate;
|
import com.google.gerrit.server.git.BatchUpdate;
|
||||||
import com.google.gerrit.server.git.BatchUpdate.Context;
|
import com.google.gerrit.server.git.BatchUpdate.Context;
|
||||||
|
import com.google.gerrit.server.mail.send.SetAssigneeSender;
|
||||||
import com.google.gerrit.server.notedb.ChangeUpdate;
|
import com.google.gerrit.server.notedb.ChangeUpdate;
|
||||||
import com.google.gerrit.server.validators.AssigneeValidationListener;
|
import com.google.gerrit.server.validators.AssigneeValidationListener;
|
||||||
import com.google.gerrit.server.validators.ValidationException;
|
import com.google.gerrit.server.validators.ValidationException;
|
||||||
import com.google.gwtorm.server.OrmException;
|
import com.google.gwtorm.server.OrmException;
|
||||||
|
import com.google.inject.Provider;
|
||||||
import com.google.inject.assistedinject.Assisted;
|
import com.google.inject.assistedinject.Assisted;
|
||||||
import com.google.inject.assistedinject.AssistedInject;
|
import com.google.inject.assistedinject.AssistedInject;
|
||||||
|
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
|
|
||||||
public class SetAssigneeOp extends BatchUpdate.Op {
|
public class SetAssigneeOp extends BatchUpdate.Op {
|
||||||
|
private static final Logger log =
|
||||||
|
LoggerFactory.getLogger(SetAssigneeOp.class);
|
||||||
|
|
||||||
public interface Factory {
|
public interface Factory {
|
||||||
SetAssigneeOp create(String assignee);
|
SetAssigneeOp create(String assignee);
|
||||||
}
|
}
|
||||||
@@ -53,6 +61,8 @@ public class SetAssigneeOp extends BatchUpdate.Op {
|
|||||||
private final String assignee;
|
private final String assignee;
|
||||||
private final String anonymousCowardName;
|
private final String anonymousCowardName;
|
||||||
private final AssigneeChanged assigneeChanged;
|
private final AssigneeChanged assigneeChanged;
|
||||||
|
private final SetAssigneeSender.Factory setAssigneeSenderFactory;
|
||||||
|
private final Provider<IdentifiedUser> user;
|
||||||
|
|
||||||
private Change change;
|
private Change change;
|
||||||
private Account newAssignee;
|
private Account newAssignee;
|
||||||
@@ -63,8 +73,10 @@ public class SetAssigneeOp extends BatchUpdate.Op {
|
|||||||
ChangeMessagesUtil cmUtil,
|
ChangeMessagesUtil cmUtil,
|
||||||
AccountInfoCacheFactory.Factory accountInfosFactory,
|
AccountInfoCacheFactory.Factory accountInfosFactory,
|
||||||
DynamicSet<AssigneeValidationListener> validationListeners,
|
DynamicSet<AssigneeValidationListener> validationListeners,
|
||||||
AssigneeChanged assigneeChanged,
|
|
||||||
@AnonymousCowardName String anonymousCowardName,
|
@AnonymousCowardName String anonymousCowardName,
|
||||||
|
AssigneeChanged assigneeChanged,
|
||||||
|
SetAssigneeSender.Factory setAssigneeSenderFactory,
|
||||||
|
Provider<IdentifiedUser> user,
|
||||||
@Assisted String assignee) {
|
@Assisted String assignee) {
|
||||||
this.accounts = accounts;
|
this.accounts = accounts;
|
||||||
this.cmUtil = cmUtil;
|
this.cmUtil = cmUtil;
|
||||||
@@ -73,6 +85,8 @@ public class SetAssigneeOp extends BatchUpdate.Op {
|
|||||||
this.assigneeChanged = assigneeChanged;
|
this.assigneeChanged = assigneeChanged;
|
||||||
this.anonymousCowardName = anonymousCowardName;
|
this.anonymousCowardName = anonymousCowardName;
|
||||||
this.assignee = checkNotNull(assignee);
|
this.assignee = checkNotNull(assignee);
|
||||||
|
this.setAssigneeSenderFactory = setAssigneeSenderFactory;
|
||||||
|
this.user = user;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -138,6 +152,15 @@ public class SetAssigneeOp extends BatchUpdate.Op {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void postUpdate(Context ctx) throws OrmException {
|
public void postUpdate(Context ctx) throws OrmException {
|
||||||
|
try {
|
||||||
|
SetAssigneeSender cm = setAssigneeSenderFactory
|
||||||
|
.create(change.getProject(), change.getId(), newAssignee.getId());
|
||||||
|
cm.setFrom(user.get().getAccountId());
|
||||||
|
cm.send();
|
||||||
|
} catch (Exception err) {
|
||||||
|
log.error("Cannot send email to new assignee of change " + change.getId(),
|
||||||
|
err);
|
||||||
|
}
|
||||||
assigneeChanged.fire(change, ctx.getAccount(), oldAssignee, ctx.getWhen());
|
assigneeChanged.fire(change, ctx.getAccount(), oldAssignee, ctx.getWhen());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -143,6 +143,7 @@ import com.google.gerrit.server.mail.send.MailTemplates;
|
|||||||
import com.google.gerrit.server.mail.send.MergedSender;
|
import com.google.gerrit.server.mail.send.MergedSender;
|
||||||
import com.google.gerrit.server.mail.send.RegisterNewEmailSender;
|
import com.google.gerrit.server.mail.send.RegisterNewEmailSender;
|
||||||
import com.google.gerrit.server.mail.send.ReplacePatchSetSender;
|
import com.google.gerrit.server.mail.send.ReplacePatchSetSender;
|
||||||
|
import com.google.gerrit.server.mail.send.SetAssigneeSender;
|
||||||
import com.google.gerrit.server.mail.send.VelocityRuntimeProvider;
|
import com.google.gerrit.server.mail.send.VelocityRuntimeProvider;
|
||||||
import com.google.gerrit.server.mime.FileTypeRegistry;
|
import com.google.gerrit.server.mime.FileTypeRegistry;
|
||||||
import com.google.gerrit.server.mime.MimeUtilFileTypeRegistry;
|
import com.google.gerrit.server.mime.MimeUtilFileTypeRegistry;
|
||||||
@@ -259,6 +260,7 @@ public class GerritGlobalModule extends FactoryModule {
|
|||||||
factory(ProjectState.Factory.class);
|
factory(ProjectState.Factory.class);
|
||||||
factory(RegisterNewEmailSender.Factory.class);
|
factory(RegisterNewEmailSender.Factory.class);
|
||||||
factory(ReplacePatchSetSender.Factory.class);
|
factory(ReplacePatchSetSender.Factory.class);
|
||||||
|
factory(SetAssigneeSender.Factory.class);
|
||||||
bind(PermissionCollection.Factory.class);
|
bind(PermissionCollection.Factory.class);
|
||||||
bind(AccountVisibility.class)
|
bind(AccountVisibility.class)
|
||||||
.toProvider(AccountVisibilityProvider.class)
|
.toProvider(AccountVisibilityProvider.class)
|
||||||
|
@@ -66,6 +66,8 @@ public class MailSoyTofuProvider implements Provider<SoyTofu> {
|
|||||||
"RestoredHtml.soy",
|
"RestoredHtml.soy",
|
||||||
"Reverted.soy",
|
"Reverted.soy",
|
||||||
"RevertedHtml.soy",
|
"RevertedHtml.soy",
|
||||||
|
"SetAssignee.soy",
|
||||||
|
"SetAssigneeHtml.soy",
|
||||||
};
|
};
|
||||||
|
|
||||||
private final SitePaths site;
|
private final SitePaths site;
|
||||||
|
@@ -0,0 +1,73 @@
|
|||||||
|
// Copyright (C) 2016 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.mail.send;
|
||||||
|
|
||||||
|
import com.google.gerrit.common.errors.EmailException;
|
||||||
|
import com.google.gerrit.reviewdb.client.Account;
|
||||||
|
import com.google.gerrit.reviewdb.client.Change;
|
||||||
|
import com.google.gerrit.reviewdb.client.Project;
|
||||||
|
import com.google.gerrit.server.mail.RecipientType;
|
||||||
|
import com.google.gwtorm.server.OrmException;
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
import com.google.inject.assistedinject.Assisted;
|
||||||
|
|
||||||
|
public class SetAssigneeSender extends ChangeEmail {
|
||||||
|
public interface Factory {
|
||||||
|
SetAssigneeSender create(Project.NameKey project, Change.Id id,
|
||||||
|
Account.Id assignee);
|
||||||
|
}
|
||||||
|
|
||||||
|
private final Account.Id assignee;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public SetAssigneeSender(EmailArguments ea,
|
||||||
|
@Assisted Project.NameKey project,
|
||||||
|
@Assisted Change.Id id,
|
||||||
|
@Assisted Account.Id assignee)
|
||||||
|
throws OrmException {
|
||||||
|
super(ea, "setassignee", newChangeData(ea, project, id));
|
||||||
|
this.assignee = assignee;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void init() throws EmailException {
|
||||||
|
super.init();
|
||||||
|
|
||||||
|
add(RecipientType.TO, assignee);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void formatChange() throws EmailException {
|
||||||
|
appendText(textTemplate("SetAssignee"));
|
||||||
|
if (useHtml()) {
|
||||||
|
appendHtml(soyHtmlTemplate("SetAssigneeHtml"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getAssigneeName() {
|
||||||
|
return getNameFor(assignee);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void setupSoyContext() {
|
||||||
|
super.setupSoyContext();
|
||||||
|
soyContextEmailData.put("assigneeName", getAssigneeName());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected boolean supportsHtml() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,71 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (C) 2016 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
{namespace com.google.gerrit.server.mail.template}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The .SetAssignee template will determine the contents of the email related
|
||||||
|
* to a user being assigned to a change.
|
||||||
|
* @param change
|
||||||
|
* @param email
|
||||||
|
* @param fromName
|
||||||
|
* @param patchSet
|
||||||
|
* @param projectName
|
||||||
|
*/
|
||||||
|
{template .SetAssignee autoescape="strict" kind="text"}
|
||||||
|
Hello{sp}
|
||||||
|
{$email.assigneeName},
|
||||||
|
|
||||||
|
{\n}
|
||||||
|
{\n}
|
||||||
|
|
||||||
|
{$fromName} has assigned a change to you.
|
||||||
|
|
||||||
|
{sp}Please visit
|
||||||
|
|
||||||
|
{\n}
|
||||||
|
{\n}
|
||||||
|
|
||||||
|
{sp}{sp}{sp}{sp}{$email.changeUrl}
|
||||||
|
|
||||||
|
{\n}
|
||||||
|
{\n}
|
||||||
|
|
||||||
|
to view the change.
|
||||||
|
|
||||||
|
{\n}
|
||||||
|
{\n}
|
||||||
|
|
||||||
|
Change subject: {$change.subject}{\n}
|
||||||
|
......................................................................{\n}
|
||||||
|
|
||||||
|
{\n}
|
||||||
|
|
||||||
|
{$email.changeDetail}{\n}
|
||||||
|
|
||||||
|
{if $email.sshHost}
|
||||||
|
{\n}
|
||||||
|
{sp}{sp}git pull ssh:{print '//'}{$email.sshHost}/{$projectName}
|
||||||
|
{sp}{$patchSet.refName}
|
||||||
|
{\n}
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{if $email.includeDiff}
|
||||||
|
{\n}
|
||||||
|
{$email.unifiedDiff}
|
||||||
|
{\n}
|
||||||
|
{/if}
|
||||||
|
{/template}
|
@@ -0,0 +1,49 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (C) 2016 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
{namespace com.google.gerrit.server.mail.template}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param email
|
||||||
|
* @param fromName
|
||||||
|
* @param patchSet
|
||||||
|
* @param projectName
|
||||||
|
*/
|
||||||
|
{template .SetAssigneeHtml autoescape="strict" kind="html"}
|
||||||
|
<p>
|
||||||
|
{$fromName} has <strong>assigned</strong> a change to{sp}
|
||||||
|
{$email.assigneeName}.{sp}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
{if $email.changeUrl}
|
||||||
|
<p>
|
||||||
|
{call .ViewChangeButton data="all" /}
|
||||||
|
</p>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{call .Pre}{param content: $email.changeDetail /}{/call}
|
||||||
|
|
||||||
|
{if $email.sshHost}
|
||||||
|
{call .Pre}{param content kind="html"}
|
||||||
|
git pull ssh:{print '//'}{$email.sshHost}/{$projectName}
|
||||||
|
{sp}{$patchSet.refName}
|
||||||
|
{/param}{/call}
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{if $email.includeDiff}
|
||||||
|
{call .Pre}{param content: $email.unifiedDiff /}{/call}
|
||||||
|
{/if}
|
||||||
|
{/template}
|
Reference in New Issue
Block a user