org.hibernate.search.util.jmx.impl.JMXRegistrar Maven / Gradle / Ivy
/*
* Hibernate Search, full-text search for your domain model
*
* License: GNU Lesser General Public License (LGPL), version 2.1 or later
* See the lgpl.txt file in the root directory or .
*/
package org.hibernate.search.util.jmx.impl;
import java.lang.management.ManagementFactory;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.atomic.LongAdder;
import javax.management.MBeanServer;
import javax.management.MalformedObjectNameException;
import javax.management.ObjectName;
import javax.management.StandardMBean;
import org.hibernate.search.batchindexing.MassIndexerProgressMonitor;
import org.hibernate.search.exception.SearchException;
import org.hibernate.search.jmx.IndexingProgressMonitorMBean;
import org.hibernate.search.jmx.StatisticsInfoMBean;
import org.hibernate.search.stat.Statistics;
import org.hibernate.search.util.StringHelper;
import org.hibernate.search.util.logging.impl.Log;
import org.hibernate.search.util.logging.impl.LoggerFactory;
import java.lang.invoke.MethodHandles;
/**
* Helper class to register JMX beans.
*
* @author Hardy Ferentschik
*/
public final class JMXRegistrar {
private static final Log log = LoggerFactory.make( MethodHandles.lookup() );
private JMXRegistrar() {
}
public static String buildMBeanName(String defaultName, String suffix) {
String objectName = defaultName;
if ( !StringHelper.isEmpty( suffix ) ) {
objectName += "[" + suffix + "]";
}
return objectName;
}
/**
* Registers the specified object with the given name to the MBean server.
*
* @param the type of the object interface
* @param object the object to register
* @param beanInterface the Management Interface exported by this MBean's implementation.
* @param name the object name to register the bean under
*
* @return The registered object name
*/
public static String registerMBean(T object, Class beanInterface, String name) {
ObjectName objectName = createObjectName( name );
MBeanServer mbs = ManagementFactory.getPlatformMBeanServer();
try {
StandardMBean mbean = new StandardMBean( object, beanInterface );
mbs.registerMBean( mbean, objectName );
}
catch (Exception e) {
throw new SearchException( "Unable to enable MBean for Hibernate Search", e );
}
return objectName.toString();
}
/**
* Unregister the MBean with the specified name.
*
* @param name The name of the bean to unregister. The {@code name} cannot be {@code null}
*
* @throws java.lang.IllegalArgumentException In case the object name is {@code null}
*/
public static void unRegisterMBean(String name) {
if ( name == null ) {
throw new IllegalArgumentException( "The object name cannot be null" );
}
ObjectName objectName = createObjectName( name );
MBeanServer mbs = ManagementFactory.getPlatformMBeanServer();
if ( mbs.isRegistered( objectName ) ) {
try {
mbs.unregisterMBean( objectName );
}
catch (Exception e) {
log.unableToUnregisterExistingMBean( name, e );
}
}
}
/**
* Checks whether a bean is registered under the given name.
*
* @param name the object name to check (as string)
*
* @return {@code true} is there is a bean registered under the given name, {@code false} otherwise.
*
* @throws java.lang.IllegalArgumentException In case the object name is {@code null}
*/
public static boolean isNameRegistered(String name) {
if ( name == null ) {
throw new IllegalArgumentException( "The object name cannot be null" );
}
ObjectName objectName = createObjectName( name );
MBeanServer mbs = ManagementFactory.getPlatformMBeanServer();
return mbs.isRegistered( objectName );
}
private static ObjectName createObjectName(String name) {
ObjectName objectName;
try {
objectName = new ObjectName( name );
}
catch (MalformedObjectNameException e) {
throw new SearchException( "Invalid JMX Bean name: " + name, e );
}
return objectName;
}
/**
* @author Hardy Ferentschik
*/
public static class StatisticsInfo implements StatisticsInfoMBean {
private final Statistics delegate;
public StatisticsInfo(Statistics delegate) {
this.delegate = delegate;
}
@Override
public void clear() {
delegate.clear();
}
@Override
public long getSearchQueryExecutionCount() {
return delegate.getSearchQueryExecutionCount();
}
@Override
public long getSearchQueryTotalTime() {
return delegate.getSearchQueryTotalTime();
}
@Override
public long getSearchQueryExecutionMaxTime() {
return delegate.getSearchQueryExecutionMaxTime();
}
@Override
public long getSearchQueryExecutionAvgTime() {
return delegate.getSearchQueryExecutionAvgTime();
}
@Override
public String getSearchQueryExecutionMaxTimeQueryString() {
return delegate.getSearchQueryExecutionMaxTimeQueryString();
}
@Override
public long getObjectLoadingTotalTime() {
return delegate.getObjectLoadingTotalTime();
}
@Override
public long getObjectLoadingExecutionMaxTime() {
return delegate.getObjectLoadingExecutionMaxTime();
}
@Override
public long getObjectLoadingExecutionAvgTime() {
return delegate.getObjectLoadingExecutionAvgTime();
}
@Override
public long getObjectsLoadedCount() {
return delegate.getObjectsLoadedCount();
}
@Override
public boolean isStatisticsEnabled() {
return delegate.isStatisticsEnabled();
}
@Override
public void setStatisticsEnabled(boolean b) {
delegate.setStatisticsEnabled( b );
}
@Override
public String getSearchVersion() {
return delegate.getSearchVersion();
}
@Override
public Set getIndexedClassNames() {
return delegate.getIndexedClassNames();
}
@Override
public int getNumberOfIndexedEntities(String entity) {
return delegate.getNumberOfIndexedEntities( entity );
}
@Override
public Map indexedEntitiesCount() {
return delegate.indexedEntitiesCount();
}
@Override
public long getIndexSize(String indexName) {
return delegate.getIndexSize( indexName );
}
@Override
public Map indexSizes() {
return delegate.indexSizes();
}
}
/**
* A JMX based mass indexer progress monitor. This monitor will allow you to follow mass indexing progress via JMX.
*
* @author Hardy Ferentschik
*/
public static class IndexingProgressMonitor implements IndexingProgressMonitorMBean, MassIndexerProgressMonitor {
private static final Log log = LoggerFactory.make( MethodHandles.lookup() );
private final LongAdder documentsDoneCounter = new LongAdder();
private final LongAdder documentsBuiltCounter = new LongAdder();
private final LongAdder totalCounter = new LongAdder();
private final LongAdder entitiesLoadedCounter = new LongAdder();
private final String registeredName;
public IndexingProgressMonitor() {
String name = IndexingProgressMonitorMBean.INDEXING_PROGRESS_MONITOR_MBEAN_OBJECT_NAME;
if ( isNameRegistered( name ) ) {
name = name + "@" + Integer.toHexString( hashCode() ); // make the name unique in case there are multiple mass indexers at the same time
}
registeredName = registerMBean( this, IndexingProgressMonitorMBean.class, name );
}
@Override
public final void documentsAdded(long increment) {
documentsDoneCounter.add( increment );
}
@Override
public final void documentsBuilt(int number) {
documentsBuiltCounter.add( number );
}
@Override
public final void entitiesLoaded(int size) {
entitiesLoadedCounter.add( size );
}
@Override
public final void addToTotalCount(long count) {
totalCounter.add( count );
}
@Override
public final void indexingCompleted() {
log.indexingCompletedAndMBeanUnregistered( totalCounter.longValue() );
unRegisterMBean( registeredName );
}
@Override
public final long getLoadedEntitiesCount() {
return entitiesLoadedCounter.longValue();
}
@Override
public final long getDocumentsAddedCount() {
return documentsDoneCounter.longValue();
}
@Override
public final long getNumberOfEntitiesToIndex() {
return totalCounter.longValue();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy