/*
* JBoss, Home of Professional Open Source.
* Copyright 2006, Red Hat Middleware LLC, and individual contributors
* as indicated by the @author tags. See the copyright.txt file in the
* distribution for a full listing of individual contributors.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software 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 software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package javax.management;
import java.lang.reflect.Method;
import java.io.Serializable;
import java.util.Arrays;
import org.jboss.mx.util.MetaDataUtil;
/**
* Describes an operation exposed by an MBean
*
* This implementation protects its immutability by taking shallow clones of all arrays
* supplied in constructors and by returning shallow array clones in getXXX() methods.
*
* @author Juha Lindfors.
* @author Trevor Squires.
* @author Adrian Brock
* @author Thomas Diesler.
*
* @version $Revision: 57200 $
*/
public class MBeanOperationInfo extends MBeanFeatureInfo
implements Serializable, Cloneable
{
// Constants -----------------------------------------------------
private static final long serialVersionUID = -6178860474881375330L;
/**
* Management operation impact: INFO. The operation should not alter the
* state of the MBean component (read operation).
*/
public static final int INFO = 0;
/**
* Management operation impact: ACTION. The operation changes the state
* of the MBean component (write operation).
*/
public static final int ACTION = 1;
/**
* Management operation impact: ACTION_INFO. Operation behaves like a
* read/write operation.
*/
public static final int ACTION_INFO = 2;
/**
* Management operation impact: UNKNOWN. Reserved for Standard MBeans.
*/
public static final int UNKNOWN = 3;
// Attributes ----------------------------------------------------
/**
* Impact of this operation.
*/
private int impact = UNKNOWN;
/**
* Signature of this operation.
*/
private MBeanParameterInfo[] signature = null;
/**
* Return type of this operation as a fully qualified class name.
*/
private String type = null;
/**
* The cached string
*/
private transient String cacheString;
/**
* The cached hashCode
*/
private transient int cacheHashCode;
// Constructors --------------------------------------------------
/**
* Constructs management operation metadata from a given Method
* object and description.
*
* @param description human readable description of this operation
* @param method used for build the metadata for the management operation
*/
public MBeanOperationInfo(String description, Method method) throws IllegalArgumentException
{
super(method.getName(), description);
this.type = method.getReturnType().getName();
Class[] sign = method.getParameterTypes();
signature = new MBeanParameterInfo[sign.length];
for (int i = 0; i < sign.length; ++i)
{
String name = sign[i].getName();
signature[i] = new MBeanParameterInfo(name, name, "MBean Operation Parameter.");
}
}
/**
* Constructs a management operation metadata.
*
* @param name name of the management operation
* @param description human readable description string of the operation
* @param inSignature inSignature of the operation
* @param returnType return type of the operation as a fully qualified class name
* @param impact impact of this operation: {@link #ACTION ACTION}, {@link #INFO INFO}, {@link #ACTION_INFO ACTION_INFO} or {@link #UNKNOWN UNKNOWN}
* @exception IllegalArgumentException if the name or return type is not a valid java identifier or it is a reserved word
* or the impact is invalid
*/
public MBeanOperationInfo(String name, String description,
MBeanParameterInfo[] inSignature,
String returnType, int impact) throws IllegalArgumentException
{
super(name, description);
if (MetaDataUtil.isValidJavaType(returnType) == false)
throw new IllegalArgumentException("Return type is not a valid java identifier (or is reserved): " + returnType);
if (impact != INFO && impact != ACTION && impact != ACTION_INFO && impact != UNKNOWN)
throw new IllegalArgumentException("Impact is invalid: " + impact);
this.signature = (null == inSignature) ? new MBeanParameterInfo[0] : (MBeanParameterInfo[]) inSignature.clone();
this.type = returnType;
this.impact = impact;
}
// Public --------------------------------------------------------
/**
* Returns a fully qualified class name of the return type of this operation.
*
* @return fully qualified class name
*/
public String getReturnType()
{
return type;
}
/**
* Returns the signature of this operation. Note: an operation with a void
* signature returns a zero-length array, not a null reference.
*
* @return operation's signature
*/
public MBeanParameterInfo[] getSignature()
{
return (MBeanParameterInfo[]) signature.clone();
}
/**
* Returns the impact of this operation. The impact is one of the following values:
* {@link #ACTION ACTION}, {@link #INFO INFO}, {@link #ACTION_INFO ACTION_INFO} or {@link #UNKNOWN UNKNOWN} (reserved for Standard MBeans).
*
* @return operation's impact
*/
public int getImpact()
{
return impact;
}
public boolean equals(Object object)
{
if (this == object)
return true;
if (object == null || (object instanceof MBeanOperationInfo) == false)
return false;
MBeanOperationInfo other = (MBeanOperationInfo) object;
if (super.equals(other) == false)
return false;
if (this.getReturnType().equals(other.getReturnType()) == false)
return false;
if (this.getImpact() != other.getImpact())
return false;
MBeanParameterInfo[] thisParams = this.getSignature();
MBeanParameterInfo[] otherParams = other.getSignature();
if (thisParams.length != otherParams.length)
return false;
for (int i = 0; i < thisParams.length; ++i)
if (thisParams[i].equals(otherParams[i]) == false)
return false;
return true;
}
public int hashCode()
{
if (cacheHashCode == 0)
{
cacheHashCode = super.hashCode();
cacheHashCode += getReturnType().hashCode();
cacheHashCode += impact;
}
return cacheHashCode;
}
/**
* @return a human readable string
*/
public String toString()
{
if (cacheString == null)
{
StringBuffer buffer = new StringBuffer(100);
buffer.append(getClass().getName()).append(":");
buffer.append(" name=").append(getName());
buffer.append(" description=").append(getDescription());
buffer.append(" signature=").append(Arrays.asList(signature));
buffer.append(" returnType=").append(getReturnType());
buffer.append(" impact=");
switch (impact)
{
case ACTION: buffer.append("ACTION"); break;
case ACTION_INFO: buffer.append("ACTION_INFO"); break;
case INFO: buffer.append("INFO"); break;
default: buffer.append("UNKNOWN");
}
cacheString = buffer.toString();
}
return cacheString;
}
// Cloneable implementation --------------------------------------
/**
* Creates a copy of this object. This is a deep copy; the MBeanParameterInfo objects
* forming the operation's signature are also cloned.
*
* @return a clone of this object
*/
public synchronized Object clone()
{
MBeanOperationInfo clone = null;
try
{
clone = (MBeanOperationInfo) super.clone();
clone.signature = (MBeanParameterInfo[])this.signature.clone();
clone.type = this.type;
clone.impact = this.impact;
}
catch(CloneNotSupportedException e)
{
}
return clone;
}
}