jdk/src/share/classes/com/sun/jmx/mbeanserver/Introspector.java
author jfdenise
Tue, 09 Dec 2008 15:57:09 +0100
changeset 1700 4506662fb2ee
parent 1699 3611e5fd6da5
child 4156 acaa49a2768a
permissions -rw-r--r--
6675526: Define an Annotation to name registered MBeans Reviewed-by: emcmanus
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     1
/*
715
f16baef3a20e 6719955: Update copyright year
xdono
parents: 687
diff changeset
     2
 * Copyright 1999-2008 Sun Microsystems, Inc.  All Rights Reserved.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
90ce3da70b43 Initial load
duke
parents:
diff changeset
     4
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
90ce3da70b43 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
90ce3da70b43 Initial load
duke
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Sun designates this
90ce3da70b43 Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
90ce3da70b43 Initial load
duke
parents:
diff changeset
     9
 * by Sun in the LICENSE file that accompanied this code.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    10
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
90ce3da70b43 Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
90ce3da70b43 Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
90ce3da70b43 Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    16
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
90ce3da70b43 Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    20
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    21
 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    22
 * CA 95054 USA or visit www.sun.com if you need additional information or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    23
 * have any questions.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    24
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    25
90ce3da70b43 Initial load
duke
parents:
diff changeset
    26
package com.sun.jmx.mbeanserver;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    27
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    28
import com.sun.jmx.remote.util.EnvHelp;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    29
import java.beans.BeanInfo;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    30
import java.beans.PropertyDescriptor;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    31
import java.lang.annotation.Annotation;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    32
import java.lang.reflect.AnnotatedElement;
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    33
import java.lang.reflect.Array;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    34
import java.lang.reflect.Constructor;
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    35
import java.lang.reflect.InvocationTargetException;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    36
import java.lang.reflect.Method;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    37
import java.lang.reflect.Modifier;
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    38
import java.lang.reflect.Proxy;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    39
import java.lang.reflect.UndeclaredThrowableException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    40
import java.util.Arrays;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    41
import java.util.HashMap;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    42
import java.util.Map;
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    43
import java.util.logging.Level;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    44
import javax.management.AttributeNotFoundException;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    45
import javax.management.Description;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    46
90ce3da70b43 Initial load
duke
parents:
diff changeset
    47
import javax.management.Descriptor;
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    48
import javax.management.DescriptorFields;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    49
import javax.management.DescriptorKey;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    50
import javax.management.DynamicMBean;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    51
import javax.management.ImmutableDescriptor;
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    52
import javax.management.MBean;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    53
import javax.management.MBeanInfo;
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    54
import javax.management.MXBean;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    55
import javax.management.NotCompliantMBeanException;
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    56
import javax.management.openmbean.CompositeData;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    57
import javax.management.openmbean.MXBeanMappingFactory;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    58
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    59
import static com.sun.jmx.defaults.JmxProperties.MBEANSERVER_LOGGER;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
    60
import com.sun.jmx.mbeanserver.Util;
34
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
    61
import com.sun.jmx.remote.util.EnvHelp;
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
    62
import java.beans.BeanInfo;
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
    63
import java.beans.PropertyDescriptor;
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
    64
import java.lang.reflect.Array;
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
    65
import java.lang.reflect.InvocationTargetException;
1699
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
    66
import java.util.LinkedHashSet;
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
    67
import java.util.Set;
1700
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    68
import java.util.regex.Matcher;
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    69
import java.util.regex.Pattern;
34
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
    70
import javax.management.AttributeNotFoundException;
1699
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
    71
import javax.management.JMX;
1700
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    72
import javax.management.ObjectName;
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    73
import javax.management.ObjectNameTemplate;
34
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
    74
import javax.management.openmbean.CompositeData;
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
    75
import javax.management.openmbean.MXBeanMappingFactory;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    76
90ce3da70b43 Initial load
duke
parents:
diff changeset
    77
/**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    78
 * This class contains the methods for performing all the tests needed to verify
90ce3da70b43 Initial load
duke
parents:
diff changeset
    79
 * that a class represents a JMX compliant MBean.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    80
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    81
 * @since 1.5
90ce3da70b43 Initial load
duke
parents:
diff changeset
    82
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    83
public class Introspector {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    84
1700
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    85
    /**
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    86
     * Pattern used to extract Attribute Names from ObjectNameTemplate Annotation
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    87
     * For example, in the following example, the Name attribute value is
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    88
     * retrieved : ":type=MyType, name={Name}"
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    89
     */
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    90
    private static Pattern OBJECT_NAME_PATTERN_TEMPLATE =
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
    91
            Pattern.compile("(\\{[^\\}]+\\})|(=\"\\{[^\\}]+\\}\")");
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    92
     /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
    93
     * ------------------------------------------
90ce3da70b43 Initial load
duke
parents:
diff changeset
    94
     *  PRIVATE CONSTRUCTORS
90ce3da70b43 Initial load
duke
parents:
diff changeset
    95
     * ------------------------------------------
90ce3da70b43 Initial load
duke
parents:
diff changeset
    96
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    97
90ce3da70b43 Initial load
duke
parents:
diff changeset
    98
    // private constructor defined to "hide" the default public constructor
90ce3da70b43 Initial load
duke
parents:
diff changeset
    99
    private Introspector() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   100
90ce3da70b43 Initial load
duke
parents:
diff changeset
   101
        // ------------------------------
90ce3da70b43 Initial load
duke
parents:
diff changeset
   102
        // ------------------------------
90ce3da70b43 Initial load
duke
parents:
diff changeset
   103
90ce3da70b43 Initial load
duke
parents:
diff changeset
   104
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   105
90ce3da70b43 Initial load
duke
parents:
diff changeset
   106
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   107
     * ------------------------------------------
90ce3da70b43 Initial load
duke
parents:
diff changeset
   108
     *  PUBLIC METHODS
90ce3da70b43 Initial load
duke
parents:
diff changeset
   109
     * ------------------------------------------
90ce3da70b43 Initial load
duke
parents:
diff changeset
   110
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   111
90ce3da70b43 Initial load
duke
parents:
diff changeset
   112
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   113
     * Tell whether a MBean of the given class is a Dynamic MBean.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   114
     * This method does nothing more than returning
90ce3da70b43 Initial load
duke
parents:
diff changeset
   115
     * <pre>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   116
     * javax.management.DynamicMBean.class.isAssignableFrom(c)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   117
     * </pre>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   118
     * This method does not check for any JMX MBean compliance:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   119
     * <ul><li>If <code>true</code> is returned, then instances of
90ce3da70b43 Initial load
duke
parents:
diff changeset
   120
     *     <code>c</code> are DynamicMBean.</li>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   121
     *     <li>If <code>false</code> is returned, then no further
90ce3da70b43 Initial load
duke
parents:
diff changeset
   122
     *     assumption can be made on instances of <code>c</code>.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   123
     *     In particular, instances of <code>c</code> may, or may not
90ce3da70b43 Initial load
duke
parents:
diff changeset
   124
     *     be JMX standard MBeans.</li>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   125
     * </ul>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   126
     * @param c The class of the MBean under examination.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   127
     * @return <code>true</code> if instances of <code>c</code> are
90ce3da70b43 Initial load
duke
parents:
diff changeset
   128
     *         Dynamic MBeans, <code>false</code> otherwise.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   129
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   130
     **/
1510
e747d3193ef2 6763639: Remove "rawtypes" warnings from JMX code
emcmanus
parents: 1322
diff changeset
   131
    public static final boolean isDynamic(final Class<?> c) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   132
        // Check if the MBean implements the DynamicMBean interface
90ce3da70b43 Initial load
duke
parents:
diff changeset
   133
        return javax.management.DynamicMBean.class.isAssignableFrom(c);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   134
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   135
90ce3da70b43 Initial load
duke
parents:
diff changeset
   136
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   137
     * Basic method for testing that a MBean of a given class can be
90ce3da70b43 Initial load
duke
parents:
diff changeset
   138
     * instantiated by the MBean server.<p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   139
     * This method checks that:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   140
     * <ul><li>The given class is a concrete class.</li>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   141
     *     <li>The given class exposes at least one public constructor.</li>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   142
     * </ul>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   143
     * If these conditions are not met, throws a NotCompliantMBeanException.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   144
     * @param c The class of the MBean we want to create.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   145
     * @exception NotCompliantMBeanException if the MBean class makes it
90ce3da70b43 Initial load
duke
parents:
diff changeset
   146
     *            impossible to instantiate the MBean from within the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   147
     *            MBeanServer.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   148
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   149
     **/
1510
e747d3193ef2 6763639: Remove "rawtypes" warnings from JMX code
emcmanus
parents: 1322
diff changeset
   150
    public static void testCreation(Class<?> c)
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   151
        throws NotCompliantMBeanException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   152
        // Check if the class is a concrete class
90ce3da70b43 Initial load
duke
parents:
diff changeset
   153
        final int mods = c.getModifiers();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   154
        if (Modifier.isAbstract(mods) || Modifier.isInterface(mods)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   155
            throw new NotCompliantMBeanException("MBean class must be concrete");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   156
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   157
90ce3da70b43 Initial load
duke
parents:
diff changeset
   158
        // Check if the MBean has a public constructor
1510
e747d3193ef2 6763639: Remove "rawtypes" warnings from JMX code
emcmanus
parents: 1322
diff changeset
   159
        final Constructor<?>[] consList = c.getConstructors();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   160
        if (consList.length == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   161
            throw new NotCompliantMBeanException("MBean class must have public constructor");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   162
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   163
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   164
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   165
    public static void checkCompliance(Class<?> mbeanClass)
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   166
    throws NotCompliantMBeanException {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   167
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   168
        // Check that @Resource is used correctly (if it used).
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   169
        MBeanInjector.validate(mbeanClass);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   170
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   171
        // Is DynamicMBean?
90ce3da70b43 Initial load
duke
parents:
diff changeset
   172
        //
90ce3da70b43 Initial load
duke
parents:
diff changeset
   173
        if (DynamicMBean.class.isAssignableFrom(mbeanClass))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   174
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   175
        // Is Standard MBean?
90ce3da70b43 Initial load
duke
parents:
diff changeset
   176
        //
90ce3da70b43 Initial load
duke
parents:
diff changeset
   177
        final Exception mbeanException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   178
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   179
            getStandardMBeanInterface(mbeanClass);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   180
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   181
        } catch (NotCompliantMBeanException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   182
            mbeanException = e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   183
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   184
        // Is MXBean?
90ce3da70b43 Initial load
duke
parents:
diff changeset
   185
        //
90ce3da70b43 Initial load
duke
parents:
diff changeset
   186
        final Exception mxbeanException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   187
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   188
            getMXBeanInterface(mbeanClass);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   189
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   190
        } catch (NotCompliantMBeanException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   191
            mxbeanException = e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   192
        }
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   193
        // Is @MBean or @MXBean class?
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   194
        // In fact we find @MBean or @MXBean as a hacky variant of
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   195
        // getStandardMBeanInterface or getMXBeanInterface.  If we get here
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   196
        // then nothing worked.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   197
        final String msg =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   198
            "MBean class " + mbeanClass.getName() + " does not implement " +
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   199
            "DynamicMBean; does not follow the Standard MBean conventions (" +
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   200
            mbeanException.toString() + "); does not follow the MXBean conventions (" +
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   201
            mxbeanException.toString() + "); and does not have or inherit the @" +
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   202
            MBean.class.getSimpleName() + " or @" + MXBean.class.getSimpleName() +
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   203
            " annotation";
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   204
        throw new NotCompliantMBeanException(msg);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   205
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   206
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   207
    /**
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   208
     * <p>Make a DynamicMBean out of the existing MBean object.  The object
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   209
     * may already be a DynamicMBean, or it may be a Standard MBean or
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   210
     * MXBean, possibly defined using {@code @MBean} or {@code @MXBean}.</p>
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   211
     * @param mbean the object to convert to a DynamicMBean.
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   212
     * @param <T> a type parameter defined for implementation convenience
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   213
     * (which would have to be removed if this method were part of the public
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   214
     * API).
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   215
     * @return the converted DynamicMBean.
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   216
     * @throws NotCompliantMBeanException if {@code mbean} is not a compliant
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   217
     * MBean object, including the case where it is null.
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   218
     */
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   219
    public static <T> DynamicMBean makeDynamicMBean(T mbean)
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   220
    throws NotCompliantMBeanException {
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   221
        if (mbean == null)
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   222
            throw new NotCompliantMBeanException("Null MBean object");
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   223
        if (mbean instanceof DynamicMBean)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   224
            return (DynamicMBean) mbean;
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   225
        final Class<?> mbeanClass = mbean.getClass();
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   226
        Class<? super T> c = null;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   227
        try {
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   228
            c = Util.cast(getStandardMBeanInterface(mbeanClass));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   229
        } catch (NotCompliantMBeanException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   230
            // Ignore exception - we need to check whether
90ce3da70b43 Initial load
duke
parents:
diff changeset
   231
            // mbean is an MXBean first.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   232
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   233
        if (c != null)
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   234
            return new StandardMBeanSupport(mbean, c);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   235
90ce3da70b43 Initial load
duke
parents:
diff changeset
   236
        try {
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   237
            c = Util.cast(getMXBeanInterface(mbeanClass));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   238
        } catch (NotCompliantMBeanException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   239
            // Ignore exception - we cannot decide whether mbean was supposed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   240
            // to be an MBean or an MXBean. We will call checkCompliance()
90ce3da70b43 Initial load
duke
parents:
diff changeset
   241
            // to generate the appropriate exception.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   242
        }
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   243
        if (c != null) {
1322
d038148778cc 6750935: The expected NotCompliantMBeanException is not thrown for the custom MXBeanMappingFactory
emcmanus
parents: 834
diff changeset
   244
            MXBeanMappingFactory factory;
d038148778cc 6750935: The expected NotCompliantMBeanException is not thrown for the custom MXBeanMappingFactory
emcmanus
parents: 834
diff changeset
   245
            try {
d038148778cc 6750935: The expected NotCompliantMBeanException is not thrown for the custom MXBeanMappingFactory
emcmanus
parents: 834
diff changeset
   246
                factory = MXBeanMappingFactory.forInterface(c);
d038148778cc 6750935: The expected NotCompliantMBeanException is not thrown for the custom MXBeanMappingFactory
emcmanus
parents: 834
diff changeset
   247
            } catch (IllegalArgumentException e) {
d038148778cc 6750935: The expected NotCompliantMBeanException is not thrown for the custom MXBeanMappingFactory
emcmanus
parents: 834
diff changeset
   248
                NotCompliantMBeanException ncmbe =
d038148778cc 6750935: The expected NotCompliantMBeanException is not thrown for the custom MXBeanMappingFactory
emcmanus
parents: 834
diff changeset
   249
                        new NotCompliantMBeanException(e.getMessage());
d038148778cc 6750935: The expected NotCompliantMBeanException is not thrown for the custom MXBeanMappingFactory
emcmanus
parents: 834
diff changeset
   250
                ncmbe.initCause(e);
d038148778cc 6750935: The expected NotCompliantMBeanException is not thrown for the custom MXBeanMappingFactory
emcmanus
parents: 834
diff changeset
   251
                throw ncmbe;
d038148778cc 6750935: The expected NotCompliantMBeanException is not thrown for the custom MXBeanMappingFactory
emcmanus
parents: 834
diff changeset
   252
            }
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   253
            return new MXBeanSupport(mbean, c, factory);
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   254
        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   255
        checkCompliance(mbeanClass);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   256
        throw new NotCompliantMBeanException("Not compliant"); // not reached
90ce3da70b43 Initial load
duke
parents:
diff changeset
   257
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   258
90ce3da70b43 Initial load
duke
parents:
diff changeset
   259
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   260
     * Basic method for testing if a given class is a JMX compliant MBean.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   261
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   262
     * @param baseClass The class to be tested
90ce3da70b43 Initial load
duke
parents:
diff changeset
   263
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   264
     * @return <code>null</code> if the MBean is a DynamicMBean,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   265
     *         the computed {@link javax.management.MBeanInfo} otherwise.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   266
     * @exception NotCompliantMBeanException The specified class is not a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   267
     *            JMX compliant MBean
90ce3da70b43 Initial load
duke
parents:
diff changeset
   268
     */
1510
e747d3193ef2 6763639: Remove "rawtypes" warnings from JMX code
emcmanus
parents: 1322
diff changeset
   269
    public static MBeanInfo testCompliance(Class<?> baseClass)
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   270
        throws NotCompliantMBeanException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   271
90ce3da70b43 Initial load
duke
parents:
diff changeset
   272
        // ------------------------------
90ce3da70b43 Initial load
duke
parents:
diff changeset
   273
        // ------------------------------
90ce3da70b43 Initial load
duke
parents:
diff changeset
   274
90ce3da70b43 Initial load
duke
parents:
diff changeset
   275
        // Check if the MBean implements the MBean or the Dynamic
90ce3da70b43 Initial load
duke
parents:
diff changeset
   276
        // MBean interface
90ce3da70b43 Initial load
duke
parents:
diff changeset
   277
        if (isDynamic(baseClass))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   278
            return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   279
90ce3da70b43 Initial load
duke
parents:
diff changeset
   280
        return testCompliance(baseClass, null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   281
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   282
1510
e747d3193ef2 6763639: Remove "rawtypes" warnings from JMX code
emcmanus
parents: 1322
diff changeset
   283
    public static void testComplianceMXBeanInterface(Class<?> interfaceClass,
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   284
                                                     MXBeanMappingFactory factory)
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   285
            throws NotCompliantMBeanException {
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   286
        MXBeanIntrospector.getInstance(factory).getAnalyzer(interfaceClass);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   287
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   288
90ce3da70b43 Initial load
duke
parents:
diff changeset
   289
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   290
     * Basic method for testing if a given class is a JMX compliant
90ce3da70b43 Initial load
duke
parents:
diff changeset
   291
     * Standard MBean.  This method is only called by the legacy code
90ce3da70b43 Initial load
duke
parents:
diff changeset
   292
     * in com.sun.management.jmx.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   293
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   294
     * @param baseClass The class to be tested.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   295
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   296
     * @param mbeanInterface the MBean interface that the class implements,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   297
     * or null if the interface must be determined by introspection.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   298
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   299
     * @return the computed {@link javax.management.MBeanInfo}.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   300
     * @exception NotCompliantMBeanException The specified class is not a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   301
     *            JMX compliant Standard MBean
90ce3da70b43 Initial load
duke
parents:
diff changeset
   302
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   303
    public static synchronized MBeanInfo
90ce3da70b43 Initial load
duke
parents:
diff changeset
   304
            testCompliance(final Class<?> baseClass,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   305
                           Class<?> mbeanInterface)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   306
            throws NotCompliantMBeanException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   307
        if (mbeanInterface == null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   308
            mbeanInterface = getStandardMBeanInterface(baseClass);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   309
        MBeanIntrospector<?> introspector = StandardMBeanIntrospector.getInstance();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   310
        return getClassMBeanInfo(introspector, baseClass, mbeanInterface);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   311
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   312
90ce3da70b43 Initial load
duke
parents:
diff changeset
   313
    private static <M> MBeanInfo
90ce3da70b43 Initial load
duke
parents:
diff changeset
   314
            getClassMBeanInfo(MBeanIntrospector<M> introspector,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   315
                              Class<?> baseClass, Class<?> mbeanInterface)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   316
    throws NotCompliantMBeanException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   317
        PerInterface<M> perInterface = introspector.getPerInterface(mbeanInterface);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   318
        return introspector.getClassMBeanInfo(baseClass, perInterface);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   319
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   320
90ce3da70b43 Initial load
duke
parents:
diff changeset
   321
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   322
     * Get the MBean interface implemented by a JMX Standard
90ce3da70b43 Initial load
duke
parents:
diff changeset
   323
     * MBean class. This method is only called by the legacy
90ce3da70b43 Initial load
duke
parents:
diff changeset
   324
     * code in "com.sun.management.jmx".
90ce3da70b43 Initial load
duke
parents:
diff changeset
   325
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   326
     * @param baseClass The class to be tested.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   327
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   328
     * @return The MBean interface implemented by the MBean.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   329
     *         Return <code>null</code> if the MBean is a DynamicMBean,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   330
     *         or if no MBean interface is found.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   331
     */
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   332
    public static Class<?> getMBeanInterface(Class<?> baseClass) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   333
        // Check if the given class implements the MBean interface
90ce3da70b43 Initial load
duke
parents:
diff changeset
   334
        // or the Dynamic MBean interface
90ce3da70b43 Initial load
duke
parents:
diff changeset
   335
        if (isDynamic(baseClass)) return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   336
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   337
            return getStandardMBeanInterface(baseClass);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   338
        } catch (NotCompliantMBeanException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   339
            return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   340
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   341
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   342
90ce3da70b43 Initial load
duke
parents:
diff changeset
   343
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   344
     * Get the MBean interface implemented by a JMX Standard MBean class.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   345
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   346
     * @param baseClass The class to be tested.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   347
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   348
     * @return The MBean interface implemented by the Standard MBean.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   349
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   350
     * @throws NotCompliantMBeanException The specified class is
90ce3da70b43 Initial load
duke
parents:
diff changeset
   351
     * not a JMX compliant Standard MBean.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   352
     */
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   353
    public static <T> Class<? super T> getStandardMBeanInterface(Class<T> baseClass)
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   354
    throws NotCompliantMBeanException {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   355
        if (baseClass.isAnnotationPresent(MBean.class))
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   356
            return baseClass;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   357
        Class<? super T> current = baseClass;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   358
        Class<? super T> mbeanInterface = null;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   359
        while (current != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   360
            mbeanInterface =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   361
                findMBeanInterface(current, current.getName());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   362
            if (mbeanInterface != null) break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   363
            current = current.getSuperclass();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   364
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   365
        if (mbeanInterface != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   366
            return mbeanInterface;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   367
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   368
            final String msg =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   369
                "Class " + baseClass.getName() +
90ce3da70b43 Initial load
duke
parents:
diff changeset
   370
                " is not a JMX compliant Standard MBean";
90ce3da70b43 Initial load
duke
parents:
diff changeset
   371
            throw new NotCompliantMBeanException(msg);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   372
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   373
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   374
90ce3da70b43 Initial load
duke
parents:
diff changeset
   375
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   376
     * Get the MXBean interface implemented by a JMX MXBean class.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   377
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   378
     * @param baseClass The class to be tested.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   379
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   380
     * @return The MXBean interface implemented by the MXBean.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   381
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   382
     * @throws NotCompliantMBeanException The specified class is
90ce3da70b43 Initial load
duke
parents:
diff changeset
   383
     * not a JMX compliant MXBean.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   384
     */
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   385
    public static <T> Class<? super T> getMXBeanInterface(Class<T> baseClass)
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   386
        throws NotCompliantMBeanException {
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   387
        if (hasMXBeanAnnotation(baseClass))
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   388
            return baseClass;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   389
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   390
            return MXBeanSupport.findMXBeanInterface(baseClass);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   391
        } catch (Exception e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   392
            throw throwException(baseClass,e);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   393
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   394
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   395
687
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   396
    public static <T> Class<? super T> getStandardOrMXBeanInterface(
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   397
            Class<T> baseClass, boolean mxbean)
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   398
    throws NotCompliantMBeanException {
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   399
        if (mxbean)
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   400
            return getMXBeanInterface(baseClass);
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   401
        else
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   402
            return getStandardMBeanInterface(baseClass);
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   403
    }
874e25a9844a 6562936: Support custom type mappings in MXBeans
emcmanus
parents: 34
diff changeset
   404
1700
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   405
    public static ObjectName templateToObjectName(Descriptor descriptor,
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   406
            DynamicMBean mbean)
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   407
            throws NotCompliantMBeanException {
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   408
        String template = (String)
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   409
            descriptor.getFieldValue(JMX.OBJECT_NAME_TEMPLATE);
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   410
        if(template == null) return null;
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   411
        try {
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   412
            Matcher m = OBJECT_NAME_PATTERN_TEMPLATE.matcher(template);
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   413
            while (m.find()){
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   414
                String grp = m.group();
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   415
                System.out.println("GROUP " + grp);
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   416
                String attributeName = null;
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   417
                boolean quote = false;
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   418
                if(grp.startsWith("=\"{")) {
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   419
                    attributeName = grp.substring(3, grp.length() - 2);
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   420
                    quote = true;
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   421
                } else
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   422
                    attributeName = grp.substring(1, grp.length() - 1);
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   423
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   424
                Object attributeValue = mbean.getAttribute(attributeName);
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   425
                String validValue = quote ?
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   426
                    "=" + ObjectName.quote(attributeValue.toString()) :
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   427
                    attributeValue.toString();
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   428
                template = template.replace(grp, validValue);
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   429
            }
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   430
            return new ObjectName(template);
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   431
        }catch(Exception ex) {
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   432
            NotCompliantMBeanException ncex = new
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   433
                    NotCompliantMBeanException(ObjectNameTemplate.class.
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   434
                    getSimpleName() + " annotation value [" + template + "] " +
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   435
                    "is invalid. " + ex);
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   436
            ncex.initCause(ex);
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   437
            throw ncex;
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   438
        }
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   439
    }
4506662fb2ee 6675526: Define an Annotation to name registered MBeans
jfdenise
parents: 1699
diff changeset
   440
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   441
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   442
     * ------------------------------------------
90ce3da70b43 Initial load
duke
parents:
diff changeset
   443
     *  PRIVATE METHODS
90ce3da70b43 Initial load
duke
parents:
diff changeset
   444
     * ------------------------------------------
90ce3da70b43 Initial load
duke
parents:
diff changeset
   445
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   446
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   447
    static boolean hasMXBeanAnnotation(Class<?> c) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   448
        MXBean m = c.getAnnotation(MXBean.class);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   449
        return (m != null && m.value());
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   450
    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   451
90ce3da70b43 Initial load
duke
parents:
diff changeset
   452
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   453
     * Try to find the MBean interface corresponding to the class aName
90ce3da70b43 Initial load
duke
parents:
diff changeset
   454
     * - i.e. <i>aName</i>MBean, from within aClass and its superclasses.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   455
     **/
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   456
    private static <T> Class<? super T> findMBeanInterface(
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   457
            Class<T> aClass, String aName) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   458
        Class<? super T> current = aClass;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   459
        while (current != null) {
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   460
            final Class<?>[] interfaces = current.getInterfaces();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   461
            final int len = interfaces.length;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   462
            for (int i=0;i<len;i++)  {
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   463
                Class<? super T> inter = Util.cast(interfaces[i]);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   464
                inter = implementsMBean(inter, aName);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   465
                if (inter != null) return inter;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   466
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   467
            current = current.getSuperclass();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   468
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   469
        return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   470
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   471
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   472
    public static String descriptionForElement(AnnotatedElement elmt) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   473
        if (elmt == null)
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   474
            return null;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   475
        Description d = elmt.getAnnotation(Description.class);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   476
        if (d == null)
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   477
            return null;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   478
        return d.value();
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   479
    }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   480
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   481
    public static String descriptionForParameter(
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   482
            Annotation[] parameterAnnotations) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   483
        for (Annotation a : parameterAnnotations) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   484
            if (a instanceof Description)
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   485
                return ((Description) a).value();
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   486
        }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   487
        return null;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   488
    }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   489
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   490
    public static String nameForParameter(
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   491
            Annotation[] parameterAnnotations) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   492
        for (Annotation a : parameterAnnotations) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   493
            Class<? extends Annotation> ac = a.annotationType();
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   494
            // You'd really have to go out of your way to have more than
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   495
            // one @Name annotation, so we don't check for that.
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   496
            if (ac.getSimpleName().equals("Name")) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   497
                try {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   498
                    Method value = ac.getMethod("value");
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   499
                    if (value.getReturnType() == String.class &&
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   500
                            value.getParameterTypes().length == 0) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   501
                        return (String) value.invoke(a);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   502
                    }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   503
                } catch (Exception e) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   504
                    MBEANSERVER_LOGGER.log(
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   505
                            Level.WARNING,
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   506
                            "Unexpected exception getting @" + ac.getName(),
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   507
                            e);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   508
                }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   509
            }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   510
        }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   511
        return null;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   512
    }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   513
1699
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   514
    public static Descriptor descriptorForElement(final AnnotatedElement elmt,
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   515
            boolean isSetter) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   516
        if (elmt == null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   517
            return ImmutableDescriptor.EMPTY_DESCRIPTOR;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   518
        final Annotation[] annots = elmt.getAnnotations();
1699
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   519
        Descriptor descr = descriptorForAnnotations(annots);
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   520
        String[] exceptions = {};
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   521
        if(elmt instanceof Method)
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   522
            exceptions = getAllExceptions(((Method) elmt).getExceptionTypes());
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   523
        else
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   524
            if(elmt instanceof Constructor<?>)
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   525
                exceptions = getAllExceptions(((Constructor<?>) elmt).
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   526
                        getExceptionTypes());
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   527
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   528
        if(exceptions.length > 0 ) {
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   529
            String fieldName = isSetter ? JMX.SET_EXCEPTIONS_FIELD :
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   530
                JMX.EXCEPTIONS_FIELD;
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   531
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   532
            String[] fieldNames = {fieldName};
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   533
            Object[] fieldValues = {exceptions};
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   534
            descr = ImmutableDescriptor.union(descr,
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   535
                    new ImmutableDescriptor(fieldNames, fieldValues));
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   536
        }
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   537
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   538
        return descr;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   539
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   540
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   541
    public static Descriptor descriptorForAnnotation(Annotation annot) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   542
        return descriptorForAnnotations(new Annotation[] {annot});
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   543
    }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   544
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   545
    public static Descriptor descriptorForAnnotations(Annotation[] annots) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   546
        if (annots.length == 0)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   547
            return ImmutableDescriptor.EMPTY_DESCRIPTOR;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   548
        Map<String, Object> descriptorMap = new HashMap<String, Object>();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   549
        for (Annotation a : annots) {
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   550
            if (a instanceof DescriptorFields)
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   551
                addDescriptorFieldsToMap(descriptorMap, (DescriptorFields) a);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   552
            addAnnotationFieldsToMap(descriptorMap, a);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   553
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   554
90ce3da70b43 Initial load
duke
parents:
diff changeset
   555
        if (descriptorMap.isEmpty())
90ce3da70b43 Initial load
duke
parents:
diff changeset
   556
            return ImmutableDescriptor.EMPTY_DESCRIPTOR;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   557
        else
90ce3da70b43 Initial load
duke
parents:
diff changeset
   558
            return new ImmutableDescriptor(descriptorMap);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   559
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   560
1699
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   561
    /**
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   562
     * Array of thrown excepions.
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   563
     * @param exceptions can be null;
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   564
     * @return An Array of Exception class names. Size is 0 if method is null.
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   565
     */
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   566
    private static String[] getAllExceptions(Class<?>[] exceptions) {
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   567
        Set<String> set = new LinkedHashSet<String>();
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   568
        for(Class<?>ex : exceptions)
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   569
            set.add(ex.getName());
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   570
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   571
        String[] arr = new String[set.size()];
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   572
        return set.toArray(arr);
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   573
    }
3611e5fd6da5 6250014: MBeanOperationInfo Descriptor field for exceptions
jfdenise
parents: 1510
diff changeset
   574
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   575
    private static void addDescriptorFieldsToMap(
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   576
            Map<String, Object> descriptorMap, DescriptorFields df) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   577
        for (String field : df.value()) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   578
            int eq = field.indexOf('=');
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   579
            if (eq < 0) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   580
                throw new IllegalArgumentException(
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   581
                        "@DescriptorFields string must contain '=': " +
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   582
                        field);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   583
            }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   584
            String name = field.substring(0, eq);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   585
            String value = field.substring(eq + 1);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   586
            addToMap(descriptorMap, name, value);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   587
        }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   588
    }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   589
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   590
    private static void addAnnotationFieldsToMap(
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   591
            Map<String, Object> descriptorMap, Annotation a) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   592
        Class<? extends Annotation> c = a.annotationType();
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   593
        Method[] elements = c.getMethods();
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   594
        for (Method element : elements) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   595
            DescriptorKey key = element.getAnnotation(DescriptorKey.class);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   596
            if (key != null) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   597
                String name = key.value();
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   598
                Object value;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   599
                try {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   600
                    value = element.invoke(a);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   601
                } catch (RuntimeException e) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   602
                    // we don't expect this - except for possibly
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   603
                    // security exceptions?
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   604
                    // RuntimeExceptions shouldn't be "UndeclaredThrowable".
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   605
                    // anyway...
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   606
                    throw e;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   607
                } catch (Exception e) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   608
                    // we don't expect this
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   609
                    throw new UndeclaredThrowableException(e);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   610
                }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   611
                if (!key.omitIfDefault() ||
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   612
                        !equals(value, element.getDefaultValue())) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   613
                    value = annotationToField(value);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   614
                    addToMap(descriptorMap, name, value);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   615
                }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   616
            }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   617
        }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   618
    }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   619
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   620
    private static void addToMap(
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   621
            Map<String, Object> descriptorMap, String name, Object value) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   622
        Object oldValue = descriptorMap.put(name, value);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   623
        if (oldValue != null && !equals(oldValue, value)) {
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   624
            final String msg =
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   625
                "Inconsistent values for descriptor field " + name +
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   626
                " from annotations: " + value + " :: " + oldValue;
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   627
            throw new IllegalArgumentException(msg);
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   628
        }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   629
    }
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   630
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   631
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   632
     * Throws a NotCompliantMBeanException or a SecurityException.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   633
     * @param notCompliant the class which was under examination
90ce3da70b43 Initial load
duke
parents:
diff changeset
   634
     * @param cause the raeson why NotCompliantMBeanException should
90ce3da70b43 Initial load
duke
parents:
diff changeset
   635
     *        be thrown.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   636
     * @return nothing - this method always throw an exception.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   637
     *         The return type makes it possible to write
90ce3da70b43 Initial load
duke
parents:
diff changeset
   638
     *         <pre> throw throwException(clazz,cause); </pre>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   639
     * @throws SecurityException - if cause is a SecurityException
90ce3da70b43 Initial load
duke
parents:
diff changeset
   640
     * @throws NotCompliantMBeanException otherwise.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   641
     **/
90ce3da70b43 Initial load
duke
parents:
diff changeset
   642
    static NotCompliantMBeanException throwException(Class<?> notCompliant,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   643
            Throwable cause)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   644
            throws NotCompliantMBeanException, SecurityException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   645
        if (cause instanceof SecurityException)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   646
            throw (SecurityException) cause;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   647
        if (cause instanceof NotCompliantMBeanException)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   648
            throw (NotCompliantMBeanException)cause;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   649
        final String classname =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   650
                (notCompliant==null)?"null class":notCompliant.getName();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   651
        final String reason =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   652
                (cause==null)?"Not compliant":cause.getMessage();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   653
        final NotCompliantMBeanException res =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   654
                new NotCompliantMBeanException(classname+": "+reason);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   655
        res.initCause(cause);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   656
        throw res;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   657
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   658
90ce3da70b43 Initial load
duke
parents:
diff changeset
   659
    // Convert a value from an annotation element to a descriptor field value
90ce3da70b43 Initial load
duke
parents:
diff changeset
   660
    // E.g. with @interface Foo {class value()} an annotation @Foo(String.class)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   661
    // will produce a Descriptor field value "java.lang.String"
90ce3da70b43 Initial load
duke
parents:
diff changeset
   662
    private static Object annotationToField(Object x) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   663
        // An annotation element cannot have a null value but never mind
90ce3da70b43 Initial load
duke
parents:
diff changeset
   664
        if (x == null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   665
            return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   666
        if (x instanceof Number || x instanceof String ||
90ce3da70b43 Initial load
duke
parents:
diff changeset
   667
                x instanceof Character || x instanceof Boolean ||
90ce3da70b43 Initial load
duke
parents:
diff changeset
   668
                x instanceof String[])
90ce3da70b43 Initial load
duke
parents:
diff changeset
   669
            return x;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   670
        // Remaining possibilities: array of primitive (e.g. int[]),
90ce3da70b43 Initial load
duke
parents:
diff changeset
   671
        // enum, class, array of enum or class.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   672
        Class<?> c = x.getClass();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   673
        if (c.isArray()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   674
            if (c.getComponentType().isPrimitive())
90ce3da70b43 Initial load
duke
parents:
diff changeset
   675
                return x;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   676
            Object[] xx = (Object[]) x;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   677
            String[] ss = new String[xx.length];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   678
            for (int i = 0; i < xx.length; i++)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   679
                ss[i] = (String) annotationToField(xx[i]);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   680
            return ss;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   681
        }
1510
e747d3193ef2 6763639: Remove "rawtypes" warnings from JMX code
emcmanus
parents: 1322
diff changeset
   682
        if (x instanceof Class<?>)
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   683
            return ((Class<?>) x).getName();
1510
e747d3193ef2 6763639: Remove "rawtypes" warnings from JMX code
emcmanus
parents: 1322
diff changeset
   684
        if (x instanceof Enum<?>)
e747d3193ef2 6763639: Remove "rawtypes" warnings from JMX code
emcmanus
parents: 1322
diff changeset
   685
            return ((Enum<?>) x).name();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   686
        // The only other possibility is that the value is another
90ce3da70b43 Initial load
duke
parents:
diff changeset
   687
        // annotation, or that the language has evolved since this code
90ce3da70b43 Initial load
duke
parents:
diff changeset
   688
        // was written.  We don't allow for either of those currently.
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   689
        // If it is indeed another annotation, then x will be a proxy
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   690
        // with an unhelpful name like $Proxy2.  So we extract the
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   691
        // proxy's interface to use that in the exception message.
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   692
        if (Proxy.isProxyClass(c))
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   693
            c = c.getInterfaces()[0];  // array "can't be empty"
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   694
        throw new IllegalArgumentException("Illegal type for annotation " +
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   695
                "element using @DescriptorKey: " + c.getName());
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   696
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   697
90ce3da70b43 Initial load
duke
parents:
diff changeset
   698
    // This must be consistent with the check for duplicate field values in
90ce3da70b43 Initial load
duke
parents:
diff changeset
   699
    // ImmutableDescriptor.union.  But we don't expect to be called very
90ce3da70b43 Initial load
duke
parents:
diff changeset
   700
    // often so this inefficient check should be enough.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   701
    private static boolean equals(Object x, Object y) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   702
        return Arrays.deepEquals(new Object[] {x}, new Object[] {y});
90ce3da70b43 Initial load
duke
parents:
diff changeset
   703
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   704
90ce3da70b43 Initial load
duke
parents:
diff changeset
   705
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   706
     * Returns the XXMBean interface or null if no such interface exists
90ce3da70b43 Initial load
duke
parents:
diff changeset
   707
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   708
     * @param c The interface to be tested
90ce3da70b43 Initial load
duke
parents:
diff changeset
   709
     * @param clName The name of the class implementing this interface
90ce3da70b43 Initial load
duke
parents:
diff changeset
   710
     */
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   711
    private static <T> Class<? super T> implementsMBean(Class<T> c, String clName) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   712
        String clMBeanName = clName + "MBean";
90ce3da70b43 Initial load
duke
parents:
diff changeset
   713
        if (c.getName().equals(clMBeanName)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   714
            return c;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   715
        }
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   716
        Class<?>[] interfaces = c.getInterfaces();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   717
        for (int i = 0;i < interfaces.length; i++) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   718
            if (interfaces[i].getName().equals(clMBeanName))
833
bfa2bef7517c 6323980: Annotations to simplify MBean development
emcmanus
parents: 687
diff changeset
   719
                return Util.cast(interfaces[i]);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   720
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   721
90ce3da70b43 Initial load
duke
parents:
diff changeset
   722
        return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   723
    }
34
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   724
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   725
    public static Object elementFromComplex(Object complex, String element)
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   726
    throws AttributeNotFoundException {
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   727
        try {
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   728
            if (complex.getClass().isArray() && element.equals("length")) {
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   729
                return Array.getLength(complex);
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   730
            } else if (complex instanceof CompositeData) {
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   731
                return ((CompositeData) complex).get(element);
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   732
            } else {
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   733
                // Java Beans introspection
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   734
                //
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   735
                BeanInfo bi = java.beans.Introspector.getBeanInfo(complex.getClass());
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   736
                PropertyDescriptor[] pds = bi.getPropertyDescriptors();
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   737
                for (PropertyDescriptor pd : pds)
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   738
                    if (pd.getName().equals(element))
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   739
                        return pd.getReadMethod().invoke(complex);
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   740
                throw new AttributeNotFoundException(
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   741
                    "Could not find the getter method for the property " +
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   742
                    element + " using the Java Beans introspector");
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   743
            }
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   744
        } catch (InvocationTargetException e) {
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   745
            throw new IllegalArgumentException(e);
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   746
        } catch (AttributeNotFoundException e) {
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   747
            throw e;
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   748
        } catch (Exception e) {
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   749
            throw EnvHelp.initCause(
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   750
                new AttributeNotFoundException(e.getMessage()), e);
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   751
        }
2d042367885f 6602310: Extensions to Query API for JMX 2.0
emcmanus
parents: 2
diff changeset
   752
    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   753
}