| // 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.lucene; |
| |
| import static com.google.gerrit.server.index.account.AccountField.ID; |
| |
| import com.google.gerrit.index.QueryOptions; |
| import com.google.gerrit.index.Schema; |
| import com.google.gerrit.index.query.DataSource; |
| import com.google.gerrit.index.query.Predicate; |
| import com.google.gerrit.index.query.QueryParseException; |
| import com.google.gerrit.reviewdb.client.Account; |
| import com.google.gerrit.server.account.AccountCache; |
| import com.google.gerrit.server.account.AccountState; |
| import com.google.gerrit.server.config.GerritServerConfig; |
| import com.google.gerrit.server.config.SitePaths; |
| import com.google.gerrit.server.index.IndexUtils; |
| import com.google.gerrit.server.index.account.AccountIndex; |
| import com.google.gwtorm.server.OrmException; |
| import com.google.gwtorm.server.ResultSet; |
| import com.google.inject.Inject; |
| import com.google.inject.Provider; |
| import com.google.inject.assistedinject.Assisted; |
| import java.io.IOException; |
| import java.nio.file.Path; |
| import java.util.ArrayList; |
| import java.util.Collections; |
| import java.util.Iterator; |
| import java.util.List; |
| import java.util.concurrent.ExecutionException; |
| import org.apache.lucene.document.Document; |
| import org.apache.lucene.index.Term; |
| import org.apache.lucene.search.IndexSearcher; |
| import org.apache.lucene.search.Query; |
| import org.apache.lucene.search.ScoreDoc; |
| import org.apache.lucene.search.SearcherFactory; |
| import org.apache.lucene.search.Sort; |
| import org.apache.lucene.search.SortField; |
| import org.apache.lucene.search.TopFieldDocs; |
| import org.apache.lucene.store.Directory; |
| import org.apache.lucene.store.FSDirectory; |
| import org.apache.lucene.store.RAMDirectory; |
| import org.eclipse.jgit.lib.Config; |
| import org.slf4j.Logger; |
| import org.slf4j.LoggerFactory; |
| |
| public class LuceneAccountIndex extends AbstractLuceneIndex<Account.Id, AccountState> |
| implements AccountIndex { |
| private static final Logger log = LoggerFactory.getLogger(LuceneAccountIndex.class); |
| |
| private static final String ACCOUNTS = "accounts"; |
| |
| private static final String ID_SORT_FIELD = sortFieldName(ID); |
| |
| private static Term idTerm(AccountState as) { |
| return idTerm(as.getAccount().getId()); |
| } |
| |
| private static Term idTerm(Account.Id id) { |
| return QueryBuilder.intTerm(ID.getName(), id.get()); |
| } |
| |
| private final GerritIndexWriterConfig indexWriterConfig; |
| private final QueryBuilder<AccountState> queryBuilder; |
| private final Provider<AccountCache> accountCache; |
| |
| private static Directory dir(Schema<AccountState> schema, Config cfg, SitePaths sitePaths) |
| throws IOException { |
| if (LuceneIndexModule.isInMemoryTest(cfg)) { |
| return new RAMDirectory(); |
| } |
| Path indexDir = LuceneVersionManager.getDir(sitePaths, ACCOUNTS, schema); |
| return FSDirectory.open(indexDir); |
| } |
| |
| @Inject |
| LuceneAccountIndex( |
| @GerritServerConfig Config cfg, |
| SitePaths sitePaths, |
| Provider<AccountCache> accountCache, |
| @Assisted Schema<AccountState> schema) |
| throws IOException { |
| super( |
| schema, |
| sitePaths, |
| dir(schema, cfg, sitePaths), |
| ACCOUNTS, |
| null, |
| new GerritIndexWriterConfig(cfg, ACCOUNTS), |
| new SearcherFactory()); |
| this.accountCache = accountCache; |
| |
| indexWriterConfig = new GerritIndexWriterConfig(cfg, ACCOUNTS); |
| queryBuilder = new QueryBuilder<>(schema, indexWriterConfig.getAnalyzer()); |
| } |
| |
| @Override |
| public void replace(AccountState as) throws IOException { |
| try { |
| replace(idTerm(as), toDocument(as)).get(); |
| } catch (ExecutionException | InterruptedException e) { |
| throw new IOException(e); |
| } |
| } |
| |
| @Override |
| public void delete(Account.Id key) throws IOException { |
| try { |
| delete(idTerm(key)).get(); |
| } catch (ExecutionException | InterruptedException e) { |
| throw new IOException(e); |
| } |
| } |
| |
| @Override |
| public DataSource<AccountState> getSource(Predicate<AccountState> p, QueryOptions opts) |
| throws QueryParseException { |
| return new QuerySource( |
| opts, |
| queryBuilder.toQuery(p), |
| new Sort(new SortField(ID_SORT_FIELD, SortField.Type.LONG, true))); |
| } |
| |
| private class QuerySource implements DataSource<AccountState> { |
| private final QueryOptions opts; |
| private final Query query; |
| private final Sort sort; |
| |
| private QuerySource(QueryOptions opts, Query query, Sort sort) { |
| this.opts = opts; |
| this.query = query; |
| this.sort = sort; |
| } |
| |
| @Override |
| public int getCardinality() { |
| // TODO(dborowitz): In contrast to the comment in |
| // LuceneChangeIndex.QuerySource#getCardinality, at this point I actually |
| // think we might just want to remove getCardinality. |
| return 10; |
| } |
| |
| @Override |
| public ResultSet<AccountState> read() throws OrmException { |
| IndexSearcher searcher = null; |
| try { |
| searcher = acquire(); |
| int realLimit = opts.start() + opts.limit(); |
| TopFieldDocs docs = searcher.search(query, realLimit, sort); |
| List<AccountState> result = new ArrayList<>(docs.scoreDocs.length); |
| for (int i = opts.start(); i < docs.scoreDocs.length; i++) { |
| ScoreDoc sd = docs.scoreDocs[i]; |
| Document doc = searcher.doc(sd.doc, IndexUtils.accountFields(opts)); |
| result.add(toAccountState(doc)); |
| } |
| final List<AccountState> r = Collections.unmodifiableList(result); |
| return new ResultSet<AccountState>() { |
| @Override |
| public Iterator<AccountState> iterator() { |
| return r.iterator(); |
| } |
| |
| @Override |
| public List<AccountState> toList() { |
| return r; |
| } |
| |
| @Override |
| public void close() { |
| // Do nothing. |
| } |
| }; |
| } catch (IOException e) { |
| throw new OrmException(e); |
| } finally { |
| if (searcher != null) { |
| try { |
| release(searcher); |
| } catch (IOException e) { |
| log.warn("cannot release Lucene searcher", e); |
| } |
| } |
| } |
| } |
| } |
| |
| private AccountState toAccountState(Document doc) { |
| Account.Id id = new Account.Id(doc.getField(ID.getName()).numericValue().intValue()); |
| // Use the AccountCache rather than depending on any stored fields in the |
| // document (of which there shouldn't be any). The most expensive part to |
| // compute anyway is the effective group IDs, and we don't have a good way |
| // to reindex when those change. |
| return accountCache.get().get(id); |
| } |
| } |