blob: 26c8f47ac05859dbbcc4f5b2f27a63da57bf22c5 [file] [log] [blame]
// Copyright (C) 2020 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.util;
import com.google.common.base.Strings;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.gerrit.entities.Account;
import com.google.gerrit.entities.AttentionSetUpdate;
import com.google.gerrit.entities.AttentionSetUpdate.Operation;
import com.google.gerrit.extensions.api.changes.AttentionSetInput;
import com.google.gerrit.extensions.common.AccountInfo;
import com.google.gerrit.extensions.common.AttentionSetInfo;
import com.google.gerrit.extensions.restapi.BadRequestException;
import com.google.gerrit.server.account.AccountLoader;
import com.google.gerrit.server.account.AccountResolver;
import com.google.gerrit.server.notedb.ChangeNotes;
import java.io.IOException;
import java.util.Collection;
import org.eclipse.jgit.errors.ConfigInvalidException;
/** Common helpers for dealing with attention set data structures. */
public class AttentionSetUtil {
/** Returns only updates where the user was added. */
public static ImmutableSet<AttentionSetUpdate> additionsOnly(
Collection<AttentionSetUpdate> updates) {
return updates.stream()
.filter(u -> u.operation() == Operation.ADD)
.collect(ImmutableSet.toImmutableSet());
}
/** Returns only updates where the user was removed. */
public static ImmutableSet<AttentionSetUpdate> removalsOnly(
Collection<AttentionSetUpdate> updates) {
return updates.stream()
.filter(u -> u.operation() == Operation.REMOVE)
.collect(ImmutableSet.toImmutableSet());
}
/**
* Validates the input for AttentionSetInput. This must be called for all inputs that relate to
* adding or removing attention set entries, except for {@link
* com.google.gerrit.server.restapi.change.RemoveFromAttentionSet}.
*/
public static void validateInput(AttentionSetInput input) throws BadRequestException {
input.user = Strings.nullToEmpty(input.user).trim();
if (input.user.isEmpty()) {
throw new BadRequestException("missing field: user");
}
input.reason = Strings.nullToEmpty(input.reason).trim();
if (input.reason.isEmpty()) {
throw new BadRequestException("missing field: reason");
}
}
/**
* Returns the {@code Account.Id} of {@code user} if the user is active on the change, and exists.
* If the user doesn't exist or is not active on the change, the same exception is thrown to
* disallow probing for account existence based on exception type.
*/
public static Account.Id resolveAccount(
AccountResolver accountResolver, ChangeNotes changeNotes, String user)
throws ConfigInvalidException, IOException, BadRequestException {
// We will throw this exception if the account doesn't exist, or if the account is not active.
// This is purposely the same exception so that users can't probe for account existence based on
// the thrown exception.
BadRequestException possibleExceptionForNotFoundOrInactiveAccount =
new BadRequestException(
String.format(
"%s doesn't exist or is not active on the change as an owner, uploader, "
+ "reviewer, or cc so they can't be added to the attention set",
user));
Account.Id attentionUserId;
try {
attentionUserId = accountResolver.resolveIgnoreVisibility(user).asUnique().account().id();
} catch (AccountResolver.UnresolvableAccountException ex) {
possibleExceptionForNotFoundOrInactiveAccount.initCause(ex);
throw possibleExceptionForNotFoundOrInactiveAccount;
}
if (!isActiveOnTheChange(changeNotes, attentionUserId)) {
throw possibleExceptionForNotFoundOrInactiveAccount;
}
return attentionUserId;
}
/**
* Returns whether {@code attentionUserId} is active on a change. Activity is defined as being a
* part of the reviewers, an uploader, or an owner of a change.
*/
private static boolean isActiveOnTheChange(ChangeNotes changeNotes, Account.Id attentionUserId) {
return changeNotes.getChange().getOwner().equals(attentionUserId)
|| changeNotes.getCurrentPatchSet().uploader().equals(attentionUserId)
|| changeNotes.getReviewers().all().stream().anyMatch(id -> id.equals(attentionUserId));
}
/**
* Returns {@link AttentionSetInfo} from {@link AttentionSetUpdate} with {@link AccountInfo}
* fields filled by {@code accountLoader}.
*/
public static AttentionSetInfo createAttentionSetInfo(
AttentionSetUpdate attentionSetUpdate, AccountLoader accountLoader) {
// Only one account is expected in attention set reason. If there are multiple, do not return
// anything instead of failing the request.
ImmutableSet<Account.Id> accountsInTemplate =
AccountTemplateUtil.parseTemplates(attentionSetUpdate.reason());
AccountInfo reasonAccount =
accountsInTemplate.size() == 1
? accountLoader.get(Iterables.getOnlyElement(accountsInTemplate))
: null;
return new AttentionSetInfo(
accountLoader.get(attentionSetUpdate.account()),
attentionSetUpdate.timestamp(),
attentionSetUpdate.reason(),
reasonAccount);
}
private AttentionSetUtil() {}
}