Downgrade default notify for WIP reviews
Sometimes reviews are posted to a change while it's in the WIP state. The owner may be preparing guidance for their upcoming reviewers, or a CI bot may be reporting about tests. In general, for the purpose of filtering messages, we distinguish between bot and human actions on the basis of whether a tag is provided. A non-null tag is assumed to indicate automation. For reviews posted by humans, the default notify level depends on whether hasReviewStarted is true for the change. If a new WIP change is created and pending reviewers are added, those reviewers should never receive an email about the change until the owner starts review. Therefore, when WIP is true, hasReviewStarted is false, and tag is null, the default notify level is downgraded from ALL to OWNER. If the change has started review, the default notify level continues to be ALL (only if tag is null). For reviews posted by bots, the default notify level is downgraded for all WIP changes, even after they've entered review. For example, a CI bot may post failing test results for a series of intermediate patch sets. This is just noise to reviewers, but useful to the owner, so when WIP is true and tag is non-null, the default notify level is downgraded to OWNER. Change-Id: I93395e36470090377d2120bb8136e7c11670d224
This commit is contained in:
parent
fc05596270
commit
ea3e3b704a
@ -3769,25 +3769,26 @@ Use the notify property of the top-level link:#review-input[ReviewInput] instead
|
|||||||
For the purposes of this table, *everyone* means *owner, reviewers, CCs, stars, and ALL_COMMENTS
|
For the purposes of this table, *everyone* means *owner, reviewers, CCs, stars, and ALL_COMMENTS
|
||||||
watchers*.
|
watchers*.
|
||||||
|
|
||||||
[options="header",cols="1,1,1,1,1,1,1"]
|
[options="header",cols="2,1,1,2,2"]
|
||||||
|=============================
|
|=============================
|
||||||
|WIP State |Review Started|Tag Given|Default|notify=ALL|notify=OWNER_REVIEWERS|notify=OWNER
|
|WIP State |Review Started|Tag Given|Default |notify=ALL
|
||||||
|Ready for review|N/A|N/A|everyone|everyone|owner, reviewers, CCs|owner
|
|Ready for review|N/A |N/A |everyone|everyone
|
||||||
|Work in progress|no |no |not sent|everyone|owner, reviewers, CCs|owner
|
|Work in progress|no |no |not sent|everyone
|
||||||
|Work in progress|no |yes|owner |everyone|owner, reviewers, CCs|owner
|
|Work in progress|no |yes |owner |everyone
|
||||||
|Work in progress|yes|no |everyone|everyone|owner, reviewers, CCs|owner
|
|Work in progress|yes |no |everyone|everyone
|
||||||
|Work in progress|yes|yes|owner |everyone|owner, reviewers, CCs|owner
|
|Work in progress|yes |yes |owner |everyone
|
||||||
|
|
||||||
|=============================
|
|=============================
|
||||||
|
|
||||||
If reviewers are added, then a second email will be sent using the "newchange"
|
If reviewers are added, then a second email will be sent using the "newchange"
|
||||||
template. The notification logic for this email is the same as for
|
template. The notification logic for this email is the same as for
|
||||||
link:#add-reviewer[Add Reviewer].
|
link:#add-reviewer[Add Reviewer].
|
||||||
|
|
||||||
[options="header",cols="1,1,1,1,1"]
|
[options="header",cols="1,1,1"]
|
||||||
|=============================
|
|=============================
|
||||||
|WIP State |Default|notify=ALL|notify=OWNER_REVIEWERS|notify=OWNER
|
|WIP State |Default |notify=ALL
|
||||||
|Ready for review|owner, reviewers, CCs|owner, reviewers, CCs|owner, reviewers, CCs|owner
|
|Ready for review|owner, reviewers, CCs|owner, reviewers, CCs
|
||||||
|Work in progress|not sent|owner, reviewers, CCs|owner, reviewers, CCs|owner
|
|Work in progress|not sent |owner, reviewers, CCs
|
||||||
|=============================
|
|=============================
|
||||||
|
|
||||||
|
|
||||||
|
@ -182,11 +182,59 @@ public class CommentSenderIT extends AbstractNotificationTest {
|
|||||||
.bcc(ALL_COMMENTS);
|
.bcc(ALL_COMMENTS);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void commentOnWipChangeByBot() throws Exception {
|
||||||
|
StagedChange sc = stageWipChange(ALL_COMMENTS);
|
||||||
|
TestAccount bot = sc.testAccount("bot");
|
||||||
|
review(bot, sc.changeId, ENABLED, null, "tag");
|
||||||
|
assertThat(sender)
|
||||||
|
.sent("comment", sc)
|
||||||
|
.to(sc.owner)
|
||||||
|
.notTo(sc.reviewer, sc.ccer)
|
||||||
|
.notTo(sc.reviewerByEmail, sc.ccerByEmail)
|
||||||
|
.notTo(sc.starrer)
|
||||||
|
.notTo(ALL_COMMENTS);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void commentOnReviewableWipChangeByBot() throws Exception {
|
||||||
|
StagedChange sc = stageReviewableWipChange(ALL_COMMENTS);
|
||||||
|
TestAccount bot = sc.testAccount("bot");
|
||||||
|
review(bot, sc.changeId, ENABLED, null, "tag");
|
||||||
|
assertThat(sender)
|
||||||
|
.sent("comment", sc)
|
||||||
|
.to(sc.owner)
|
||||||
|
.notTo(sc.reviewer, sc.ccer)
|
||||||
|
.notTo(sc.reviewerByEmail, sc.ccerByEmail)
|
||||||
|
.notTo(sc.starrer)
|
||||||
|
.notTo(ALL_COMMENTS);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void commentOnReviewableWipChangeByBotNotifyAll() throws Exception {
|
||||||
|
StagedChange sc = stageWipChange(ALL_COMMENTS);
|
||||||
|
TestAccount bot = sc.testAccount("bot");
|
||||||
|
review(bot, sc.changeId, ENABLED, ALL, "tag");
|
||||||
|
assertThat(sender)
|
||||||
|
.sent("comment", sc)
|
||||||
|
.to(sc.owner)
|
||||||
|
.cc(sc.reviewer, sc.ccer)
|
||||||
|
.cc(sc.reviewerByEmail, sc.ccerByEmail)
|
||||||
|
.bcc(sc.starrer)
|
||||||
|
.bcc(ALL_COMMENTS);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void commentOnReviewableWipChangeByOwner() throws Exception {
|
public void commentOnReviewableWipChangeByOwner() throws Exception {
|
||||||
StagedChange sc = stageReviewableWipChange(ALL_COMMENTS);
|
StagedChange sc = stageReviewableWipChange(ALL_COMMENTS);
|
||||||
review(sc.owner, sc.changeId, ENABLED);
|
review(sc.owner, sc.changeId, ENABLED);
|
||||||
assertThat(sender).notSent();
|
assertThat(sender)
|
||||||
|
.sent("comment", sc)
|
||||||
|
.notTo(sc.owner)
|
||||||
|
.cc(sc.reviewer, sc.ccer)
|
||||||
|
.cc(sc.reviewerByEmail, sc.ccerByEmail)
|
||||||
|
.bcc(sc.starrer)
|
||||||
|
.bcc(ALL_COMMENTS);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void review(TestAccount account, String changeId, EmailStrategy strategy)
|
private void review(TestAccount account, String changeId, EmailStrategy strategy)
|
||||||
@ -197,9 +245,20 @@ public class CommentSenderIT extends AbstractNotificationTest {
|
|||||||
private void review(
|
private void review(
|
||||||
TestAccount account, String changeId, EmailStrategy strategy, @Nullable NotifyHandling notify)
|
TestAccount account, String changeId, EmailStrategy strategy, @Nullable NotifyHandling notify)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
|
review(account, changeId, strategy, notify, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void review(
|
||||||
|
TestAccount account,
|
||||||
|
String changeId,
|
||||||
|
EmailStrategy strategy,
|
||||||
|
@Nullable NotifyHandling notify,
|
||||||
|
@Nullable String tag)
|
||||||
|
throws Exception {
|
||||||
setEmailStrategy(account, strategy);
|
setEmailStrategy(account, strategy);
|
||||||
ReviewInput in = ReviewInput.recommend();
|
ReviewInput in = ReviewInput.recommend();
|
||||||
in.notify = notify;
|
in.notify = notify;
|
||||||
|
in.tag = tag;
|
||||||
gApi.changes().id(changeId).revision("current").review(in);
|
gApi.changes().id(changeId).revision("current").review(in);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -225,9 +225,9 @@ public class PostReview
|
|||||||
checkRobotComments(revision, input.robotComments);
|
checkRobotComments(revision, input.robotComments);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
NotifyHandling reviewerNotify = input.notify;
|
||||||
if (input.notify == null) {
|
if (input.notify == null) {
|
||||||
input.notify =
|
input.notify = defaultNotify(revision.getChange(), input);
|
||||||
revision.getChange().isWorkInProgress() ? NotifyHandling.OWNER : NotifyHandling.ALL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ListMultimap<RecipientType, Account.Id> accountsToNotify =
|
ListMultimap<RecipientType, Account.Id> accountsToNotify =
|
||||||
@ -321,12 +321,28 @@ public class PostReview
|
|||||||
reviewerResult.gatherResults();
|
reviewerResult.gatherResults();
|
||||||
}
|
}
|
||||||
|
|
||||||
emailReviewers(revision.getChange(), reviewerResults, input.notify, accountsToNotify);
|
emailReviewers(revision.getChange(), reviewerResults, reviewerNotify, accountsToNotify);
|
||||||
}
|
}
|
||||||
|
|
||||||
return Response.ok(output);
|
return Response.ok(output);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private NotifyHandling defaultNotify(Change c, ReviewInput in) {
|
||||||
|
if (!c.isWorkInProgress()) {
|
||||||
|
return NotifyHandling.ALL;
|
||||||
|
}
|
||||||
|
if (in.tag != null) {
|
||||||
|
// Treat the presence of tag as indicator of a "bot".
|
||||||
|
return NotifyHandling.OWNER;
|
||||||
|
}
|
||||||
|
if (!c.hasReviewStarted()) {
|
||||||
|
// If review hasn't started we want to minimize recipients, even on
|
||||||
|
// "human" comments.
|
||||||
|
return NotifyHandling.OWNER;
|
||||||
|
}
|
||||||
|
return NotifyHandling.ALL;
|
||||||
|
}
|
||||||
|
|
||||||
private void emailReviewers(
|
private void emailReviewers(
|
||||||
Change change,
|
Change change,
|
||||||
List<PostReviewers.Addition> reviewerAdditions,
|
List<PostReviewers.Addition> reviewerAdditions,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user