| // 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.lucene; |
| |
| import static com.google.common.collect.Iterables.getOnlyElement; |
| import static com.google.gerrit.index.project.ProjectField.NAME_SPEC; |
| |
| import com.google.common.collect.ImmutableSet; |
| import com.google.gerrit.common.Nullable; |
| import com.google.gerrit.entities.Project; |
| import com.google.gerrit.exceptions.StorageException; |
| import com.google.gerrit.index.QueryOptions; |
| import com.google.gerrit.index.Schema; |
| import com.google.gerrit.index.Schema.Values; |
| import com.google.gerrit.index.SchemaFieldDefs.SchemaField; |
| import com.google.gerrit.index.project.ProjectData; |
| import com.google.gerrit.index.project.ProjectIndex; |
| 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.server.config.GerritServerConfig; |
| import com.google.gerrit.server.config.SitePaths; |
| import com.google.gerrit.server.index.IndexUtils; |
| import com.google.gerrit.server.index.options.AutoFlush; |
| import com.google.gerrit.server.project.ProjectCache; |
| import com.google.gerrit.server.project.ProjectState; |
| 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.concurrent.ExecutionException; |
| import org.apache.lucene.document.Document; |
| import org.apache.lucene.document.SortedDocValuesField; |
| import org.apache.lucene.index.Term; |
| import org.apache.lucene.search.SearcherFactory; |
| import org.apache.lucene.search.Sort; |
| import org.apache.lucene.search.SortField; |
| import org.apache.lucene.store.ByteBuffersDirectory; |
| import org.apache.lucene.store.Directory; |
| import org.apache.lucene.store.FSDirectory; |
| import org.apache.lucene.util.BytesRef; |
| import org.eclipse.jgit.lib.Config; |
| |
| public class LuceneProjectIndex extends AbstractLuceneIndex<Project.NameKey, ProjectData> |
| implements ProjectIndex { |
| private static final String PROJECTS = "projects"; |
| |
| private static final String NAME_SORT_FIELD = sortFieldName(NAME_SPEC); |
| |
| private static Term idTerm(ProjectData projectState) { |
| return idTerm(projectState.getProject().getNameKey()); |
| } |
| |
| private static Term idTerm(Project.NameKey nameKey) { |
| return QueryBuilder.stringTerm(NAME_SPEC.getName(), nameKey.get()); |
| } |
| |
| private final GerritIndexWriterConfig indexWriterConfig; |
| private final QueryBuilder<ProjectData> queryBuilder; |
| private final Provider<ProjectCache> projectCache; |
| |
| private static Directory dir(Schema<ProjectData> schema, Config cfg, SitePaths sitePaths) |
| throws IOException { |
| if (LuceneIndexModule.isInMemoryTest(cfg)) { |
| return new ByteBuffersDirectory(); |
| } |
| Path indexDir = LuceneVersionManager.getDir(sitePaths, PROJECTS, schema); |
| return FSDirectory.open(indexDir); |
| } |
| |
| @Inject |
| LuceneProjectIndex( |
| @GerritServerConfig Config cfg, |
| SitePaths sitePaths, |
| Provider<ProjectCache> projectCache, |
| @Assisted Schema<ProjectData> schema, |
| AutoFlush autoFlush) |
| throws IOException { |
| super( |
| schema, |
| sitePaths, |
| dir(schema, cfg, sitePaths), |
| PROJECTS, |
| ImmutableSet.of(), |
| null, |
| new GerritIndexWriterConfig(cfg, PROJECTS), |
| new SearcherFactory(), |
| autoFlush, |
| ProjectIndex.ENTITY_TO_KEY); |
| this.projectCache = projectCache; |
| |
| indexWriterConfig = new GerritIndexWriterConfig(cfg, PROJECTS); |
| queryBuilder = new QueryBuilder<>(schema, indexWriterConfig.getAnalyzer()); |
| } |
| |
| @Override |
| void add(Document doc, Values<ProjectData> values) { |
| // Add separate DocValues field for the field that is needed for sorting. |
| SchemaField<ProjectData, ?> f = values.getField(); |
| if (f == NAME_SPEC) { |
| String value = (String) getOnlyElement(values.getValues()); |
| doc.add(new SortedDocValuesField(NAME_SORT_FIELD, new BytesRef(value))); |
| } |
| super.add(doc, values); |
| } |
| |
| @Override |
| public void replace(ProjectData projectState) { |
| try { |
| replace(idTerm(projectState), toDocument(projectState)).get(); |
| } catch (ExecutionException | InterruptedException e) { |
| throw new StorageException(e); |
| } |
| } |
| |
| @Override |
| public void insert(ProjectData projectState) { |
| try { |
| insert(toDocument(projectState)).get(); |
| } catch (ExecutionException | InterruptedException e) { |
| throw new StorageException(e); |
| } |
| } |
| |
| @Override |
| public void delete(Project.NameKey nameKey) { |
| try { |
| delete(idTerm(nameKey)).get(); |
| } catch (ExecutionException | InterruptedException e) { |
| throw new StorageException(e); |
| } |
| } |
| |
| @Override |
| public DataSource<ProjectData> getSource(Predicate<ProjectData> p, QueryOptions opts) |
| throws QueryParseException { |
| return new LuceneQuerySource( |
| opts.filterFields(IndexUtils::projectFields), |
| queryBuilder.toQuery(p), |
| new Sort(new SortField(NAME_SORT_FIELD, SortField.Type.STRING, false))); |
| } |
| |
| @Nullable |
| @Override |
| protected ProjectData fromDocument(Document doc) { |
| Project.NameKey nameKey = Project.nameKey(doc.getField(NAME_SPEC.getName()).stringValue()); |
| return projectCache.get().get(nameKey).map(ProjectState::toProjectData).orElse(null); |
| } |
| } |