Package org.hibernate.search.impl

Source Code of org.hibernate.search.impl.MassIndexerImpl

/*
* Hibernate, Relational Persistence for Idiomatic Java
*
* Copyright (c) 2010, Red Hat, Inc. and/or its affiliates or third-party contributors as
* indicated by the @author tags or express copyright attribution
* statements applied by the authors.  All third-party contributions are
* distributed under license by Red Hat, Inc.
*
* This copyrighted material is made available to anyone wishing to use, modify,
* copy, or redistribute it subject to the terms and conditions of the GNU
* Lesser General Public License, as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License
* for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this distribution; if not, write to:
* Free Software Foundation, Inc.
* 51 Franklin Street, Fifth Floor
* Boston, MA  02110-1301  USA
*/
package org.hibernate.search.impl;

import java.util.HashSet;
import java.util.Set;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;

import org.hibernate.engine.spi.SessionFactoryImplementor;
import org.hibernate.search.engine.spi.SearchFactoryImplementor;
import org.hibernate.search.util.logging.impl.Log;
import org.hibernate.CacheMode;
import org.hibernate.search.MassIndexer;
import org.hibernate.search.batchindexing.impl.BatchCoordinator;
import org.hibernate.search.batchindexing.impl.Executors;
import org.hibernate.search.batchindexing.MassIndexerProgressMonitor;
import org.hibernate.search.jmx.IndexingProgressMonitor;
import org.hibernate.search.util.logging.impl.LoggerFactory;

/**
* Prepares and configures a BatchIndexingWorkspace to start rebuilding
* the indexes for all entity instances in the database.
* The type of these entities is either all indexed entities or a
* subset, always including all subtypes.
*
* @author Sanne Grinovero
*/
public class MassIndexerImpl implements MassIndexer {

  private static final Log log = LoggerFactory.make();

  private final SearchFactoryImplementor searchFactoryImplementor;
  private final SessionFactoryImplementor sessionFactory;

  protected Set<Class<?>> rootEntities = new HashSet<Class<?>>();

  // default settings defined here:
  private int typesToIndexInParallel = 1;
  private int objectLoadingThreads = 2; //loading the main entity
  private int collectionLoadingThreads = 4; //also responsible for loading of lazy @IndexedEmbedded collections
  private int objectLoadingBatchSize = 10;
  private long objectsLimit = 0; //means no limit at all
  private CacheMode cacheMode = CacheMode.IGNORE;
  private boolean optimizeAtEnd = true;
  private boolean purgeAtStart = true;
  private boolean optimizeAfterPurge = true;
  private MassIndexerProgressMonitor monitor;
  private int idFetchSize = 100; //reasonable default as we only load IDs

  protected MassIndexerImpl(SearchFactoryImplementor searchFactory, SessionFactoryImplementor sessionFactory, Class<?>... entities) {
    this.searchFactoryImplementor = searchFactory;
    this.sessionFactory = sessionFactory;
    rootEntities = toRootEntities( searchFactoryImplementor, entities );
    if ( searchFactoryImplementor.isJMXEnabled() ) {
      monitor = new IndexingProgressMonitor();
    }
    else {
      monitor = new SimpleIndexingProgressMonitor();
    }
  }

  /**
   * From the set of classes a new set is built containing all indexed
   * subclasses, but removing then all subtypes of indexed entities.
   *
   * @param selection
   *
   * @return a new set of entities
   */
  private static Set<Class<?>> toRootEntities(SearchFactoryImplementor searchFactoryImplementor, Class<?>... selection) {
    Set<Class<?>> entities = new HashSet<Class<?>>();
    //first build the "entities" set containing all indexed subtypes of "selection".
    for ( Class<?> entityType : selection ) {
      Set<Class<?>> targetedClasses = searchFactoryImplementor.getIndexedTypesPolymorphic(
          new Class[] {
              entityType
          }
      );
      if ( targetedClasses.isEmpty() ) {
        String msg = entityType.getName() + " is not an indexed entity or a subclass of an indexed entity";
        throw new IllegalArgumentException( msg );
      }
      entities.addAll( targetedClasses );
    }
    Set<Class<?>> cleaned = new HashSet<Class<?>>();
    Set<Class<?>> toRemove = new HashSet<Class<?>>();
    //now remove all repeated types to avoid duplicate loading by polymorphic query loading
    for ( Class<?> type : entities ) {
      boolean typeIsOk = true;
      for ( Class<?> existing : cleaned ) {
        if ( existing.isAssignableFrom( type ) ) {
          typeIsOk = false;
          break;
        }
        if ( type.isAssignableFrom( existing ) ) {
          toRemove.add( existing );
        }
      }
      if ( typeIsOk ) {
        cleaned.add( type );
      }
    }
    cleaned.removeAll( toRemove );
    log.debugf( "Targets for indexing job: %s", cleaned );
    return cleaned;
  }

  @Override
  public MassIndexer typesToIndexInParallel(int numberOfThreads) {
    if ( numberOfThreads < 1 ) {
      throw new IllegalArgumentException( "numberOfThreads must be at least 1" );
    }
    this.typesToIndexInParallel = Math.min( numberOfThreads, rootEntities.size() );
    return this;
  }

  @Override
  public MassIndexer cacheMode(CacheMode cacheMode) {
    if ( cacheMode == null ) {
      throw new IllegalArgumentException( "cacheMode must not be null" );
    }
    this.cacheMode = cacheMode;
    return this;
  }

  @Override
  public MassIndexer threadsToLoadObjects(int numberOfThreads) {
    if ( numberOfThreads < 1 ) {
      throw new IllegalArgumentException( "numberOfThreads must be at least 1" );
    }
    this.objectLoadingThreads = numberOfThreads;
    return this;
  }

  @Override
  public MassIndexer batchSizeToLoadObjects(int batchSize) {
    if ( batchSize < 1 ) {
      throw new IllegalArgumentException( "batchSize must be at least 1" );
    }
    this.objectLoadingBatchSize = batchSize;
    return this;
  }

  @Override
  public MassIndexer threadsForSubsequentFetching(int numberOfThreads) {
    if ( numberOfThreads < 1 ) {
      throw new IllegalArgumentException( "numberOfThreads must be at least 1" );
    }
    this.collectionLoadingThreads = numberOfThreads;
    return this;
  }

  @Override
  @Deprecated
  public MassIndexer threadsForIndexWriter(int numberOfThreads) {
    log.massIndexerIndexWriterThreadsIgnored();
    return this;
  }

  @Override
  public MassIndexer progressMonitor(MassIndexerProgressMonitor monitor) {
    this.monitor = monitor;
    return this;
  }

  @Override
  public MassIndexer optimizeOnFinish(boolean optimize) {
    this.optimizeAtEnd = optimize;
    return this;
  }

  @Override
  public MassIndexer optimizeAfterPurge(boolean optimize) {
    this.optimizeAfterPurge = optimize;
    return this;
  }

  @Override
  public MassIndexer purgeAllOnStart(boolean purgeAll) {
    this.purgeAtStart = purgeAll;
    return this;
  }

  @Override
  public Future<?> start() {
    BatchCoordinator coordinator = createCoordinator();
    ExecutorService executor = Executors.newFixedThreadPool( 1, "batch coordinator" );
    try {
      Future<?> submit = executor.submit( coordinator );
      return submit;
    }
    finally {
      executor.shutdown();
    }
  }

  @Override
  public void startAndWait() throws InterruptedException {
    BatchCoordinator coordinator = createCoordinator();
    coordinator.run();
    if ( Thread.currentThread().isInterrupted() ) {
      throw new InterruptedException();
    }
  }

  protected BatchCoordinator createCoordinator() {
    return new BatchCoordinator(
        rootEntities, searchFactoryImplementor, sessionFactory,
        typesToIndexInParallel, objectLoadingThreads, collectionLoadingThreads,
        cacheMode, objectLoadingBatchSize, objectsLimit,
        optimizeAtEnd, purgeAtStart, optimizeAfterPurge,
        monitor, idFetchSize
    );
  }

  @Override
  public MassIndexer limitIndexedObjectsTo(long maximum) {
    this.objectsLimit = maximum;
    return this;
  }

  @Override
  public MassIndexer idFetchSize(int idFetchSize) {
    // don't check for positive/zero values as it's actually used by some databases
    // as special values which might be useful.
    this.idFetchSize = idFetchSize;
    return this;
  }
}
TOP

Related Classes of org.hibernate.search.impl.MassIndexerImpl

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.