package org.jolokia.converter.json; import java.lang.reflect.Array; import java.lang.reflect.InvocationTargetException; import java.util.List; import java.util.Stack; import javax.management.AttributeNotFoundException; import org.jolokia.converter.object.StringToObjectConverter; import org.json.simple.JSONArray; /* * Copyright 2009-2013 Roland Huss * * 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. */ /** * Extractor for extracting arrays of any kind. * * @author roland * @since Apr 19, 2009 */ public class ArrayExtractor implements Extractor { /** {@inheritDoc} */ public Class getType() { // Special handler, no specific Type return null; } /** * Extract an array and, if to be jsonified, put it into an {@link JSONArray}. An index can be used (on top of * the extra args stack) in order to specify a single value within the array. * * @param pConverter the global converter in order to be able do dispatch for * serializing inner data types * @param pValue the value to convert (must be an aary) * @param pPathParts extra arguments stack, which is popped to get an index for extracting a single element * of the array * @param jsonify whether to convert to a JSON object/list or whether the plain object * should be returned. The later is required for writing an inner value * @return the extracted object * @throws AttributeNotFoundException * @throws IndexOutOfBoundsException if an index is used which points outside the given list */ public Object extractObject(ObjectToJsonConverter pConverter, Object pValue, Stack<String> pPathParts,boolean jsonify) throws AttributeNotFoundException { int length = pConverter.getCollectionLength(Array.getLength(pValue)); String pathPart = pPathParts.isEmpty() ? null : pPathParts.pop(); if (pathPart != null) { return extractWithPath(pConverter, pValue, pPathParts, jsonify, pathPart); } else { return jsonify ? extractArray(pConverter, pValue, pPathParts, jsonify, length) : pValue; } } /** * Set a value in an array * * @param pConverter the global converter in order to be able do dispatch for * serializing inner data types * @param pInner object on which to set the value (which must be a {@link List}) * @param pIndex index (as string) where to set the value within the array * @param pValue the new value to set * @return the old value at this index * @throws IllegalAccessException * @throws InvocationTargetException */ public Object setObjectValue(StringToObjectConverter pConverter, Object pInner, String pIndex, Object pValue) throws IllegalAccessException, InvocationTargetException { Class clazz = pInner.getClass(); if (!clazz.isArray()) { throw new IllegalArgumentException("Not an array to set a value, but " + clazz + ". (index = " + pIndex + ", value = " + pValue + ")"); } int idx; try { idx = Integer.parseInt(pIndex); } catch (NumberFormatException exp) { throw new IllegalArgumentException("Non-numeric index for accessing array " + pInner + ". (index = " + pIndex + ", value to set = " + pValue + ")",exp); } Class type = clazz.getComponentType(); Object value = pConverter.prepareValue(type.getName(), pValue); Object oldValue = Array.get(pInner, idx); Array.set(pInner, idx, value); return oldValue; } /** {@inheritDoc} */ public boolean canSetValue() { return true; } private List<Object> extractArray(ObjectToJsonConverter pConverter, Object pValue, Stack<String> pPath, boolean jsonify, int pLength) throws AttributeNotFoundException { List<Object> ret = new JSONArray(); for (int i = 0; i < pLength; i++) { Stack<String> path = (Stack<String>) pPath.clone(); try { Object obj = Array.get(pValue, i); ret.add(pConverter.extractObject(obj, path, jsonify)); } catch (ValueFaultHandler.AttributeFilteredException exp) { // Filtered ... } } if (ret.isEmpty() && pLength > 0) { throw new ValueFaultHandler.AttributeFilteredException(); } return ret; } private Object extractWithPath(ObjectToJsonConverter pConverter, Object pValue, Stack<String> pPath, boolean jsonify, String pPathPart) throws AttributeNotFoundException { try { Object obj = Array.get(pValue, Integer.parseInt(pPathPart)); return pConverter.extractObject(obj, pPath, jsonify); } catch (NumberFormatException exp) { ValueFaultHandler faultHandler = pConverter.getValueFaultHandler(); return faultHandler.handleException( new AttributeNotFoundException("Index '" + pPathPart + "' is not numeric for accessing array")); } catch (ArrayIndexOutOfBoundsException exp) { ValueFaultHandler faultHandler = pConverter.getValueFaultHandler(); return faultHandler.handleException( new AttributeNotFoundException("Index '" + pPathPart + "' is out-of-bound for array of size " + Array.getLength(pValue))); } } }