Package io.fabric8.jolokia.client

Source Code of io.fabric8.jolokia.client.OpenTypeGenerator

/*
* #%L
* Fabric8 :: SPI
* %%
* Copyright (C) 2014 Red Hat
* %%
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
*      http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
* #L%
*/
package io.fabric8.jolokia.client;

import java.beans.ConstructorProperties;
import java.lang.reflect.Array;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;

import javax.management.openmbean.ArrayType;
import javax.management.openmbean.CompositeData;
import javax.management.openmbean.CompositeDataSupport;
import javax.management.openmbean.CompositeType;
import javax.management.openmbean.OpenDataException;
import javax.management.openmbean.OpenType;
import javax.management.openmbean.SimpleType;
import javax.management.openmbean.TabularData;
import javax.management.openmbean.TabularDataSupport;
import javax.management.openmbean.TabularType;

import org.jboss.gravia.utils.IllegalStateAssertion;

/**
* A utility that can marshal an MXBean compliant java type into an opendata type and vise versa.
*
* @author thomas.diesler@jboss.com
* @since 23-Jun-2014
*/
public final class OpenTypeGenerator {

    // Hide ctor
    private OpenTypeGenerator() {
    }

    public static Object toOpenData(OpenType<?> otype, Object value) throws OpenDataException {
        Object result;
        if (value != null && !otype.isValue(value)) {
            if (otype instanceof CompositeType) {
                result = toCompositeData((CompositeType) otype, value);
            } else if (otype instanceof TabularType) {
                result = toTabularData((TabularType) otype, (Map<?, ?>) value);
            } else if (otype instanceof ArrayType) {
                result = toArrayData((ArrayType<?>) otype, value);
            } else if (otype == SimpleType.BYTE && value instanceof Number) {
                result = Byte.parseByte(value.toString());
            } else {
                throw new OpenDataException("Unsupported open type: " + otype);
            }
        } else {
            result = value;
        }
        boolean isAssignable = result == null || otype.isValue(result);
        IllegalStateAssertion.assertTrue(isAssignable, "Value " + result + " is not a value of: " + otype);
        return result;
    }

    private static TabularData toTabularData(TabularType ttype, Map<?, ?> value) throws OpenDataException {
        TabularDataSupport tdata = new TabularDataSupport(ttype);
        CompositeType rowType = ttype.getRowType();
        OpenType<?> keyType = rowType.getType("key");
        OpenType<?> valueType = rowType.getType("value");
        for (Map.Entry<?, ?> entry : value.entrySet()) {
            Map<String, Object> rowData = new HashMap<>();
            Object keyData = toOpenData(keyType, entry.getKey());
            Object valData = toOpenData(valueType, entry.getValue());
            rowData.put("key", keyData);
            rowData.put("value", valData);
            tdata.put(toCompositeData(rowType, rowData));
        }
        return tdata;
    }

    static Object toArrayData(ArrayType<?> atype, Object value) throws OpenDataException {
        List<?> items;
        if (value instanceof Collection) {
            items = new ArrayList<Object>((Collection<?>) value);
        } else {
            items = Arrays.asList(value);
        }
        Object array = getOpenTypeArray(atype, ClassLoader.getSystemClassLoader(), items.size());
        OpenType<?> elementType = atype.getElementOpenType();
        for (int i = 0; i < items.size(); i++) {
            Object val = toOpenData(elementType, items.get(i));
            Array.set(array, i, val);
        }
        return array;
    }

    private static Object getterValue(Object bean, OpenType<?> itemType , String itemName) throws OpenDataException {
        try {
            Method method = null;
            Class<? extends Object> beanClass = bean.getClass();
            String prefix = itemType == SimpleType.BOOLEAN ? "is" : "get";
            String methodName = prefix + itemName.substring(0, 1).toUpperCase() + itemName.substring(1);
            for (Method aux : beanClass.getMethods()) {
                if (methodName.equals(aux.getName()) && aux.getParameterTypes().length == 0) {
                    method = aux;
                    break;
                }
            }
            IllegalStateAssertion.assertNotNull(method, "Cannot find getter: " + beanClass.getName() + "." + methodName);
            return method.invoke(bean, (Object[]) null);
        } catch (Exception ex) {
            OpenDataException odex = new OpenDataException("Cannot invoke getter for: " + itemName);
            odex.initCause(ex);
            throw odex;
        }
    }

    public static Object fromOpenData(OpenType<?> otype, ClassLoader classLoader, Object value) throws OpenDataException {
        if (value == null) {
            return null;
        }
        Object result;
        if (otype instanceof CompositeType) {
            result = fromCompositeData(classLoader, (CompositeData) value);
        } else if (otype instanceof TabularType) {
            result = fromTabularData(classLoader, (TabularData) value);
        } else if (otype instanceof ArrayType) {
            result = fromArrayData((ArrayType<?>) otype, classLoader, value);
        } else if (otype instanceof SimpleType) {
            result = value;
        } else {
            throw new OpenDataException("Unsupported open type: " + otype);
        }
        return result;
    }
   
    private static Object fromCompositeData(ClassLoader classLoader, CompositeData cdata) throws OpenDataException {
        if (cdata == null)
            return null;
       
        Object result;
        CompositeType ctype = cdata.getCompositeType();
        String typeName = ctype.getTypeName();
        if (typeName.startsWith("java.util.Map")) {
            Object openKey = cdata.get("key");
            Object openVal = cdata.get("value");
            OpenType<?> keyType = ctype.getType("key");
            OpenType<?> valType = ctype.getType("value");
            Object key = fromOpenData(keyType, classLoader, openKey);
            Object value = fromOpenData(valType, classLoader, openVal);
            result = Collections.singletonMap(key, value);
        } else {
            Class<?> targetType;
            try {
                targetType = classLoader.loadClass(typeName);
            } catch (ClassNotFoundException ex) {
                OpenDataException odex = new OpenDataException("Cannot load target type: " + typeName);
                odex.initCause(ex);
                throw odex;
            }
            Constructor<?> ctor = null;
            boolean isDefaultCtor = false;
            for (Constructor<?> aux : targetType.getConstructors()) {
                isDefaultCtor = aux.getParameterTypes().length == 0;
                if (isDefaultCtor) {
                    ctor = aux;
                    break;
                } else if (aux.getAnnotation(ConstructorProperties.class) != null) {
                    ctor = aux;
                }
            }
            IllegalStateAssertion.assertNotNull(ctor, "Cannot mxbean compliant constructor for: " + targetType.getName());
            try {
                if (isDefaultCtor) {
                    result = ctor.newInstance((Object[]) null);
                    for (String key : ctype.keySet()) {
                        OpenType<?> itemType = ctype.getType(key);
                        Object itemValue = cdata.get(key);
                        Object javaValue = fromOpenData(itemType, classLoader, itemValue);
                        invokeSetter(result, key, javaValue);
                    }
                } else {
                    List<Object> params = new ArrayList<>();
                    ConstructorProperties props = ctor.getAnnotation(ConstructorProperties.class);
                    for (String key : props.value()) {
                        OpenType<?> itemType = ctype.getType(key);
                        Object itemValue = cdata.get(key);
                        Object javaValue = fromOpenData(itemType, classLoader, itemValue);
                        params.add(javaValue);
                    }
                    Class<?>[] paramTypes = ctor.getParameterTypes();
                    for (Object param : params) {
                        int index = params.indexOf(param);
                        Class<?> paramType = paramTypes[index];
                        param = toTargetType(paramType, param);
                        if (param != params.get(index)) {
                            params.set(index, param);
                        }
                    }
                    result = ctor.newInstance(params.toArray());
                }
            } catch (Exception ex) {
                OpenDataException odex = new OpenDataException("Cannot construct object from: " + cdata);
                odex.initCause(ex);
                throw odex;
            }
        }
        return result;
    }

    @SuppressWarnings("unchecked")
    private static Map<?, ?> fromTabularData(ClassLoader classLoader, TabularData tdata) throws OpenDataException {
        Map<Object, Object> result = new LinkedHashMap<>();
        if (tdata != null) {
            for (CompositeData cdata : (Collection<CompositeData>)tdata.values()) {
                Map<Object, Object> rowValue = (Map<Object, Object>) fromCompositeData(classLoader, cdata);
                result.putAll(rowValue);
            }
        }
        return result;
    }

    private static Object fromArrayData(ArrayType<?> atype, ClassLoader classLoader, Object value) throws OpenDataException {
        OpenType<?> elementType = atype.getElementOpenType();
        Object array;
        if (value == null) {
            array = getJavaTypeArray(atype, classLoader, 0);
        } else if (value.getClass().isArray()) {
            int length = Array.getLength(value);
            array = getJavaTypeArray(atype, classLoader, length);
            for (int i = 0; i < length; i++) {
                Object val = Array.get(value, i);
                val = fromOpenData(elementType, classLoader, val);
                Array.set(array, i, val);
            }
        } else if (value instanceof Collection<?>) {
            List<?> list = new ArrayList<Object>((Collection<?>) value);
            array = getJavaTypeArray(atype, classLoader, list.size());
            for (int i = 0; i < list.size(); i++) {
                Object val = list.get(i);
                val = fromOpenData(elementType, classLoader, val);
                Array.set(array, i, val);
            }
        } else {
            throw new IllegalArgumentException("Unsupported value type: " + value);
        }
        return array;
    }

    private static void invokeSetter(Object target, String itemName, Object value) throws OpenDataException {
        try {
            Method method = null;
            Class<? extends Object> beanClass = target.getClass();
            String methodName = "set" + itemName.substring(0, 1).toUpperCase() + itemName.substring(1);
            for (Method aux : beanClass.getMethods()) {
                if (methodName.equals(aux.getName()) && aux.getParameterTypes().length == 1) {
                    method = aux;
                    break;
                }
            }
            IllegalStateAssertion.assertNotNull(method, "Cannot find setter: " + beanClass.getName() + "." + methodName);
            Class<?> paramType = method.getParameterTypes()[0];
            value = toTargetType(paramType, value);
            method.invoke(target, value);
        } catch (Exception ex) {
            OpenDataException odex = new OpenDataException("Cannot invoke setter for: " + itemName);
            odex.initCause(ex);
            throw odex;
        }
    }
   
    static CompositeData toCompositeData(CompositeType ctype, Object value) throws OpenDataException {
        Map<String, Object> items = new HashMap<>();
        for(String key : ctype.keySet()) {
            OpenType<?> itemType = ctype.getType(key);
            Object rawValue;
            if (value instanceof Map) {
                rawValue = ((Map<?, ?>) value).get(key);
            } else {
                rawValue = getterValue(value, itemType, key);
            }
            Object openValue = toOpenData(itemType, rawValue);
            items.put(key, openValue);
        }
        return new CompositeDataSupport(ctype, items);
    }

    static Object getOpenTypeArray(ArrayType<?> atype, ClassLoader classLoader, int dimension) throws OpenDataException {
        Class<?> compType;
        OpenType<?> elementType = atype.getElementOpenType();
        try {
            if (atype.isPrimitiveArray()) {
                compType = Class.forName(atype.getTypeName()).getComponentType();
            } else if (elementType instanceof CompositeType) {
                compType = CompositeData.class;
            } else if (elementType instanceof TabularType) {
                compType = TabularData.class;
            } else {
                compType = classLoader.loadClass(elementType.getTypeName());
            }
        } catch (ClassNotFoundException ex) {
            OpenDataException odex = new OpenDataException("Cannot load array type: " + atype);
            odex.initCause(ex);
            throw odex;
        }
        return Array.newInstance(compType, dimension);
    }

    static Object getJavaTypeArray(ArrayType<?> atype, ClassLoader classLoader, int dimension) throws OpenDataException {
        Class<?> compType;
        OpenType<?> elementType = atype.getElementOpenType();
        try {
            if (atype.isPrimitiveArray()) {
                compType = Class.forName(atype.getTypeName()).getComponentType();
            } else {
                compType = classLoader.loadClass(elementType.getTypeName());
            }
        } catch (ClassNotFoundException ex) {
            OpenDataException odex = new OpenDataException("Cannot load array type: " + atype);
            odex.initCause(ex);
            throw odex;
        }
        return Array.newInstance(compType, dimension);
    }

    static Object toTargetType(Class<?> targetType, Object value) {
        if (value == null || targetType.isPrimitive() || targetType.isAssignableFrom(value.getClass())) {
            return value;
        }
        Object result = value;
        if (value.getClass().isArray()) {
            if (targetType == List.class) {
                result = new ArrayList<Object>(Arrays.asList((Object[]) value));
            } else if (targetType == Set.class) {
                result = new HashSet<Object>(Arrays.asList((Object[]) value));
            }
        }
        boolean isAssignable = targetType.isAssignableFrom(result.getClass());
        IllegalStateAssertion.assertTrue(isAssignable, "Value '" + value + "' not assignable to: " + targetType.getName());
        return result;
    }
}
TOP

Related Classes of io.fabric8.jolokia.client.OpenTypeGenerator

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.