Package org.apache.servicemix.jbi.framework

Source Code of org.apache.servicemix.jbi.framework.ComponentMBeanImpl

/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements.  See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.servicemix.jbi.framework;

import java.io.File;
import java.io.IOException;
import java.util.Properties;

import javax.jbi.JBIException;
import javax.jbi.component.Component;
import javax.jbi.component.ComponentLifeCycle;
import javax.jbi.component.ServiceUnitManager;
import javax.jbi.management.DeploymentException;
import javax.jbi.management.LifeCycleMBean;
import javax.management.JMException;
import javax.management.MBeanAttributeInfo;
import javax.management.MBeanOperationInfo;
import javax.management.ObjectName;

import org.apache.servicemix.jbi.container.ActivationSpec;
import org.apache.servicemix.jbi.container.JBIContainer;
import org.apache.servicemix.jbi.event.ComponentEvent;
import org.apache.servicemix.jbi.event.ComponentListener;
import org.apache.servicemix.jbi.management.AttributeInfoHelper;
import org.apache.servicemix.jbi.management.BaseLifeCycle;
import org.apache.servicemix.jbi.management.ManagementContext;
import org.apache.servicemix.jbi.management.OperationInfoHelper;
import org.apache.servicemix.jbi.messaging.DeliveryChannelImpl;
import org.apache.servicemix.jbi.util.XmlPersistenceSupport;
import org.apache.xbean.classloader.DestroyableClassLoader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
* Defines basic statistics on the Component
*/
public class ComponentMBeanImpl extends BaseLifeCycle implements ComponentMBean {
   
    private static final transient Logger LOGGER = LoggerFactory.getLogger(ComponentMBeanImpl.class);
   
    private boolean exchangeThrottling;
    private long throttlingTimeout = 100;
    private int throttlingInterval = 1;
    private Component component;
    private ComponentLifeCycle lifeCycle;
    private ServiceUnitManager suManager;
    private ComponentContextImpl context;
    private ActivationSpec activationSpec;
    private ObjectName mBeanName;
    private JBIContainer container;
    private ComponentNameSpace componentName;
    private String description = "POJO Component";
    private int queueCapacity = 1024;
    private boolean pojo;
    private boolean binding;
    private boolean service;
    private File stateFile;
    private String[] sharedLibraries;

    /**
     * Construct with it's id and delivery channel Id
     *
     * @param name
     * @param description
     * @param component
     * @param binding
     * @param service
     * @param sharedLibraries
     */
    public ComponentMBeanImpl(JBIContainer container,
                              ComponentNameSpace name,
                              String description,
                              Component component,
                              boolean binding,
                              boolean service,
                              String[] sharedLibraries) {
        this.componentName = name;
        this.container = container;
        this.component = component;
        this.description = description;
        this.binding = binding;
        this.service = service;
        this.sharedLibraries = sharedLibraries;
    }
   
    public void dispose() {
        ClassLoader cl = component.getClass().getClassLoader();
        lifeCycle = null;
        suManager = null;
        component = null;
        if (cl instanceof DestroyableClassLoader) {
            ((DestroyableClassLoader) cl).destroy();
        }
        fireEvent(ComponentEvent.COMPONENT_UNINSTALLED);
    }

    /**
     * Register the MBeans for this Component
     * @param ctx
     * @return ObjectName
     * @throws JBIException
     */
    public ObjectName registerMBeans(ManagementContext ctx) throws JBIException {
        try {
            mBeanName = ctx.createObjectName(this);
            ctx.registerMBean(mBeanName, this, ComponentMBean.class);
            return mBeanName;
        catch (Exception e) {
            String errorStr = "Failed to register MBeans";
            LOGGER.error(errorStr, e);
            throw new JBIException(errorStr, e);
        }
    }
   
    /**
     * Unregister Component MBeans
     * @param ctx
     * @throws JBIException
     */
    public void unregisterMbeans(ManagementContext ctx) throws JBIException {
        ctx.unregisterMBean(mBeanName);
    }

    /**
     * Set the Context
     *
     * @param ctx
     */
    public void setContext(ComponentContextImpl ctx) {
        this.context = ctx;
        this.stateFile = ctx.getEnvironment().getStateFile();
    }

    /**
     * Get the JMX ObjectName for any additional MBean for this component. If there is none, return null.
     *
     * @return ObjectName the JMX object name of the additional MBean or null if there is no additional MBean.
     */
    public ObjectName getExtensionMBeanName() {
        if (isInitialized() || isStarted() || isStopped()) {
            return lifeCycle.getExtensionMBeanName();
        } else {
            return null;
        }
    }
   
    /**
     * Get the name of the item
     * @return the name
     */
    public String getName() {
        return componentName.getName();
    }
   
    /**
     * Get the type of the item
     * @return the type
     */
    public String getType() {
        return "Component";
    }
   
    public String getSubType() {
        return "LifeCycle";
    }
   
   /**
     * Get the Description of the item
     * @return the description
     */
    public String getDescription() {
        return description;
    }

   
    public void init() throws JBIException {
        LOGGER.info("Initializing component: {}", getName());
        if (context != null && component != null) {
            DeliveryChannelImpl channel = new DeliveryChannelImpl(this);
            channel.setContext(context);
            context.setDeliveryChannel(channel);
            super.init();
            fireEvent(ComponentEvent.COMPONENT_INITIALIZED);
            ClassLoader loader = Thread.currentThread().getContextClassLoader();
            try {
                Thread.currentThread().setContextClassLoader(component.getClass().getClassLoader());
                getLifeCycle().init(context);
            } finally {
                Thread.currentThread().setContextClassLoader(loader);
            }
        }
    }
   
    /**
     * Start the item.
     *
     * @exception javax.jbi.JBIException if the item fails to start.
     */
    public void start() throws javax.jbi.JBIException {
        LOGGER.info("Starting component: {}", getName());
        try {
            doStart();
            persistRunningState();
            getContainer().getRegistry().checkPendingAssemblies();
        } catch (JBIException e) {
            LOGGER.error("Could not start component", e);
            throw e;
        } catch (RuntimeException e) {
            LOGGER.error("Could not start component", e);
            throw e;
        } catch (Error e) {
            LOGGER.error("Could not start component", e);
            throw e;
        }
    }

    /**
     * Stop the item. This suspends current messaging activities.
     *
     * @exception javax.jbi.JBIException if the item fails to stop.
     */
    public void stop() throws javax.jbi.JBIException {
        LOGGER.info("Stopping component: {}", getName());
        try {
            doStop();
            persistRunningState();
        } catch (JBIException e) {
            LOGGER.error("Could not stop component", e);
            throw e;
        } catch (RuntimeException e) {
            LOGGER.error("Could not start component", e);
            throw e;
        } catch (Error e) {
            LOGGER.error("Could not start component", e);
            throw e;
        }
    }

    /**
     * Shut down the item. The releases resources, preparatory to uninstallation.
     *
     * @exception javax.jbi.JBIException if the item fails to shut down.
     */
    public void shutDown() throws javax.jbi.JBIException {
        LOGGER.info("Shutting down component: {}", getName());
        try {
            doShutDown();
            persistRunningState();
        } catch (JBIException e) {
            LOGGER.error("Could not shutDown component", e);
            throw e;
        } catch (RuntimeException e) {
            LOGGER.error("Could not start component", e);
            throw e;
        } catch (Error e) {
            LOGGER.error("Could not start component", e);
            throw e;
        }
    }
   
    public void setShutdownStateAfterInstall() {
        setCurrentState(SHUTDOWN);
    }
   
    /**
     * Start the item - doesn't persist the state
     *
     * @exception javax.jbi.JBIException if the item fails to start.
     */
    public void doStart() throws javax.jbi.JBIException {
        if (isShutDown()) {
            // need to re-initialze before starting
            init();
        }
        if (!isStarted()) {
            ClassLoader loader = Thread.currentThread().getContextClassLoader();
            try {
                Thread.currentThread().setContextClassLoader(component.getClass().getClassLoader());
                getLifeCycle().start();
            } finally {
                Thread.currentThread().setContextClassLoader(loader);
            }
            super.start();
            initServiceAssemblies();
            startServiceAssemblies();
        }
        fireEvent(ComponentEvent.COMPONENT_STARTED);
    }

    /**
     * Stop the item - doesn't persist the state
     *
     * @exception javax.jbi.JBIException
     *                if the item fails to stop.
     */
    public void doStop() throws javax.jbi.JBIException {
        if (isUnknown() || isStarted()) {
            stopServiceAssemblies();
            ClassLoader loader = Thread.currentThread().getContextClassLoader();
            try {
                Thread.currentThread().setContextClassLoader(component.getClass().getClassLoader());
                getLifeCycle().stop();
            } finally {
                Thread.currentThread().setContextClassLoader(loader);
            }
            super.stop();
        }
        fireEvent(ComponentEvent.COMPONENT_STOPPED);
    }

    /**
     * Shut down the item - doesn't persist the state
     *
     * @exception javax.jbi.JBIException if the item fails to shut down.
     */
    public void doShutDown() throws javax.jbi.JBIException {
        // Transition from UNKNOWN to SHUTDOWN is done at installation time
        // In this case or if the component is already shut down, do nothing
        if (!isUnknown() && !isShutDown()) {
            doStop();
            shutDownServiceAssemblies();
            ClassLoader loader = Thread.currentThread().getContextClassLoader();
            try {
                Thread.currentThread().setContextClassLoader(component.getClass().getClassLoader());
                getLifeCycle().shutDown();
            } finally {
                Thread.currentThread().setContextClassLoader(loader);
            }
            if (getDeliveryChannel() != null) {
                getDeliveryChannel().close();
                setDeliveryChannel(null);
            }
            lifeCycle = null;
            suManager = null;
        }
        super.shutDown();
        fireEvent(ComponentEvent.COMPONENT_SHUTDOWN);
    }
   
   
    /**
     * Set the initial running state of the Component
     * @throws JBIException
     */
    public void setInitialRunningState() throws JBIException {
        if (!isPojo()) {
            String name = getName();
            String runningState = getRunningStateFromStore();
            LOGGER.info("Setting running state for Component: {} to {}", name, runningState);
            if (runningState != null) {
                if (runningState.equals(LifeCycleMBean.STARTED)) {
                    doStart();
                } else if (runningState.equals(LifeCycleMBean.STOPPED)) {
                    doStart();
                    doStop();
                } else if (runningState.equals(LifeCycleMBean.SHUTDOWN)) {
                    doShutDown();
                }
            }
        }
    }
   
    /**
     * Persist the running state
     */
    public void persistRunningState() {
        if (!isPojo()) {
            String name = getName();
            try {
                String currentState = getCurrentState();
                Properties props = new Properties();
                props.setProperty("state", currentState);
                XmlPersistenceSupport.write(stateFile, props);
            } catch (IOException e) {
                LOGGER.error("Failed to write current running state for Component: {}", name, e);
            }
        }
    }
  
    /**
     * @return the current running state from disk
     */
    public String getRunningStateFromStore() {
        String result = LifeCycleMBean.UNKNOWN;
        String name = getName();
        try {
            Properties props = (Properties) XmlPersistenceSupport.read(stateFile);
            result = props.getProperty("state", result);
        } catch (Exception e) {
            LOGGER.error("Failed to read running state for Component: {}", name, e);
        }
        return result;
    }

    /**
     * @return the capacity of the inbound queue
     */
    public int getInboundQueueCapacity() {
        return queueCapacity;
    }
   
    /**
     * Set the inbound queue capacity
     * @param value
     */
    public void setInboundQueueCapacity(int value) {
        if (getDeliveryChannel() != null) {
            throw new IllegalStateException("The component must be shut down before changing queue capacity");
        }
        this.queueCapacity = value;
    }
   
    /**
     * @return Returns the deliveryChannel.
     */
    public DeliveryChannelImpl getDeliveryChannel() {
        return (DeliveryChannelImpl) context.getDeliveryChannel();
    }

    /**
     * @param deliveryChannel
     *            The deliveryChannel to set.
     */
    public void setDeliveryChannel(DeliveryChannelImpl deliveryChannel) {
        context.setDeliveryChannel(deliveryChannel);
    }

    /**
     * @return the ActivateionSpec
     */
    public ActivationSpec getActivationSpec() {
        return activationSpec;
    }

    /**
     * @return Returns the pojo.
     */
    public boolean isPojo() {
        return pojo;
    }

    /**
     * Set the ActivationSpec
     *
     * @param activationSpec
     */
    public void setActivationSpec(ActivationSpec activationSpec) {
        this.activationSpec = activationSpec;
    }

    /**
     * Is MessageExchange sender throttling enabled ?
     *
     * @return true if throttling enabled
     */
    public boolean isExchangeThrottling() {
        return exchangeThrottling;
    }

    /**
     * Set message throttling
     *
     * @param value
     */
    public void setExchangeThrottling(boolean value) {
        this.exchangeThrottling = value;
    }

    /**
     * Get the throttling timeout
     *
     * @return throttling tomeout (ms)
     */
    public long getThrottlingTimeout() {
        return throttlingTimeout;
    }

    /**
     * Set the throttling timout
     *
     * @param value (ms)
     */
    public void setThrottlingTimeout(long value) {
        throttlingTimeout = value;
    }

    /**
     * Get the interval for throttling - number of Exchanges set before the throttling timeout is applied
     *
     * @return interval for throttling
     */
    public int getThrottlingInterval() {
        return throttlingInterval;
    }

    /**
     * Set the throttling interval number of Exchanges set before the throttling timeout is applied
     *
     * @param value
     */
    public void setThrottlingInterval(int value) {
        throttlingInterval = value;
    }

    /**
     * Get an array of MBeanAttributeInfo
     *
     * @return array of AttributeInfos
     * @throws JMException
     */
    public MBeanAttributeInfo[] getAttributeInfos() throws JMException {
        AttributeInfoHelper helper = new AttributeInfoHelper();
        helper.addAttribute(getObjectToManage(), "componentType", "the type of this component (BC, SE, POJO)");
        helper.addAttribute(getObjectToManage(), "inboundQueueCapacity", "capacity of the inbound queue");
        helper.addAttribute(getObjectToManage(), "exchangeThrottling", "apply throttling");
        helper.addAttribute(getObjectToManage(), "throttlingTimeout", "timeout for throttling");
        helper.addAttribute(getObjectToManage(), "throttlingInterval", "exchange intervals before throttling");
        helper.addAttribute(getObjectToManage(), "extensionMBeanName", "extension mbean name");
        return AttributeInfoHelper.join(super.getAttributeInfos(), helper.getAttributeInfos());
    }

    /**
     * Get an array of MBeanOperationInfo
     *
     * @return array of OperationInfos
     * @throws JMException
     */
    public MBeanOperationInfo[] getOperationInfos() throws JMException {
        OperationInfoHelper helper = new OperationInfoHelper();
        return OperationInfoHelper.join(super.getOperationInfos(), helper.getOperationInfos());
    }

    public void firePropertyChanged(String name, Object oldValue, Object newValue) {
        super.firePropertyChanged(name, oldValue, newValue);
    }

    protected void initServiceAssemblies() throws DeploymentException {
    }

    protected void startServiceAssemblies() throws DeploymentException {
    }

    protected void stopServiceAssemblies() throws DeploymentException {
        Registry registry = getContainer().getRegistry();
        String[] sas = registry.getDeployedServiceAssembliesForComponent(getName());
        for (int i = 0; i < sas.length; i++) {
            ServiceAssemblyLifeCycle sa = registry.getServiceAssembly(sas[i]);
            if (sa.isStarted()) {
                try {
                    sa.stop(false, false);
                    registry.addPendingAssembly(sa);
                } catch (Exception e) {
                    LOGGER.error("Error stopping service assembly {}", sas[i]);
                }
            }
        }
    }

    protected void shutDownServiceAssemblies() throws DeploymentException {
        Registry registry = getContainer().getRegistry();
        String[] sas = registry.getDeployedServiceAssembliesForComponent(getName());
        for (int i = 0; i < sas.length; i++) {
            ServiceAssemblyLifeCycle sa = registry.getServiceAssembly(sas[i]);
            if (sa.isStopped()) {
                try {
                    sa.shutDown(false);
                    registry.addPendingAssembly(sa);
                } catch (Exception e) {
                    LOGGER.error("Error shutting down service assembly {}", sas[i]);
                }
            }
        }
    }
   
    protected void fireEvent(int type) {
        ComponentEvent event = new ComponentEvent(this, type);
        ComponentListener[] listeners = (ComponentListener[]) getContainer().getListeners(ComponentListener.class);
        for (int i = 0; i < listeners.length; i++) {
            switch (type) {
            case ComponentEvent.COMPONENT_INITIALIZED:
                listeners[i].componentInitialized(event);
                break;
            case ComponentEvent.COMPONENT_STARTED:
                listeners[i].componentStarted(event);
                break;
            case ComponentEvent.COMPONENT_STOPPED:
                listeners[i].componentStopped(event);
                break;
            case ComponentEvent.COMPONENT_SHUTDOWN:
                listeners[i].componentShutDown(event);
                break;
            case ComponentEvent.COMPONENT_UNINSTALLED:
                listeners[i].componentUninstalled(event);
                break;
            default:
                break;
            }
        }
       
    }

    public ComponentLifeCycle getLifeCycle() {
        if (lifeCycle == null) {
            lifeCycle = component.getLifeCycle();
        }
        return lifeCycle;
    }

    public ServiceUnitManager getServiceUnitManager() {
        if (suManager == null) {
            suManager = component.getServiceUnitManager();
        }
        return suManager;
    }

    public JBIContainer getContainer() {
        return container;
    }

    public Component getComponent() {
        return component;
    }
   
    public ComponentNameSpace getComponentNameSpace() {
        return componentName;
    }

    public ComponentContextImpl getContext() {
        return context;
    }
    public ObjectName getMBeanName() {
        return mBeanName;
    }
    public boolean isBinding() {
        return binding;
    }
    public boolean isService() {
        return service;
    }

    public void setPojo(boolean pojo) {
        this.pojo = pojo;
    }

    public boolean isEngine() {
        return service;
    }

    /**
     * @return the sharedLibraries
     */
    public String[] getSharedLibraries() {
        return sharedLibraries;
    }

    /**
     * @return the component type
     */
    public String getComponentType() {
        return isBinding() ? "binding-component" : isEngine() ? "service-engine" : "pojo";
    }

}
TOP

Related Classes of org.apache.servicemix.jbi.framework.ComponentMBeanImpl

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.