blob: 9c2cbeea6ca4c7f2b799ba9b98bda6f004f3274b [file] [log] [blame]
// Copyright (C) 2022 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.acceptance.api.change;
import static com.google.common.truth.Truth.assertThat;
import static com.google.gerrit.acceptance.testsuite.project.TestProjectUpdate.allowLabel;
import static com.google.gerrit.server.group.SystemGroupBackend.REGISTERED_USERS;
import static com.google.gerrit.server.project.testing.TestLabels.labelBuilder;
import static com.google.gerrit.server.project.testing.TestLabels.value;
import static com.google.gerrit.testing.GerritJUnit.assertThrows;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.gerrit.acceptance.AbstractDaemonTest;
import com.google.gerrit.acceptance.PushOneCommit;
import com.google.gerrit.acceptance.TestAccount;
import com.google.gerrit.acceptance.testsuite.project.ProjectOperations;
import com.google.gerrit.acceptance.testsuite.request.RequestScopeOperations;
import com.google.gerrit.common.RawInputUtil;
import com.google.gerrit.entities.AccountGroup;
import com.google.gerrit.entities.Change;
import com.google.gerrit.entities.LabelId;
import com.google.gerrit.entities.LabelType;
import com.google.gerrit.entities.PatchSet;
import com.google.gerrit.entities.PatchSetApproval;
import com.google.gerrit.entities.RefNames;
import com.google.gerrit.extensions.api.changes.ReviewInput;
import com.google.gerrit.extensions.common.ChangeInfo;
import com.google.gerrit.server.approval.ApprovalCopier;
import com.google.gerrit.server.approval.ApprovalCopier.ApprovalCopyResult;
import com.google.gerrit.server.approval.ApprovalCopier.Result.PatchSetApprovalData;
import com.google.gerrit.server.approval.ApprovalsUtil;
import com.google.gerrit.server.group.SystemGroupBackend;
import com.google.gerrit.server.project.testing.TestLabels;
import com.google.gerrit.server.util.AccountTemplateUtil;
import com.google.gerrit.server.util.time.TimeUtil;
import com.google.inject.Inject;
import org.junit.Test;
/**
* Tests to verify that copied/outdated approvals are included into the change message that is
* posted on patch set creation. Includes verifying that the copied/outdated approvals in the change
* message are correctly formatted.
*
* <p>Some of the tests only verify the correct formatting of the copied/outdated approvals in the
* change message that is done by {@link
* ApprovalsUtil#formatApprovalCopierResult(com.google.gerrit.server.approval.ApprovalCopier.Result)
* }. This method does the formatting based on the inputs that it gets, but it doesn't do any
* verification of these inputs. This means it's possible to provide inputs that are inconsistent
* with the approval copying logic in {@link ApprovalCopier}. E.g. it's possible to provide "is:MAX"
* as a passing atom for a "Code-Review-1" vote and have "is:MAX" highlighted as passing in the
* message although the "Code-Review-1" vote doesn't match with "is:MAX". For easier readability the
* formatting tests avoid using such inconsistent input data, but it's not impossible that in some
* cases we made a mistake and the input data is inconsistent.
*/
public class CopiedApprovalsInChangeMessageIT extends AbstractDaemonTest {
@Inject private ApprovalsUtil approvalsUtil;
@Inject private ProjectOperations projectOperations;
@Inject private RequestScopeOperations requestScopeOperations;
@Test
public void cannotFormatWithNullApprovalCopierResult() throws Exception {
NullPointerException exception =
assertThrows(
NullPointerException.class,
() -> approvalsUtil.formatApprovalCopierResult(/* approvalCopierResult= */ null));
assertThat(exception).hasMessageThat().isEqualTo("approvalCopierResult");
}
@Test
public void format_noCopiedApprovals_noOutdatedApprovals() throws Exception {
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(), /* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult)).isEmpty();
}
@Test
public void formatOutdatedApproval_noCopyCondition() throws Exception {
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(skippedEval(patchSetApproval)));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue("Outdated Votes:\n* Code-Review+1 (copy condition: \"NEVER\")\n");
}
@Test
public void formatCopiedApproval_withCopyCondition_noUserInPredicate() throws Exception {
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", -2);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval,
/* copied= */ true,
/* copyCondition= */ "is:MIN OR is:MAX",
/* passingAtoms= */ ImmutableSet.of("is:MIN"),
/* failingAtoms= */ ImmutableSet.of("is:MAX"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue("Copied Votes:\n* Code-Review-2 (copy condition: \"**is:MIN** OR is:MAX\")\n");
}
@Test
public void formatOutdatedApproval_withCopyCondition_noUserInPredicate() throws Exception {
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", 2);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval,
/* copied= */ false,
/* copyCondition= */ "changekind:TRIVIAL_REBASE is:MAX",
/* passingAtoms= */ ImmutableSet.of("is:MAX"),
/* failingAtoms= */ ImmutableSet.of("changekind:TRIVIAL_REBASE"))));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Outdated Votes:\n* Code-Review+2 (copy condition:"
+ " \"changekind:TRIVIAL_REBASE **is:MAX**\")\n");
}
@Test
public void formatOutdatedApproval_withNonParseableCopyCondition_noUserInPredicate()
throws Exception {
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(errorEval(patchSetApproval, "foo bar baz")));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Outdated Votes:\n* Code-Review+1 (non-parseable copy condition: \"foo bar baz\")\n");
}
@Test
public void formatCopiedApproval_withCopyCondition_withUserInPredicate() throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", 2);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:MAX approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:MAX", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Copied Votes:\n"
+ "* Code-Review+2 by %s"
+ " (copy condition: \"is:MIN OR (**is:MAX** **approverin:%s**)\")\n",
AccountTemplateUtil.getAccountTemplate(admin.id()), groupUuid));
}
@Test
public void formatOutdatedApproval_withCopyCondition_withUserInPredicate() throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval,
/* copied= */ false,
/* copyCondition= */ String.format(
"is:MIN OR (is:MAX approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN", "is:MAX"))));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Outdated Votes:\n"
+ "* Code-Review+1 by %s (copy condition: \"is:MIN"
+ " OR (is:MAX **approverin:%s**)\")\n",
AccountTemplateUtil.getAccountTemplate(admin.id()), groupUuid));
}
@Test
public void formatCopiedApproval_withCopyCondition_withUserInPredicateThatContainNonVisibleGroup()
throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", 2);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:MAX approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:MAX", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN"))),
/* outdatedApprovals= */ ImmutableSet.of());
// Set 'user' as the current user in the request scope.
// 'user' cannot see the Administrators group that is used in the copy condition.
// Parsing the copy condition should still succeed since ApprovalsUtil should use the internal
// user that can see everything when parsing the copy condition.
requestScopeOperations.setApiUser(user.id());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Copied Votes:\n"
+ "* Code-Review+2 by %s"
+ " (copy condition: \"is:MIN OR (**is:MAX** **approverin:%s**)\")\n",
AccountTemplateUtil.getAccountTemplate(admin.id()), groupUuid));
}
@Test
public void
formatOutdatedpproval_withCopyCondition_withUserInPredicateThatContainNonVisibleGroup()
throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:MAX approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN", "is:MAX"))));
// Set 'user' as the current user in the request scope.
// 'user' cannot see the Administrators group that is used in the copy condition.
// Parsing the copy condition should still succeed since ApprovalsUtil should use the internal
// user that can see everything when parsing the copy condition.
requestScopeOperations.setApiUser(user.id());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Outdated Votes:\n"
+ "* Code-Review+1 by %s (copy condition: \"is:MIN"
+ " OR (is:MAX **approverin:%s**)\")\n",
AccountTemplateUtil.getAccountTemplate(admin.id()), groupUuid));
}
@Test
public void formatOutdatedApproval_withNonParseableCopyCondition_withUserInPredicate()
throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
errorEval(
patchSetApproval,
String.format("is:MIN OR (is:MAX approverin:%s) OR foo bar baz", groupUuid))));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Outdated Votes:\n"
+ "* Code-Review+1 (non-parseable copy condition: \"is:MIN"
+ " OR (is:MAX approverin:%s) OR foo bar baz\")\n",
groupUuid));
}
@Test
public void formatMultipleApprovals_sameVote_noCopyCondition() throws Exception {
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 1);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
skippedEval(patchSetApproval1), skippedEval(patchSetApproval2)));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue("Outdated Votes:\n* Code-Review+1 (copy condition: \"NEVER\")\n");
}
@Test
public void formatMultipleApprovals_differentLabel_noCopyCondition() throws Exception {
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 1);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Verified", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
skippedEval(patchSetApproval1), skippedEval(patchSetApproval2)));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Outdated Votes:\n"
+ "* Code-Review+1 (copy condition: \"NEVER\")\n"
+ "* Verified+1 (copy condition: \"NEVER\")\n");
}
@Test
public void formatMultipleApprovals_differentValue_noCopyCondition() throws Exception {
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
skippedEval(patchSetApproval1), skippedEval(patchSetApproval2)));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue("Outdated Votes:\n* Code-Review+1, Code-Review+2 (copy condition: \"NEVER\")\n");
}
@Test
public void formatMultipleApprovals_sameVote_withCopyCondition_noUserInPredicate()
throws Exception {
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 2);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval1,
/* copied= */ true,
/* copyCondition= */ "is:MIN OR is:MAX",
/* passingAtoms= */ ImmutableSet.of("is:MAX"),
/* failingAtoms= */ ImmutableSet.of("is:MIN")),
createApprovalData(
patchSetApproval2,
/* copied= */ true,
/* copyCondition= */ "is:MIN OR is:MAX",
/* passingAtoms= */ ImmutableSet.of("is:MAX"),
/* failingAtoms= */ ImmutableSet.of("is:MIN"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue("Copied Votes:\n* Code-Review+2 (copy condition: \"is:MIN OR **is:MAX**\")\n");
}
@Test
public void formatMultipleApprovals_differentLabel_withCopyCondition_noUserInPredicate()
throws Exception {
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Verified", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval1,
/* copied= */ true,
/* copyCondition= */ "is:MIN OR is:MAX",
/* passingAtoms= */ ImmutableSet.of("is:MAX"),
/* failingAtoms= */ ImmutableSet.of("is:MIN")),
createApprovalData(
patchSetApproval2,
/* copied= */ true,
/* copyCondition= */ "is:MIN OR is:MAX",
/* passingAtoms= */ ImmutableSet.of("is:MAX"),
/* failingAtoms= */ ImmutableSet.of("is:MIN"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Copied Votes:\n"
+ "* Code-Review+2 (copy condition: \"is:MIN OR **is:MAX**\")\n"
+ "* Verified+1 (copy condition: \"is:MIN OR **is:MAX**\")\n");
}
@Test
public void
formatMultipleApprovals_differentValue_withCopyCondition_noUserInPredicate_samePassingAtoms()
throws Exception {
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval1,
/* copied= */ true,
/* copyCondition= */ "changekind:REWORK",
/* passingAtoms= */ ImmutableSet.of("changekind:REWORK"),
/* failingAtoms= */ ImmutableSet.of()),
createApprovalData(
patchSetApproval2,
/* copied= */ true,
/* copyCondition= */ "changekind:REWORK",
/* passingAtoms= */ ImmutableSet.of("changekind:REWORK"),
/* failingAtoms= */ ImmutableSet.of())),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Copied Votes:\n"
+ "* Code-Review+1, Code-Review+2 (copy condition: \"**changekind:REWORK**\")\n");
}
@Test
public void
formatMultipleApprovals_differentValue_withCopyCondition_noUserInPredicate_differentPassingAtoms()
throws Exception {
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval1,
/* copied= */ true,
/* copyCondition= */ "is:1 OR is:2",
/* passingAtoms= */ ImmutableSet.of("is:2"),
/* failingAtoms= */ ImmutableSet.of("is:1")),
createApprovalData(
patchSetApproval2,
/* copied= */ true,
/* copyCondition= */ "is:1 OR is:2",
/* passingAtoms= */ ImmutableSet.of("is:1"),
/* failingAtoms= */ ImmutableSet.of("is:2"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Copied Votes:\n"
+ "* Code-Review+1 (copy condition: \"**is:1** OR is:2\")\n"
+ "* Code-Review+2 (copy condition: \"is:1 OR **is:2**\")\n");
}
@Test
public void formatMultipleApprovals_sameVote_withNonParseableCopyCondition_noUserInPredicate()
throws Exception {
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 1);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
errorEval(patchSetApproval1, "foo bar baz"),
errorEval(patchSetApproval2, "foo bar baz")));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Outdated Votes:\n* Code-Review+1 (non-parseable copy condition: \"foo bar baz\")\n");
}
@Test
public void
formatMultipleApprovals_differentLabel_withNonParseableCopyCondition_noUserInPredicate()
throws Exception {
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 1);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Verified", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
errorEval(patchSetApproval1, "foo bar baz"),
errorEval(patchSetApproval2, "foo bar baz")));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Outdated Votes:\n"
+ "* Code-Review+1 (non-parseable copy condition: \"foo bar baz\")\n"
+ "* Verified+1 (non-parseable copy condition: \"foo bar baz\")\n");
}
@Test
public void
formatMultipleApprovals_differentValue_withNonParseableCopyCondition_noUserInPredicate()
throws Exception {
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
errorEval(patchSetApproval1, "foo bar baz"),
errorEval(patchSetApproval2, "foo bar baz")));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Outdated Votes:\n"
+ "* Code-Review+1, Code-Review+2"
+ " (non-parseable copy condition: \"foo bar baz\")\n");
}
@Test
public void
formatMultipleApprovals_sameVote_withCopyCondition_withUserInPredicate_samePassingAtoms()
throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval1 = createPatchSetApproval(user, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(admin, "Code-Review", 2);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval1,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:MAX approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:MAX", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN")),
createApprovalData(
patchSetApproval2,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:MAX approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:MAX", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Copied Votes:\n"
+ "* Code-Review+2 by %s, %s"
+ " (copy condition: \"is:MIN OR (**is:MAX** **approverin:%s**)\")\n",
AccountTemplateUtil.getAccountTemplate(admin.id()),
AccountTemplateUtil.getAccountTemplate(user.id()),
groupUuid));
}
@Test
public void
formatMultipleApprovals_sameVote_withCopyCondition_withUserInPredicate_differentPassingAtoms()
throws Exception {
String administratorsGroupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
String registeredUsersGroupUuid = SystemGroupBackend.REGISTERED_USERS.get();
PatchSetApproval patchSetApproval1 = createPatchSetApproval(user, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(admin, "Code-Review", 2);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval1,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:MAX approverin:%s) OR (is:MAX approverin:%s)",
administratorsGroupUuid, registeredUsersGroupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:MAX", String.format("approverin:%s", registeredUsersGroupUuid)),
/* failingAtoms= */ ImmutableSet.of(
"is:MIN", String.format("approverin:%s", administratorsGroupUuid))),
createApprovalData(
patchSetApproval2,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:MAX approverin:%s) OR (is:MAX approverin:%s)",
administratorsGroupUuid, registeredUsersGroupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:MAX",
String.format("approverin:%s", administratorsGroupUuid),
String.format("approverin:%s", registeredUsersGroupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Copied Votes:\n"
+ "* Code-Review+2 by %s"
+ " (copy condition: \"is:MIN OR (**is:MAX** **approverin:%s**)"
+ " OR (**is:MAX** **approverin:%s**)\")\n"
+ "* Code-Review+2 by %s"
+ " (copy condition: \"is:MIN OR (**is:MAX** approverin:%s)"
+ " OR (**is:MAX** **approverin:%s**)\")\n",
AccountTemplateUtil.getAccountTemplate(admin.id()),
administratorsGroupUuid,
registeredUsersGroupUuid,
AccountTemplateUtil.getAccountTemplate(user.id()),
administratorsGroupUuid,
registeredUsersGroupUuid));
}
@Test
public void formatMultipleApprovals_differentLabel_withCopyCondition_withUserInPredicate()
throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval1 = createPatchSetApproval(user, "Code-Review", -2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(admin, "Verified", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval1,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:MAX approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of("is:MIN"),
/* failingAtoms= */ ImmutableSet.of(
"is:MAX", String.format("approverin:%s", groupUuid))),
createApprovalData(
patchSetApproval2,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:MAX approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:MAX", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Copied Votes:\n"
+ "* Code-Review-2 by %s (copy condition: \"**is:MIN**"
+ " OR (is:MAX approverin:%s)\")\n"
+ "* Verified+1 by %s (copy condition: \"is:MIN"
+ " OR (**is:MAX** **approverin:%s**)\")\n",
AccountTemplateUtil.getAccountTemplate(user.id()),
groupUuid,
AccountTemplateUtil.getAccountTemplate(admin.id()),
groupUuid));
}
@Test
public void
formatMultipleApprovals_differentValue_withCopyCondition_withUserInPredicate_samePassingAtoms()
throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval1,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:ANY approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:ANY", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN")),
createApprovalData(
patchSetApproval2,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:ANY approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:ANY", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Copied Votes:\n"
+ "* Code-Review+1 by %s, Code-Review+2 by %s"
+ " (copy condition: \"is:MIN OR (**is:ANY** **approverin:%s**)\")\n",
AccountTemplateUtil.getAccountTemplate(user.id()),
AccountTemplateUtil.getAccountTemplate(admin.id()),
groupUuid));
}
@Test
public void
formatMultipleApprovals_differentValue_withCopyCondition_withUserInPredicate_differentPassingAtoms()
throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval1,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:1 approverin:%s) OR (is:2 approverin:%s)",
groupUuid, groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:2", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN", "is:1")),
createApprovalData(
patchSetApproval2,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:1 approverin:%s) OR (is:2 approverin:%s)",
groupUuid, groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:1", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN", "is:2"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Copied Votes:\n"
+ "* Code-Review+1 by %s"
+ " (copy condition: \"is:MIN OR (**is:1** **approverin:%s**)"
+ " OR (is:2 **approverin:%s**)\")\n"
+ "* Code-Review+2 by %s"
+ " (copy condition: \"is:MIN OR (is:1 **approverin:%s**)"
+ " OR (**is:2** **approverin:%s**)\")\n",
AccountTemplateUtil.getAccountTemplate(user.id()),
groupUuid,
groupUuid,
AccountTemplateUtil.getAccountTemplate(admin.id()),
groupUuid,
groupUuid));
}
@Test
public void formatMultipleApprovals_differentAndSameValue_withCopyCondition_withUserInPredicate()
throws Exception {
TestAccount user2 = accountCreator.user2();
String groupUuid = SystemGroupBackend.REGISTERED_USERS.get();
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user2, "Code-Review", 1);
PatchSetApproval patchSetApproval3 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
createApprovalData(
patchSetApproval1,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:ANY approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:ANY", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN")),
createApprovalData(
patchSetApproval2,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:ANY approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:ANY", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN")),
createApprovalData(
patchSetApproval3,
/* copied= */ true,
/* copyCondition= */ String.format(
"is:MIN OR (is:ANY approverin:%s)", groupUuid),
/* passingAtoms= */ ImmutableSet.of(
"is:ANY", String.format("approverin:%s", groupUuid)),
/* failingAtoms= */ ImmutableSet.of("is:MIN"))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Copied Votes:\n"
+ "* Code-Review+1 by %s, %s, Code-Review+2 by %s"
+ " (copy condition: \"is:MIN OR (**is:ANY** **approverin:%s**)\")\n",
AccountTemplateUtil.getAccountTemplate(user.id()),
AccountTemplateUtil.getAccountTemplate(user2.id()),
AccountTemplateUtil.getAccountTemplate(admin.id()),
groupUuid));
}
@Test
public void formatMultipleApprovals_sameVote_withNonParseableCopyCondition_withUserInPredicate()
throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 1);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
errorEval(
patchSetApproval1,
String.format("is:MIN OR (is:MAX approverin:%s) OR foo bar baz", groupUuid)),
errorEval(
patchSetApproval2,
String.format("is:MIN OR (is:MAX approverin:%s) OR foo bar baz", groupUuid))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Copied Votes:\n"
+ "* Code-Review+1 (non-parseable copy condition: \"is:MIN"
+ " OR (is:MAX approverin:%s) OR foo bar baz\")\n",
groupUuid));
}
@Test
public void
formatMultipleApprovals_differentLabel_withNonParseableCopyCondition_withUserInPredicate()
throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 1);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Verified", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
errorEval(
patchSetApproval1,
String.format("is:MIN OR (is:MAX approverin:%s) OR foo bar baz", groupUuid)),
errorEval(
patchSetApproval2,
String.format("is:MIN OR (is:MAX approverin:%s) OR foo bar baz", groupUuid))));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Outdated Votes:\n"
+ "* Code-Review+1 (non-parseable copy condition: \"is:MIN"
+ " OR (is:MAX approverin:%s) OR foo bar baz\")\n"
+ "* Verified+1 (non-parseable copy condition: \"is:MIN"
+ " OR (is:MAX approverin:%s) OR foo bar baz\")\n",
groupUuid, groupUuid));
}
@Test
public void
formatMultipleApprovals_differentValue_withNonParseableCopyCondition_withUserInPredicate()
throws Exception {
String groupUuid =
groupCache.get(AccountGroup.nameKey("Administrators")).get().getGroupUUID().get();
PatchSetApproval patchSetApproval1 = createPatchSetApproval(admin, "Code-Review", 2);
PatchSetApproval patchSetApproval2 = createPatchSetApproval(user, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
errorEval(
patchSetApproval1,
String.format("is:MIN OR (is:MAX approverin:%s) OR foo bar baz", groupUuid)),
errorEval(
patchSetApproval2,
String.format("is:MIN OR (is:MAX approverin:%s) OR foo bar baz", groupUuid))));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
String.format(
"Outdated Votes:\n"
+ "* Code-Review+1, Code-Review+2 (non-parseable copy condition: \"is:MIN"
+ " OR (is:MAX approverin:%s) OR foo bar baz\")\n",
groupUuid));
}
@Test
public void copiedAndOutdatedApprovalsAreIncludedInChangeMessageOnPatchSetCreationByPush()
throws Exception {
// Add Verified label without copy condition.
try (ProjectConfigUpdate u = updateProject(project)) {
LabelType.Builder verified =
labelBuilder(
LabelId.VERIFIED, value(1, "Passes"), value(0, "No score"), value(-1, "Failed"));
u.getConfig().upsertLabelType(verified.build());
u.save();
}
projectOperations
.project(project)
.forUpdate()
.add(
allowLabel(TestLabels.verified().getName())
.ref(RefNames.REFS_HEADS + "*")
.group(REGISTERED_USERS)
.range(-1, 1))
.update();
PushOneCommit.Result r = createChange();
// Vote Code-Review-2 (sticky because it's a veto vote and the Code-Review label has "is:MIN" as
// part of its copy condition)
gApi.changes().id(r.getChangeId()).current().review(ReviewInput.reject());
// Vote Verified+1 (not sticky because the Verified label has no copy condition)
gApi.changes().id(r.getChangeId()).current().review(new ReviewInput().label("Verified", 1));
amendChange(r.getChangeId()).assertOkStatus();
ChangeInfo change = change(r).get();
assertThat(Iterables.getLast(change.messages).message)
.isEqualTo(
"Uploaded patch set 2.\n"
+ "\n"
+ "Copied Votes:\n"
+ "* Code-Review-2 (copy condition: \"changekind:NO_CHANGE"
+ " OR changekind:TRIVIAL_REBASE OR **is:MIN**\")\n"
+ "\n"
+ "Outdated Votes:\n"
+ "* Verified+1 (copy condition: \"NEVER\")\n");
}
@Test
public void
copiedAndOutdatedApprovalsAreIncludedInChangeMessageOnPatchSetCreationByPush_withReviewMessage()
throws Exception {
// Add Verified label without copy condition.
try (ProjectConfigUpdate u = updateProject(project)) {
LabelType.Builder verified =
labelBuilder(
LabelId.VERIFIED, value(1, "Passes"), value(0, "No score"), value(-1, "Failed"));
u.getConfig().upsertLabelType(verified.build());
u.save();
}
projectOperations
.project(project)
.forUpdate()
.add(
allowLabel(TestLabels.verified().getName())
.ref(RefNames.REFS_HEADS + "*")
.group(REGISTERED_USERS)
.range(-1, 1))
.update();
PushOneCommit.Result r = createChange();
// Vote Code-Review-2 (sticky because it's a veto vote and the Code-Review label has "is:MIN" as
// part of its copy condition)
gApi.changes().id(r.getChangeId()).current().review(ReviewInput.reject());
// Vote Verified+1 (not sticky because the Verified label has no copy condition)
gApi.changes().id(r.getChangeId()).current().review(new ReviewInput().label("Verified", 1));
String reviewMessage = "Foo-Bar-Baz";
amendChange(r.getChangeId(), "refs/for/master%m=" + reviewMessage, admin, testRepo)
.assertOkStatus();
ChangeInfo change = change(r).get();
assertThat(Iterables.getLast(change.messages).message)
.isEqualTo(
"Uploaded patch set 2.\n"
+ "\n"
+ "Foo-Bar-Baz\n"
+ "\n"
+ "Copied Votes:\n"
+ "* Code-Review-2 (copy condition: \"changekind:NO_CHANGE"
+ " OR changekind:TRIVIAL_REBASE OR **is:MIN**\")\n"
+ "\n"
+ "Outdated Votes:\n"
+ "* Verified+1 (copy condition: \"NEVER\")\n");
}
@Test
public void copiedAndOutdatedApprovalsAreIncludedInChangeMessageOnPatchSetCreationByApi()
throws Exception {
// Add Verified label without copy condition.
try (ProjectConfigUpdate u = updateProject(project)) {
LabelType.Builder verified =
labelBuilder(
LabelId.VERIFIED, value(1, "Passes"), value(0, "No score"), value(-1, "Failed"));
u.getConfig().upsertLabelType(verified.build());
u.save();
}
projectOperations
.project(project)
.forUpdate()
.add(
allowLabel(TestLabels.verified().getName())
.ref(RefNames.REFS_HEADS + "*")
.group(REGISTERED_USERS)
.range(-1, 1))
.update();
PushOneCommit.Result r = createChange();
// Vote Code-Review-2 (sticky because it's a veto vote and the Code-Review label has "is:MIN" as
// part of its copy condition)
gApi.changes().id(r.getChangeId()).current().review(ReviewInput.reject());
// Vote Verified+1 (not sticky because the Verified label has no copy condition)
gApi.changes().id(r.getChangeId()).current().review(new ReviewInput().label("Verified", 1));
gApi.changes().id(r.getChangeId()).edit().modifyFile("a.txt", RawInputUtil.create("content"));
gApi.changes().id(r.getChangeId()).edit().publish();
ChangeInfo change = change(r).get();
assertThat(Iterables.getLast(change.messages).message)
.isEqualTo(
"Patch Set 2: Published edit on patch set 1.\n"
+ "\n"
+ "Copied Votes:\n"
+ "* Code-Review-2 (copy condition: \"changekind:NO_CHANGE"
+ " OR changekind:TRIVIAL_REBASE OR **is:MIN**\")\n"
+ "\n"
+ "Outdated Votes:\n"
+ "* Verified+1 (copy condition: \"NEVER\")\n");
}
@Test
public void forcedCopyIncludedInChangeMessage() {
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", -1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(
PatchSetApprovalData.create(
patchSetApproval,
ApprovalCopyResult.create(
/* labelCopyCondition= */ "is:MIN OR is:MAX",
/* labelCopy= */ false,
/* copyEnforcement= */ "is:negative",
/* forcedCopy= */ true,
/* copyRestriction= */ "changekind:REWORK",
/* forcedNonCopy= */ true,
ImmutableSet.of("is:negative", "changekind:REWORK"),
ImmutableSet.of("is:MIN", "is:MAX")))),
/* outdatedApprovals= */ ImmutableSet.of());
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Copied Votes:\n"
+ "* Code-Review-1 (forced copy condition\\*: \"**is:negative**\")\n\n"
+ "\\* The label has `labelCopyEnforcement` or `labelCopyRestriction` configured."
+ " Only the most relevant condition that determined the outcome is shown.\n");
}
@Test
public void forcedNonCopyIncludedInChangeMessage() {
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", 2);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
PatchSetApprovalData.create(
patchSetApproval,
ApprovalCopyResult.create(
/* labelCopyCondition= */ "is:MIN OR is:MAX",
/* labelCopy= */ true,
/* copyEnforcement= */ "is:negative",
/* forcedCopy= */ false,
/* copyRestriction= */ "changekind:REWORK",
/* forcedNonCopy= */ true,
ImmutableSet.of("is:MAX", "changekind:REWORK"),
ImmutableSet.of("is:MIN", "is:negative")))));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Outdated Votes:\n"
+ "* Code-Review+2 (forced copy restriction\\*: \"**changekind:REWORK**\")\n\n"
+ "\\* The label has `labelCopyEnforcement` or `labelCopyRestriction` configured."
+ " Only the most relevant condition that determined the outcome is shown.\n");
}
@Test
public void forcedRulesPresent_asteriskIncludedInChangeMessage() {
PatchSetApproval patchSetApproval = createPatchSetApproval(admin, "Code-Review", 1);
ApprovalCopier.Result approvalCopierResult =
ApprovalCopier.Result.create(
/* copiedApprovals= */ ImmutableSet.of(),
/* outdatedApprovals= */ ImmutableSet.of(
PatchSetApprovalData.create(
patchSetApproval,
ApprovalCopyResult.create(
/* labelCopyCondition= */ "is:MIN OR is:MAX",
/* labelCopy= */ false,
/* copyEnforcement= */ "is:negative",
/* forcedCopy= */ false,
/* copyRestriction= */ "changekind:REWORK",
/* forcedNonCopy= */ true,
ImmutableSet.of("changekind:REWORK"),
ImmutableSet.of("is:MIN", "is:MAX", "is:negative")))));
assertThat(approvalsUtil.formatApprovalCopierResult(approvalCopierResult))
.hasValue(
"Outdated Votes:\n"
+ "* Code-Review+1 (copy condition\\*: \"is:MIN OR is:MAX\")\n\n"
+ "\\* The label has `labelCopyEnforcement` or `labelCopyRestriction` configured."
+ " Only the most relevant condition that determined the outcome is shown.\n");
}
private PatchSetApproval createPatchSetApproval(
TestAccount testAccount, String label, int value) {
return PatchSetApproval.builder()
.key(
PatchSetApproval.key(
PatchSet.id(Change.id(1), 1), testAccount.id(), LabelId.create(label)))
.value(value)
.granted(TimeUtil.now())
.build();
}
private PatchSetApprovalData skippedEval(PatchSetApproval psa) {
return PatchSetApprovalData.create(psa, ApprovalCopyResult.createEvaluationSkipped());
}
private PatchSetApprovalData errorEval(PatchSetApproval psa, String copyCondition) {
return PatchSetApprovalData.create(
psa,
ApprovalCopyResult.create(
copyCondition,
/* labelCopy= */ false,
/* copyEnforcement= */ null,
/* forcedCopy= */ false,
/* copyRestriction= */ null,
/* forcedNonCopy= */ false,
ImmutableSet.of(),
ImmutableSet.of()));
}
private PatchSetApprovalData createApprovalData(
PatchSetApproval psa,
boolean copied,
String copyCondition,
ImmutableSet<String> passingAtoms,
ImmutableSet<String> failingAtoms) {
return PatchSetApprovalData.create(
psa,
ApprovalCopyResult.create(
copyCondition,
/* labelCopy= */ copied,
/* copyEnforcement= */ null,
/* forcedCopy= */ false,
/* copyRestriction= */ null,
/* forcedNonCopy= */ false,
passingAtoms,
failingAtoms));
}
}