blob: 7fb61fcbe5bf39eeb9b7cc573fc324831e28d980 [file] [log] [blame]
// Copyright (C) 2016 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.externalids;
import static java.util.stream.Collectors.toSet;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;
import com.google.common.collect.ImmutableSetMultimap;
import com.google.common.collect.Iterables;
import com.google.common.collect.ListMultimap;
import com.google.common.collect.Multimap;
import com.google.common.collect.MultimapBuilder;
import com.google.gerrit.reviewdb.client.Account;
import com.google.inject.Inject;
import com.google.inject.Singleton;
import com.google.inject.name.Named;
import java.io.IOException;
import java.util.Collections;
import java.util.Set;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import java.util.function.Consumer;
import org.eclipse.jgit.lib.ObjectId;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/** Caches external IDs of all accounts. The external IDs are always loaded from NoteDb. */
@Singleton
class ExternalIdCacheImpl implements ExternalIdCache {
private static final Logger log = LoggerFactory.getLogger(ExternalIdCacheImpl.class);
public static final String CACHE_NAME = "external_ids_map";
private final LoadingCache<ObjectId, ImmutableSetMultimap<Account.Id, ExternalId>>
extIdsByAccount;
private final ExternalIdReader externalIdReader;
private final Lock lock;
@Inject
ExternalIdCacheImpl(
@Named(CACHE_NAME)
LoadingCache<ObjectId, ImmutableSetMultimap<Account.Id, ExternalId>> extIdsByAccount,
ExternalIdReader externalIdReader) {
this.extIdsByAccount = extIdsByAccount;
this.externalIdReader = externalIdReader;
this.lock = new ReentrantLock(true /* fair */);
}
@Override
public void onCreate(ObjectId newNotesRev, Iterable<ExternalId> extIds) throws IOException {
updateCache(
newNotesRev,
m -> {
for (ExternalId extId : extIds) {
m.put(extId.accountId(), extId);
}
});
}
@Override
public void onRemove(ObjectId newNotesRev, Iterable<ExternalId> extIds) throws IOException {
updateCache(
newNotesRev,
m -> {
for (ExternalId extId : extIds) {
m.remove(extId.accountId(), extId);
}
});
}
@Override
public void onRemoveByKeys(
ObjectId newNotesRev, Account.Id accountId, Iterable<ExternalId.Key> extIdKeys)
throws IOException {
updateCache(
newNotesRev,
m -> {
for (ExternalId extId : m.get(accountId)) {
for (ExternalId.Key extIdKey : extIdKeys) {
if (extIdKey.equals(extId.key())) {
m.remove(accountId, extId);
break;
}
}
}
});
}
@Override
public void onRemoveByKeys(ObjectId newNotesRev, Iterable<ExternalId.Key> extIdKeys)
throws IOException {
updateCache(
newNotesRev,
m -> {
for (ExternalId extId : m.values()) {
for (ExternalId.Key extIdKey : extIdKeys) {
if (extIdKey.equals(extId.key())) {
m.remove(extId.accountId(), extId);
break;
}
}
}
});
}
@Override
public void onUpdate(ObjectId newNotesRev, Iterable<ExternalId> updatedExtIds)
throws IOException {
updateCache(
newNotesRev,
m -> {
for (ExternalId updatedExtId : updatedExtIds) {
for (ExternalId extId : m.get(updatedExtId.accountId())) {
if (updatedExtId.key().equals(extId.key())) {
m.remove(updatedExtId.accountId(), extId);
break;
}
}
m.put(updatedExtId.accountId(), updatedExtId);
}
});
}
@Override
public void onReplace(
ObjectId newNotesRev,
Account.Id accountId,
Iterable<ExternalId> toRemove,
Iterable<ExternalId> toAdd)
throws IOException {
ExternalIdsUpdate.checkSameAccount(Iterables.concat(toRemove, toAdd), accountId);
updateCache(
newNotesRev,
m -> {
for (ExternalId extId : toRemove) {
m.remove(extId.accountId(), extId);
}
for (ExternalId extId : toAdd) {
m.put(extId.accountId(), extId);
}
});
}
@Override
public void onReplaceByKeys(
ObjectId newNotesRev,
Account.Id accountId,
Iterable<ExternalId.Key> toRemove,
Iterable<ExternalId> toAdd)
throws IOException {
ExternalIdsUpdate.checkSameAccount(toAdd, accountId);
updateCache(
newNotesRev,
m -> {
for (ExternalId extId : m.get(accountId)) {
for (ExternalId.Key extIdKey : toRemove) {
if (extIdKey.equals(extId.key())) {
m.remove(accountId, extId);
}
}
}
for (ExternalId extId : toAdd) {
m.put(extId.accountId(), extId);
}
});
}
@Override
public void onReplaceByKeys(
ObjectId newNotesRev, Iterable<ExternalId.Key> toRemove, Iterable<ExternalId> toAdd)
throws IOException {
updateCache(
newNotesRev,
m -> {
for (ExternalId extId : m.values()) {
for (ExternalId.Key extIdKey : toRemove) {
if (extIdKey.equals(extId.key())) {
m.remove(extId.accountId(), extId);
}
}
}
for (ExternalId extId : toAdd) {
m.put(extId.accountId(), extId);
}
});
}
@Override
public void onReplace(
ObjectId newNotesRev, Iterable<ExternalId> toRemove, Iterable<ExternalId> toAdd)
throws IOException {
updateCache(
newNotesRev,
m -> {
for (ExternalId extId : toRemove) {
m.remove(extId.accountId(), extId);
}
for (ExternalId extId : toAdd) {
m.put(extId.accountId(), extId);
}
});
}
@Override
public Set<ExternalId> byAccount(Account.Id accountId) throws IOException {
try {
return extIdsByAccount.get(externalIdReader.readRevision()).get(accountId);
} catch (ExecutionException e) {
log.warn("Cannot list external ids by account", e);
return Collections.emptySet();
}
}
@Override
public Set<ExternalId> byEmail(String email) throws IOException {
try {
return extIdsByAccount
.get(externalIdReader.readRevision())
.values()
.stream()
.filter(e -> email.equals(e.email()))
.collect(toSet());
} catch (ExecutionException e) {
log.warn("Cannot list external ids by email", e);
return Collections.emptySet();
}
}
private void updateCache(ObjectId newNotesRev, Consumer<Multimap<Account.Id, ExternalId>> update)
throws IOException {
lock.lock();
try {
ListMultimap<Account.Id, ExternalId> m =
MultimapBuilder.hashKeys()
.arrayListValues()
.build(extIdsByAccount.get(externalIdReader.readRevision()));
update.accept(m);
extIdsByAccount.put(newNotesRev, ImmutableSetMultimap.copyOf(m));
} catch (ExecutionException e) {
log.warn("Cannot update external IDs", e);
} finally {
lock.unlock();
}
}
static class Loader extends CacheLoader<ObjectId, ImmutableSetMultimap<Account.Id, ExternalId>> {
private final ExternalIdReader externalIdReader;
@Inject
Loader(ExternalIdReader externalIdReader) {
this.externalIdReader = externalIdReader;
}
@Override
public ImmutableSetMultimap<Account.Id, ExternalId> load(ObjectId notesRev) throws Exception {
Multimap<Account.Id, ExternalId> extIdsByAccount =
MultimapBuilder.hashKeys().arrayListValues().build();
for (ExternalId extId : externalIdReader.all(notesRev)) {
extIdsByAccount.put(extId.accountId(), extId);
}
return ImmutableSetMultimap.copyOf(extIdsByAccount);
}
}
}