/* * Copyright (C) 2010 The Android Open Source Project * * 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. */ package org.apache.tapestry5.json; import java.util.ArrayList; import java.util.Collections; import java.util.Iterator; import java.util.List; // Note: this class was written without inspecting the non-free org.json sourcecode. /** * A dense indexed sequence of values. Values may be any mix of * {@link JSONObject JSONObjects}, other {@link JSONArray JSONArrays}, Strings, * Booleans, Integers, Longs, Doubles, {@code null} or {@link JSONObject#NULL}. * Values may not be {@link Double#isNaN() NaNs}, {@link Double#isInfinite() * infinities}, or of any type not listed here. * * {@code JSONArray} has the same type coercion behavior and * optional/mandatory accessors as {@link JSONObject}. See that class' * documentation for details. * * <strong>Warning:</strong> this class represents null in two incompatible * ways: the standard Java {@code null} reference, and the sentinel value {@link * JSONObject#NULL}. In particular, {@code get} fails if the requested index * holds the null reference, but succeeds if it holds {@code JSONObject.NULL}. * * Instances of this class are not thread safe. */ public final class JSONArray extends JSONCollection implements Iterable<Object> { private final List<Object> values; /** * Creates a {@code JSONArray} with no values. */ public JSONArray() { values = new ArrayList<Object>(); } /** * Creates a new {@code JSONArray} with values from the next array in the * tokener. * * @param readFrom a tokener whose nextValue() method will yield a * {@code JSONArray}. * @throws RuntimeException if the parse fails or doesn't yield a * {@code JSONArray}. */ JSONArray(JSONTokener readFrom) { /* * Getting the parser to populate this could get tricky. Instead, just * parse to temporary JSONArray and then steal the data from that. */ Object object = readFrom.nextValue(JSONArray.class); if (object instanceof JSONArray) { values = ((JSONArray) object).values; } else { throw JSON.typeMismatch(object, "JSONArray"); } } /** * Creates a new {@code JSONArray} with values from the JSON string. * * @param json a JSON-encoded string containing an array. * @throws RuntimeException if the parse fails or doesn't yield a {@code * JSONArray}. */ public JSONArray(String json) { this(new JSONTokener(json)); } /** * Creates a new {@code JSONArray} with values from the given primitive array. * * @param values The values to use. * @throws RuntimeException if any of the values are non-finite double values (i.e. NaN or infinite) */ public JSONArray(Object... values) { this(); for (int i = 0; i < values.length; ++i) { put(values[i]); } } /** * Create a new array, and adds all values fro the iterable to the array (using {@link #putAll(Iterable)}. * * This is implemented as a static method so as not to break the semantics of the existing {@link #JSONArray(Object...)} constructor. * Adding a constructor of type Iterable would change the meaning of <code>new JSONArray(new JSONArray())</code>. * * @param iterable * collection ot value to include, or null * @since 5.4 */ public static JSONArray from(Iterable<?> iterable) { return new JSONArray().putAll(iterable); } /** * @return Returns the number of values in this array. */ public int length() { return values.size(); } /** * Appends {@code value} to the end of this array. * * @param value a {@link JSONObject}, {@link JSONArray}, String, Boolean, * Integer, Long, Double, or {@link JSONObject#NULL}}. May * not be {@link Double#isNaN() NaNs} or {@link Double#isInfinite() * infinities}. Unsupported values are not permitted and will cause the * array to be in an inconsistent state. * @return this array. */ public JSONArray put(Object value) { JSONObject.testValidity(value); values.add(value); return this; } /** * Same as {@link #put}, with added validity checks. * * @param value The value to append. */ void checkedPut(Object value) { JSONObject.testValidity(value); if (value instanceof Number) { JSON.checkDouble(((Number) value).doubleValue()); } put(value); } /** * Sets the value at {@code index} to {@code value}, null padding this array * to the required length if necessary. If a value already exists at {@code * index}, it will be replaced. * * @param index Where to put the value. * @param value a {@link JSONObject}, {@link JSONArray}, String, Boolean, * Integer, Long, Double, {@link JSONObject#NULL}, or {@code null}. May * not be {@link Double#isNaN() NaNs} or {@link Double#isInfinite() * infinities}. * @return this array. * @throws RuntimeException If the value cannot be represented as a finite double value. */ public JSONArray put(int index, Object value) { if (index < 0) { throw new RuntimeException("JSONArray[" + index + "] not found."); } JSONObject.testValidity(value); if (value instanceof Number) { // deviate from the original by checking all Numbers, not just floats & doubles JSON.checkDouble(((Number) value).doubleValue()); } while (values.size() <= index) { values.add(null); } values.set(index, value); return this; } /** * Returns true if this array has no value at {@code index}, or if its value * is the {@code null} reference or {@link JSONObject#NULL}. * * @param index Which value to check. * @return true if the value is null. */ public boolean isNull(int index) { Object value = values.get(index); return value == null || value == JSONObject.NULL; } /** * Returns the value at {@code index}. * * @param index Which value to get. * @return the value at the specified location. * @throws RuntimeException if this array has no value at {@code index}, or if * that value is the {@code null} reference. This method returns * normally if the value is {@code JSONObject#NULL}. */ public Object get(int index) { try { Object value = values.get(index); if (value == null) { throw new RuntimeException("Value at " + index + " is null."); } return value; } catch (IndexOutOfBoundsException e) { throw new RuntimeException("Index " + index + " out of range [0.." + values.size() + ")"); } } /** * Removes and returns the value at {@code index}, or null if the array has no value * at {@code index}. * * @param index Which value to remove. * @return The value previously at the specified location. */ public Object remove(int index) { if (index < 0 || index >= values.size()) { return null; } return values.remove(index); } /** * Returns the value at {@code index} if it exists and is a boolean or can * be coerced to a boolean. * * @param index Which value to get. * @return the value at the specified location. * @throws RuntimeException if the value at {@code index} doesn't exist or * cannot be coerced to a boolean. */ public boolean getBoolean(int index) { Object object = get(index); Boolean result = JSON.toBoolean(object); if (result == null) { throw JSON.typeMismatch(true, index, object, "Boolean"); } return result; } /** * Returns the value at {@code index} if it exists and is a double or can * be coerced to a double. * * @param index Which value to get. * @return the value at the specified location. * @throws RuntimeException if the value at {@code index} doesn't exist or * cannot be coerced to a double. */ public double getDouble(int index) { Object object = get(index); Double result = JSON.toDouble(object); if (result == null) { throw JSON.typeMismatch(true, index, object, "number"); } return result; } /** * Returns the value at {@code index} if it exists and is an int or * can be coerced to an int. * * @param index Which value to get. * @return the value at the specified location. * @throws RuntimeException if the value at {@code index} doesn't exist or * cannot be coerced to a int. */ public int getInt(int index) { Object object = get(index); Integer result = JSON.toInteger(object); if (result == null) { throw JSON.typeMismatch(true, index, object, "int"); } return result; } /** * Returns the value at {@code index} if it exists and is a long or * can be coerced to a long. * * @param index Which value to get. * @return the value at the specified location. * @throws RuntimeException if the value at {@code index} doesn't exist or * cannot be coerced to a long. */ public long getLong(int index) { Object object = get(index); Long result = JSON.toLong(object); if (result == null) { throw JSON.typeMismatch(true, index, object, "long"); } return result; } /** * Returns the value at {@code index} if it exists, coercing it if * necessary. * * @param index Which value to get. * @return the value at the specified location. * @throws RuntimeException if no such value exists. */ public String getString(int index) { Object object = get(index); String result = JSON.toString(object); if (result == null) { throw JSON.typeMismatch(true, index, object, "String"); } return result; } /** * Returns the value at {@code index} if it exists and is a {@code * JSONArray}. * * @param index Which value to get. * @return the value at the specified location. * @throws RuntimeException if the value doesn't exist or is not a {@code * JSONArray}. */ public JSONArray getJSONArray(int index) { Object object = get(index); if (object instanceof JSONArray) { return (JSONArray) object; } else { throw JSON.typeMismatch(true, index, object, "JSONArray"); } } /** * Returns the value at {@code index} if it exists and is a {@code * JSONObject}. * * @param index Which value to get. * @return the value at the specified location. * @throws RuntimeException if the value doesn't exist or is not a {@code * JSONObject}. */ public JSONObject getJSONObject(int index) { Object object = get(index); if (object instanceof JSONObject) { return (JSONObject) object; } else { throw JSON.typeMismatch(true, index, object, "JSONObject"); } } @Override public boolean equals(Object o) { return o instanceof JSONArray && ((JSONArray) o).values.equals(values); } @Override public int hashCode() { // diverge from the original, which doesn't implement hashCode return values.hashCode(); } void print(JSONPrintSession session) { session.printSymbol('['); session.indent(); boolean comma = false; for (Object value : values) { if (comma) session.printSymbol(','); session.newline(); JSONObject.printValue(session, value); comma = true; } session.outdent(); if (comma) session.newline(); session.printSymbol(']'); } /** * Puts all objects from the collection into this JSONArray, using {@link #put(Object)}. * * @param collection * List, array, JSONArray, or other iterable object, or null * @return this JSONArray * @since 5.4 */ public JSONArray putAll(Iterable<?> collection) { if (collection != null) { for (Object o : collection) { put(o); } } return this; } /** * Returns an unmodifiable list of the contents of the array. This is a wrapper around the list's internal * storage and is live (changes to the JSONArray affect the returned List). * * @return unmodifiable list of array contents * @since 5.4 */ public List<Object> toList() { return Collections.unmodifiableList(values); } @Override public Iterator<Object> iterator() { return values.iterator(); } }