/*
* Copyright (c) 1999, 2006, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code 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 General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package javax.management;
import java.lang.reflect.Method;
import java.security.AccessController;
import com.sun.jmx.mbeanserver.GetPropertyAction;
import com.sun.jmx.mbeanserver.Introspector;
Describes an MBean attribute exposed for management. Instances of
this class are immutable. Subclasses may be mutable but this is
not recommended.
Since: 1.5
/**
* Describes an MBean attribute exposed for management. Instances of
* this class are immutable. Subclasses may be mutable but this is
* not recommended.
*
* @since 1.5
*/
@SuppressWarnings("serial") // serialVersionUID not constant
public class MBeanAttributeInfo extends MBeanFeatureInfo implements Cloneable {
/* Serial version */
private static final long serialVersionUID;
static {
/* For complicated reasons, the serialVersionUID changed
between JMX 1.0 and JMX 1.1, even though JMX 1.1 did not
have compatibility code for this class. So the
serialization produced by this class with JMX 1.2 and
jmx.serial.form=1.0 is not the same as that produced by
this class with JMX 1.1 and jmx.serial.form=1.0. However,
the serialization without that property is the same, and
that is the only form required by JMX 1.2.
*/
long uid = 8644704819898565848L;
try {
GetPropertyAction act = new GetPropertyAction("jmx.serial.form");
String form = AccessController.doPrivileged(act);
if ("1.0".equals(form))
uid = 7043855487133450673L;
} catch (Exception e) {
// OK: exception means no compat with 1.0, too bad
}
serialVersionUID = uid;
}
static final MBeanAttributeInfo[] NO_ATTRIBUTES =
new MBeanAttributeInfo[0];
@serial The actual attribute type.
/**
* @serial The actual attribute type.
*/
private final String attributeType;
@serial The attribute write right.
/**
* @serial The attribute write right.
*/
private final boolean isWrite;
@serial The attribute read right.
/**
* @serial The attribute read right.
*/
private final boolean isRead;
@serial Indicates if this method is a "is"
/**
* @serial Indicates if this method is a "is"
*/
private final boolean is;
Constructs an MBeanAttributeInfo
object.
Params: - name – The name of the attribute.
- type – The type or class name of the attribute.
- description – A human readable description of the attribute.
- isReadable – True if the attribute has a getter method, false otherwise.
- isWritable – True if the attribute has a setter method, false otherwise.
- isIs – True if this attribute has an "is" getter, false otherwise.
Throws: - IllegalArgumentException – if
isIs
is true but isReadable
is not, or if isIs
is true and type
is not boolean
or java.lang.Boolean
. (New code should always use boolean
rather than java.lang.Boolean
.)
/**
* Constructs an <CODE>MBeanAttributeInfo</CODE> object.
*
* @param name The name of the attribute.
* @param type The type or class name of the attribute.
* @param description A human readable description of the attribute.
* @param isReadable True if the attribute has a getter method, false otherwise.
* @param isWritable True if the attribute has a setter method, false otherwise.
* @param isIs True if this attribute has an "is" getter, false otherwise.
*
* @throws IllegalArgumentException if {@code isIs} is true but
* {@code isReadable} is not, or if {@code isIs} is true and
* {@code type} is not {@code boolean} or {@code java.lang.Boolean}.
* (New code should always use {@code boolean} rather than
* {@code java.lang.Boolean}.)
*/
public MBeanAttributeInfo(String name,
String type,
String description,
boolean isReadable,
boolean isWritable,
boolean isIs) {
this(name, type, description, isReadable, isWritable, isIs,
(Descriptor) null);
}
Constructs an MBeanAttributeInfo
object.
Params: - name – The name of the attribute.
- type – The type or class name of the attribute.
- description – A human readable description of the attribute.
- isReadable – True if the attribute has a getter method, false otherwise.
- isWritable – True if the attribute has a setter method, false otherwise.
- isIs – True if this attribute has an "is" getter, false otherwise.
- descriptor – The descriptor for the attribute. This may be null
which is equivalent to an empty descriptor.
Throws: - IllegalArgumentException – if
isIs
is true but isReadable
is not, or if isIs
is true and type
is not boolean
or java.lang.Boolean
. (New code should always use boolean
rather than java.lang.Boolean
.)
Since: 1.6
/**
* Constructs an <CODE>MBeanAttributeInfo</CODE> object.
*
* @param name The name of the attribute.
* @param type The type or class name of the attribute.
* @param description A human readable description of the attribute.
* @param isReadable True if the attribute has a getter method, false otherwise.
* @param isWritable True if the attribute has a setter method, false otherwise.
* @param isIs True if this attribute has an "is" getter, false otherwise.
* @param descriptor The descriptor for the attribute. This may be null
* which is equivalent to an empty descriptor.
*
* @throws IllegalArgumentException if {@code isIs} is true but
* {@code isReadable} is not, or if {@code isIs} is true and
* {@code type} is not {@code boolean} or {@code java.lang.Boolean}.
* (New code should always use {@code boolean} rather than
* {@code java.lang.Boolean}.)
*
* @since 1.6
*/
public MBeanAttributeInfo(String name,
String type,
String description,
boolean isReadable,
boolean isWritable,
boolean isIs,
Descriptor descriptor) {
super(name, description, descriptor);
this.attributeType = type;
this.isRead = isReadable;
this.isWrite = isWritable;
if (isIs && !isReadable) {
throw new IllegalArgumentException("Cannot have an \"is\" getter " +
"for a non-readable attribute");
}
if (isIs && !type.equals("java.lang.Boolean") &&
!type.equals("boolean")) {
throw new IllegalArgumentException("Cannot have an \"is\" getter " +
"for a non-boolean attribute");
}
this.is = isIs;
}
This constructor takes the name of a simple attribute, and Method objects for reading and writing the attribute. The Descriptor
of the constructed object will include fields contributed by any annotations on the Method
objects that contain the DescriptorKey
meta-annotation.
Params: - name – The programmatic name of the attribute.
- description – A human readable description of the attribute.
- getter – The method used for reading the attribute value.
May be null if the property is write-only.
- setter – The method used for writing the attribute value.
May be null if the attribute is read-only.
Throws: - IntrospectionException – There is a consistency
problem in the definition of this attribute.
/**
* <p>This constructor takes the name of a simple attribute, and Method
* objects for reading and writing the attribute. The {@link Descriptor}
* of the constructed object will include fields contributed by any
* annotations on the {@code Method} objects that contain the
* {@link DescriptorKey} meta-annotation.
*
* @param name The programmatic name of the attribute.
* @param description A human readable description of the attribute.
* @param getter The method used for reading the attribute value.
* May be null if the property is write-only.
* @param setter The method used for writing the attribute value.
* May be null if the attribute is read-only.
* @exception IntrospectionException There is a consistency
* problem in the definition of this attribute.
*/
public MBeanAttributeInfo(String name,
String description,
Method getter,
Method setter) throws IntrospectionException {
this(name,
attributeType(getter, setter),
description,
(getter != null),
(setter != null),
isIs(getter),
ImmutableDescriptor.union(Introspector.descriptorForElement(getter),
Introspector.descriptorForElement(setter)));
}
Returns a shallow clone of this instance.
The clone is obtained by simply calling super.clone(),
thus calling the default native shallow cloning mechanism
implemented by Object.clone().
No deeper cloning of any internal field is made.
Since this class is immutable, cloning is chiefly of
interest to subclasses.
/**
* <p>Returns a shallow clone of this instance.
* The clone is obtained by simply calling <tt>super.clone()</tt>,
* thus calling the default native shallow cloning mechanism
* implemented by <tt>Object.clone()</tt>.
* No deeper cloning of any internal field is made.</p>
*
* <p>Since this class is immutable, cloning is chiefly of
* interest to subclasses.</p>
*/
public Object clone () {
try {
return super.clone() ;
} catch (CloneNotSupportedException e) {
// should not happen as this class is cloneable
return null;
}
}
Returns the class name of the attribute.
Returns: the class name.
/**
* Returns the class name of the attribute.
*
* @return the class name.
*/
public String getType() {
return attributeType;
}
Whether the value of the attribute can be read.
Returns: True if the attribute can be read, false otherwise.
/**
* Whether the value of the attribute can be read.
*
* @return True if the attribute can be read, false otherwise.
*/
public boolean isReadable() {
return isRead;
}
Whether new values can be written to the attribute.
Returns: True if the attribute can be written to, false otherwise.
/**
* Whether new values can be written to the attribute.
*
* @return True if the attribute can be written to, false otherwise.
*/
public boolean isWritable() {
return isWrite;
}
Indicates if this attribute has an "is" getter.
Returns: true if this attribute has an "is" getter.
/**
* Indicates if this attribute has an "is" getter.
*
* @return true if this attribute has an "is" getter.
*/
public boolean isIs() {
return is;
}
public String toString() {
String access;
if (isReadable()) {
if (isWritable())
access = "read/write";
else
access = "read-only";
} else if (isWritable())
access = "write-only";
else
access = "no-access";
return
getClass().getName() + "[" +
"description=" + getDescription() + ", " +
"name=" + getName() + ", " +
"type=" + getType() + ", " +
access + ", " +
(isIs() ? "isIs, " : "") +
"descriptor=" + getDescriptor() +
"]";
}
Compare this MBeanAttributeInfo to another.
Params: - o – the object to compare to.
Returns: true if and only if o
is an MBeanAttributeInfo such that its MBeanFeatureInfo.getName()
, getType()
, MBeanFeatureInfo.getDescription()
, isReadable()
, isWritable()
, and isIs()
values are equal (not necessarily identical) to those of this MBeanAttributeInfo.
/**
* Compare this MBeanAttributeInfo to another.
*
* @param o the object to compare to.
*
* @return true if and only if <code>o</code> is an MBeanAttributeInfo such
* that its {@link #getName()}, {@link #getType()}, {@link
* #getDescription()}, {@link #isReadable()}, {@link
* #isWritable()}, and {@link #isIs()} values are equal (not
* necessarily identical) to those of this MBeanAttributeInfo.
*/
public boolean equals(Object o) {
if (o == this)
return true;
if (!(o instanceof MBeanAttributeInfo))
return false;
MBeanAttributeInfo p = (MBeanAttributeInfo) o;
return (p.getName().equals(getName()) &&
p.getType().equals(getType()) &&
p.getDescription().equals(getDescription()) &&
p.getDescriptor().equals(getDescriptor()) &&
p.isReadable() == isReadable() &&
p.isWritable() == isWritable() &&
p.isIs() == isIs());
}
/* We do not include everything in the hashcode. We assume that
if two operations are different they'll probably have different
names or types. The penalty we pay when this assumption is
wrong should be less than the penalty we would pay if it were
right and we needlessly hashed in the description and parameter
array. */
public int hashCode() {
return getName().hashCode() ^ getType().hashCode();
}
private static boolean isIs(Method getter) {
return (getter != null &&
getter.getName().startsWith("is") &&
(getter.getReturnType().equals(Boolean.TYPE) ||
getter.getReturnType().equals(Boolean.class)));
}
Finds the type of the attribute.
/**
* Finds the type of the attribute.
*/
private static String attributeType(Method getter, Method setter)
throws IntrospectionException {
Class type = null;
if (getter != null) {
if (getter.getParameterTypes().length != 0) {
throw new IntrospectionException("bad getter arg count");
}
type = getter.getReturnType();
if (type == Void.TYPE) {
throw new IntrospectionException("getter " + getter.getName() +
" returns void");
}
}
if (setter != null) {
Class params[] = setter.getParameterTypes();
if (params.length != 1) {
throw new IntrospectionException("bad setter arg count");
}
if (type == null)
type = params[0];
else if (type != params[0]) {
throw new IntrospectionException("type mismatch between " +
"getter and setter");
}
}
if (type == null) {
throw new IntrospectionException("getter and setter cannot " +
"both be null");
}
return type.getName();
}
}