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

com.google.gerrit.server.index.account.StalenessChecker Maven / Gradle / Ivy

There is a newer version: 3.10.0-rc4
Show newest version
// Copyright (C) 2017 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.index.account;

import static com.google.common.base.Preconditions.checkState;
import static java.nio.charset.StandardCharsets.UTF_8;
import static java.util.Objects.requireNonNull;

import com.google.common.base.Splitter;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ListMultimap;
import com.google.common.collect.MultimapBuilder;
import com.google.gerrit.index.IndexConfig;
import com.google.gerrit.index.QueryOptions;
import com.google.gerrit.index.RefState;
import com.google.gerrit.index.query.FieldBundle;
import com.google.gerrit.reviewdb.client.Account;
import com.google.gerrit.reviewdb.client.Project;
import com.google.gerrit.reviewdb.client.RefNames;
import com.google.gerrit.server.account.externalids.ExternalId;
import com.google.gerrit.server.account.externalids.ExternalIds;
import com.google.gerrit.server.config.AllUsersName;
import com.google.gerrit.server.git.GitRepositoryManager;
import com.google.gerrit.server.index.IndexUtils;
import com.google.inject.Inject;
import com.google.inject.Singleton;
import java.io.IOException;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.Ref;
import org.eclipse.jgit.lib.Repository;

/**
 * Checks if documents in the account index are stale.
 *
 * 

An index document is considered stale if the stored ref state differs from the SHA1 of the * user branch or if the stored external ID states don't match with the external IDs of the account * from the refs/meta/external-ids branch. */ @Singleton public class StalenessChecker { public static final ImmutableSet FIELDS = ImmutableSet.of( AccountField.ID.getName(), AccountField.REF_STATE.getName(), AccountField.EXTERNAL_ID_STATE.getName()); private final AccountIndexCollection indexes; private final GitRepositoryManager repoManager; private final AllUsersName allUsersName; private final ExternalIds externalIds; private final IndexConfig indexConfig; @Inject StalenessChecker( AccountIndexCollection indexes, GitRepositoryManager repoManager, AllUsersName allUsersName, ExternalIds externalIds, IndexConfig indexConfig) { this.indexes = indexes; this.repoManager = repoManager; this.allUsersName = allUsersName; this.externalIds = externalIds; this.indexConfig = indexConfig; } public boolean isStale(Account.Id id) throws IOException { AccountIndex i = indexes.getSearchIndex(); if (i == null) { // No index; caller couldn't do anything if it is stale. return false; } if (!i.getSchema().hasField(AccountField.REF_STATE) || !i.getSchema().hasField(AccountField.EXTERNAL_ID_STATE)) { // Index version not new enough for this check. return false; } Optional result = i.getRaw(id, QueryOptions.create(indexConfig, 0, 1, IndexUtils.accountFields(FIELDS))); if (!result.isPresent()) { // The document is missing in the index. try (Repository repo = repoManager.openRepository(allUsersName)) { Ref ref = repo.exactRef(RefNames.refsUsers(id)); // Stale if the account actually exists. return ref != null; } } for (Map.Entry e : RefState.parseStates(result.get().getValue(AccountField.REF_STATE)).entries()) { try (Repository repo = repoManager.openRepository(e.getKey())) { if (!e.getValue().match(repo)) { // Ref was modified since the account was indexed. return true; } } } Set extIds = externalIds.byAccount(id); ListMultimap extIdStates = parseExternalIdStates(result.get().getValue(AccountField.EXTERNAL_ID_STATE)); if (extIdStates.size() != extIds.size()) { // External IDs of the account were modified since the account was indexed. return true; } for (ExternalId extId : extIds) { if (!extIdStates.containsEntry(extId.key().sha1(), extId.blobId())) { // External IDs of the account were modified since the account was indexed. return true; } } return false; } public static ListMultimap parseExternalIdStates( Iterable extIdStates) { ListMultimap result = MultimapBuilder.hashKeys().arrayListValues().build(); if (extIdStates == null) { return result; } for (byte[] b : extIdStates) { requireNonNull(b, "invalid external ID state"); String s = new String(b, UTF_8); List parts = Splitter.on(':').splitToList(s); checkState(parts.size() == 2, "invalid external ID state: %s", s); result.put(ObjectId.fromString(parts.get(0)), ObjectId.fromString(parts.get(1))); } return result; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy