blob: 09d1bbdc4f9740ee6e947e5806505444c3d3b81f [file] [log] [blame]
/*
* Copyright (c) 2010, Rickard Öberg. All Rights Reserved.
*
* 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 org.apache.zest.library.jmx;
import java.lang.reflect.AccessibleObject;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import javax.management.Attribute;
import javax.management.AttributeList;
import javax.management.AttributeNotFoundException;
import javax.management.Descriptor;
import javax.management.DynamicMBean;
import javax.management.InstanceAlreadyExistsException;
import javax.management.InvalidAttributeValueException;
import javax.management.MBeanAttributeInfo;
import javax.management.MBeanException;
import javax.management.MBeanInfo;
import javax.management.MBeanOperationInfo;
import javax.management.MBeanParameterInfo;
import javax.management.MBeanRegistrationException;
import javax.management.MBeanServer;
import javax.management.MalformedObjectNameException;
import javax.management.NotCompliantMBeanException;
import javax.management.ObjectName;
import javax.management.ReflectionException;
import javax.management.modelmbean.DescriptorSupport;
import org.apache.zest.api.Qi4j;
import org.apache.zest.api.activation.Activation;
import org.apache.zest.api.activation.ActivatorAdapter;
import org.apache.zest.api.activation.Activators;
import org.apache.zest.api.association.AssociationStateHolder;
import org.apache.zest.api.composite.Composite;
import org.apache.zest.api.composite.CompositeInstance;
import org.apache.zest.api.configuration.Configuration;
import org.apache.zest.api.entity.EntityComposite;
import org.apache.zest.api.entity.EntityDescriptor;
import org.apache.zest.api.injection.scope.Service;
import org.apache.zest.api.injection.scope.Structure;
import org.apache.zest.api.mixin.Mixins;
import org.apache.zest.api.property.Property;
import org.apache.zest.api.property.PropertyDescriptor;
import org.apache.zest.api.service.ServiceComposite;
import org.apache.zest.api.service.ServiceDescriptor;
import org.apache.zest.api.service.ServiceReference;
import org.apache.zest.api.structure.Application;
import org.apache.zest.api.structure.Module;
import org.apache.zest.api.type.EnumType;
import org.apache.zest.api.unitofwork.UnitOfWork;
import org.apache.zest.api.unitofwork.UnitOfWorkCompletionException;
import org.apache.zest.api.unitofwork.UnitOfWorkFactory;
import org.apache.zest.spi.Qi4jSPI;
import static org.apache.zest.functional.Iterables.first;
/**
* Expose ConfigurationComposites through JMX.
* Allow configurations to be edited, and the services to be restarted.
*/
@Mixins( ConfigurationManagerService.Mixin.class )
@Activators( ConfigurationManagerService.Activator.class )
public interface ConfigurationManagerService
extends ServiceComposite
{
void exportConfigurableServices()
throws Exception;
void unexportConfigurableServices()
throws Exception;
class Activator
extends ActivatorAdapter<ServiceReference<ConfigurationManagerService>>
{
@Override
public void afterActivation( ServiceReference<ConfigurationManagerService> activated )
throws Exception
{
activated.get().exportConfigurableServices();
}
@Override
public void beforePassivation( ServiceReference<ConfigurationManagerService> passivating )
throws Exception
{
passivating.get().unexportConfigurableServices();
}
}
abstract class Mixin
implements ConfigurationManagerService
{
@Structure
UnitOfWorkFactory uowf;
@Service
MBeanServer server;
@Structure
Application application;
@Structure
Qi4jSPI spi;
@Service
Iterable<ServiceReference<?>> configurableServices;
private List<ObjectName> configurationNames = new ArrayList<ObjectName>();
@Override
public void exportConfigurableServices()
throws NotCompliantMBeanException, MBeanRegistrationException, InstanceAlreadyExistsException, MalformedObjectNameException
{
for( ServiceReference<?> configurableService : configurableServices )
{
Object service = configurableService.get();
if( !( service instanceof Composite ) )
{
continue; // Skip imported services
}
// Check if service has configuration
CompositeInstance compositeInstance = Qi4j.FUNCTION_COMPOSITE_INSTANCE_OF.map( (Composite) service );
try
{
Configuration config = compositeInstance.newProxy( Configuration.class );
}
catch( Exception e )
{
// Service does not have configuration
continue;
}
String serviceClass = first(compositeInstance.types()).getName();
String name = configurableService.identity();
ServiceDescriptor serviceDescriptor = spi.serviceDescriptorFor( configurableService );
Module module = spi.moduleOf( configurableService );
Class<Object> configurationClass = serviceDescriptor.configurationType();
if( configurationClass != null )
{
EntityDescriptor descriptor = module.entityDescriptor( configurationClass.getName() );
List<MBeanAttributeInfo> attributes = new ArrayList<MBeanAttributeInfo>();
Map<String, AccessibleObject> properties = new HashMap<String, AccessibleObject>();
for( PropertyDescriptor persistentProperty : descriptor.state().properties() )
{
if( !persistentProperty.isImmutable() )
{
String propertyName = persistentProperty.qualifiedName().name();
String type = persistentProperty.valueType().mainType().getName();
Descriptor attrDescriptor = new DescriptorSupport();
attrDescriptor.setField( "name", propertyName );
attrDescriptor.setField( "descriptorType", "attribute" );
if( persistentProperty.valueType() instanceof EnumType )
{
type = String.class.getName();
// Try to add legal values
try
{
Set<String> legalValues = new LinkedHashSet();
Class<?> enumType = getClass().getClassLoader()
.loadClass( persistentProperty.valueType().mainType().getName() );
for( Field field : enumType.getFields() )
{
legalValues.add( field.getName() );
}
attrDescriptor.setField( "legalValues", legalValues );
}
catch( ClassNotFoundException e )
{
// Ignore
e.printStackTrace();
}
}
attributes.add( new MBeanAttributeInfo( propertyName, type, propertyName, true, true, type.equals( "java.lang.Boolean" ), attrDescriptor ) );
properties.put( propertyName, persistentProperty.accessor() );
}
}
List<MBeanOperationInfo> operations = new ArrayList<MBeanOperationInfo>();
operations.add( new MBeanOperationInfo( "restart", "Restart service", new MBeanParameterInfo[ 0 ], "java.lang.String", MBeanOperationInfo.ACTION_INFO ) );
MBeanInfo mbeanInfo = new MBeanInfo( serviceClass, name, attributes.toArray( new MBeanAttributeInfo[ attributes
.size() ] ), null, operations.toArray( new MBeanOperationInfo[ operations.size() ] ), null );
Object mbean = new ConfigurableService( configurableService, mbeanInfo, name, properties );
ObjectName configurableServiceName;
ObjectName serviceName = Qi4jMBeans.findServiceName( server, application.name(), name );
if( serviceName != null )
{
configurableServiceName = new ObjectName( serviceName.toString() + ",name=Configuration" );
}
else
{
configurableServiceName = new ObjectName( "Configuration:name=" + name );
}
server.registerMBean( mbean, configurableServiceName );
configurationNames.add( configurableServiceName );
}
}
}
@Override
public void unexportConfigurableServices()
throws Exception
{
for( ObjectName configurableServiceName : configurationNames )
{
server.unregisterMBean( configurableServiceName );
}
}
abstract class EditableConfiguration
implements DynamicMBean
{
MBeanInfo info;
String identity;
Map<String, AccessibleObject> propertyNames;
EditableConfiguration( MBeanInfo info, String identity, Map<String, AccessibleObject> propertyNames )
{
this.info = info;
this.identity = identity;
this.propertyNames = propertyNames;
}
@Override
public Object getAttribute( String name )
throws AttributeNotFoundException, MBeanException, ReflectionException
{
UnitOfWork uow = uowf.newUnitOfWork();
try
{
EntityComposite configuration = uow.get( EntityComposite.class, identity );
AssociationStateHolder state = spi.stateOf( configuration );
AccessibleObject accessor = propertyNames.get( name );
Property<Object> property = state.propertyFor( accessor );
Object object = property.get();
if( object instanceof Enum )
{
object = object.toString();
}
return object;
}
catch( Exception ex )
{
throw new ReflectionException( ex, "Could not get attribute " + name );
}
finally
{
uow.discard();
}
}
@Override
public void setAttribute( Attribute attribute )
throws AttributeNotFoundException, InvalidAttributeValueException, MBeanException, ReflectionException
{
UnitOfWork uow = uowf.newUnitOfWork();
try
{
EntityComposite configuration = uow.get( EntityComposite.class, identity );
AssociationStateHolder state = spi.stateOf( (EntityComposite) configuration );
AccessibleObject accessor = propertyNames.get( attribute.getName() );
Property<Object> property = state.propertyFor( accessor );
PropertyDescriptor propertyDescriptor = spi.propertyDescriptorFor( property );
if( EnumType.isEnum( propertyDescriptor.type() ) )
{
property.set( Enum.valueOf( (Class<Enum>) propertyDescriptor.type(), attribute.getValue()
.toString() ) );
}
else
{
property.set( attribute.getValue() );
}
try
{
uow.complete();
}
catch( UnitOfWorkCompletionException e )
{
throw new ReflectionException( e );
}
}
finally
{
uow.discard();
}
}
@Override
public AttributeList getAttributes( String[] names )
{
AttributeList list = new AttributeList();
for( String name : names )
{
try
{
Object value = getAttribute( name );
list.add( new Attribute( name, value ) );
}
catch( AttributeNotFoundException e )
{
e.printStackTrace();
}
catch( MBeanException e )
{
e.printStackTrace();
}
catch( ReflectionException e )
{
e.printStackTrace();
}
}
return list;
}
@Override
public AttributeList setAttributes( AttributeList attributeList )
{
AttributeList list = new AttributeList();
for( int i = 0; i < list.size(); i++ )
{
Attribute attribute = (Attribute) list.get( i );
try
{
setAttribute( attribute );
list.add( attribute );
}
catch( AttributeNotFoundException e )
{
e.printStackTrace();
}
catch( InvalidAttributeValueException e )
{
e.printStackTrace();
}
catch( MBeanException e )
{
e.printStackTrace();
}
catch( ReflectionException e )
{
e.printStackTrace();
}
}
return list;
}
@Override
public MBeanInfo getMBeanInfo()
{
return info;
}
}
class ConfigurableService
extends EditableConfiguration
{
private ServiceReference<?> serviceRef;
ConfigurableService( ServiceReference<?> serviceReference,
MBeanInfo info,
String identity,
Map<String, AccessibleObject> propertyNames
)
{
super( info, identity, propertyNames );
this.serviceRef = serviceReference;
}
@Override
public Object invoke( String s, Object[] objects, String[] strings )
throws MBeanException, ReflectionException
{
if( s.equals( "restart" ) )
{
try
{
// Refresh and restart
if( serviceRef.isActive() )
{
// Refresh configuration
CompositeInstance compositeInstance = Qi4j.FUNCTION_COMPOSITE_INSTANCE_OF
.map( (Composite) serviceRef.get() );
compositeInstance.newProxy( Configuration.class ).refresh();
( (Activation) serviceRef ).passivate();
( (Activation) serviceRef ).activate();
}
return "Service restarted";
}
catch( Exception e )
{
return "Could not restart service:" + e.getMessage();
}
}
return "Unknown operation";
}
}
}
}