com.google.gerrit.server.account.AccountLoader Maven / Gradle / Ivy
// Copyright (C) 2014 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.account;
import static com.google.common.base.Preconditions.checkArgument;
import com.google.gerrit.common.Nullable;
import com.google.gerrit.entities.Account;
import com.google.gerrit.extensions.common.AccountInfo;
import com.google.gerrit.server.account.AccountDirectory.FillOptions;
import com.google.gerrit.server.logging.Metadata;
import com.google.gerrit.server.logging.TraceContext;
import com.google.gerrit.server.logging.TraceContext.TraceTimer;
import com.google.gerrit.server.permissions.PermissionBackendException;
import com.google.inject.assistedinject.Assisted;
import com.google.inject.assistedinject.AssistedInject;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.EnumSet;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
/**
 * AccountLoader is the class that populates properties of the AccountInfo provided to it.
 *
 * The class is designed to be used in the following way:
 *
 * 
 *   - Call {@code get} to get AccountInfo for a given id that will be filled on next fill.
 *   
 - Call {@code put} to provide AccountInfo that will be filled on the next fill.
 *   
 - Call {@code fill} to populate properties of the AccountInfo.
 *   
 - Call {@code get} if needed again to get filled AccountInfo.
 * 
 
 */
public class AccountLoader {
  public static final Set DETAILED_OPTIONS =
      Collections.unmodifiableSet(
          EnumSet.of(
              FillOptions.ID,
              FillOptions.NAME,
              FillOptions.EMAIL,
              FillOptions.USERNAME,
              FillOptions.DISPLAY_NAME,
              FillOptions.STATUS,
              FillOptions.STATE,
              FillOptions.AVATARS,
              FillOptions.TAGS));
  public interface Factory {
    AccountLoader create(boolean detailed);
    AccountLoader create(Set options);
  }
  private final InternalAccountDirectory directory;
  private final Set options;
  // Single AccountInfo per AccountId that is actually evaluated. All others (if any) in "provided"
  // are copies of these.
  private final Map primeAccountInfo;
  // Extra AccountInfo provided by callers that should be populated after fill().
  private final List provided;
  @AssistedInject
  AccountLoader(InternalAccountDirectory directory, @Assisted boolean detailed) {
    this(directory, detailed ? DETAILED_OPTIONS : InternalAccountDirectory.ID_ONLY);
  }
  @AssistedInject
  AccountLoader(InternalAccountDirectory directory, @Assisted Set options) {
    this.directory = directory;
    this.options = options;
    primeAccountInfo = new HashMap<>();
    provided = new ArrayList<>();
  }
  /**
   * Return AccountInfo for given id.
   *
   * If called before {@code fill} the AccountInfo is unfilled and will be filled on next call to
   * fill.
   *
   * 
If called after {@code fill} will return filled AccountInfo only if account with this id was
   * specified in one of {@code get} or {@code put} call before the call to {@code fill}. Otherwise,
   * returns unfilled AccountInfo.
   */
  @Nullable
  public synchronized AccountInfo get(@Nullable Account.Id id) {
    if (id == null) {
      return null;
    }
    AccountInfo info = primeAccountInfo.get(id);
    if (info == null) {
      info = new AccountInfo(id.get());
      primeAccountInfo.put(id, info);
    }
    return info;
  }
  /** Provide AccountInfo that will be filled on the next fill. */
  public synchronized void put(AccountInfo info) {
    checkArgument(info._accountId != null, "_accountId field required");
    provided.add(info);
  }
  /**
   * Populates properties of the {@link AccountInfo} previously returned from {@code get} or
   * provided by {@code put}
   */
  @SuppressWarnings("ReferenceEquality") // Intentional reference equality check
  public void fill() throws PermissionBackendException {
    try (TraceTimer timer = TraceContext.newTimer("Fill accounts", Metadata.empty())) {
      for (AccountInfo info : provided) {
        primeAccountInfo.putIfAbsent(Account.id(info._accountId), info);
      }
      directory.fillAccountInfo(primeAccountInfo.values(), options);
      for (AccountInfo info : provided) {
        AccountInfo filledInfo = primeAccountInfo.get(Account.id(info._accountId));
        // Check if it's the same instance.
        if (filledInfo != info) {
          filledInfo.copyTo(info);
        }
      }
    }
  }
  /** Same as {@link #fill()}, but also populate {@link AccountInfo} in {@code infos} */
  public void fill(Collection extends AccountInfo> infos) throws PermissionBackendException {
    for (AccountInfo info : infos) {
      put(info);
    }
    fill();
  }
  /** Same as {@link #fill()}, but also create and populate {@link AccountInfo} for provided id. */
  @Nullable
  public AccountInfo fillOne(@Nullable Account.Id id) throws PermissionBackendException {
    AccountInfo info = get(id);
    fill();
    return info;
  }
}