All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.gerrit.server.account.AccountLoader Maven / Gradle / Ivy

The newest version!
// 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: * *

    *
  1. Call {@code get} to get AccountInfo for a given id that will be filled on next fill. *
  2. Call {@code put} to provide AccountInfo that will be filled on the next fill. *
  3. Call {@code fill} to populate properties of the AccountInfo. *
  4. 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 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; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy