blob: e6c195d114dec224df4509cb2539e11ca5f94235 [file] [log] [blame]
/*
* Copyright (c) 2005-2010 The Android Open Source Project
* Copyright (c) 2015-2017 BSI Business Systems Integration AG
*
* 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.
*
* Contributors:
* The Android Open Source Project - initial implementation
* BSI Business Systems Integration AG - changes and improvements
*/
package org.json;
import java.lang.reflect.Array;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import org.json.JSONStringer.Scope;
// Note: this class was written without inspecting the non-free org.json sourcecode.
/*
* Changes to the original code:
* -----------------------------
* - Applied Scout code formatting rules
* - equals() method changed to be symmetric
*
* Copyright (c) 2015 BSI Business Systems Integration AG.
*/
/**
* 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.
* <p>
* {@code JSONArray} has the same type coercion behavior and optional/mandatory accessors as {@link JSONObject}. See
* that class' documentation for details.
* <p>
* <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}.
* <p>
* Instances of this class are not thread safe. Although this class is nonfinal, it was not designed for inheritance and
* should not be subclassed. In particular, self-use by overridable methods is not specified. See <i>Effective Java</i>
* Item 17, "Design and Document or inheritance or else prohibit it" for further information.
*/
public class JSONArray {
private final List<Object> m_values;
/**
* Creates a {@code JSONArray} with no values.
*/
public JSONArray() {
m_values = new ArrayList<>();
}
/**
* Creates a new {@code JSONArray} by copying all values from the given collection.
*
* @param copyFrom
* a collection whose values are of supported types. Unsupported values are not permitted and will yield an
* array in an inconsistent state.
*/
public JSONArray(Collection<?> copyFrom) {
this();
if (copyFrom != null) {
for (Object aCopyFrom : copyFrom) {
put(JSONObject.wrap(aCopyFrom));
}
}
}
/**
* 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 JSONException
* if the parse fails or doesn't yield a {@code JSONArray}.
*/
public 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();
if (object instanceof JSONArray) {
JSONArray jsonArray = (JSONArray) object;
m_values = jsonArray.m_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 JSONException
* 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.
*/
public JSONArray(Object array) {
if (!array.getClass().isArray()) {
throw new JSONException("Not a primitive array: " + array.getClass());
}
final int length = Array.getLength(array);
m_values = new ArrayList<>(length);
for (int i = 0; i < length; ++i) {
put(JSONObject.wrap(Array.get(array, i)));
}
}
/**
* Returns the number of values in this array.
*/
public int length() {
return m_values.size();
}
/**
* Appends {@code value} to the end of this array.
*
* @return this array.
*/
public JSONArray put(boolean value) {
m_values.add(value);
return this;
}
/**
* Appends {@code value} to the end of this array.
*
* @param value
* a finite value. May not be {@link Double#isNaN() NaNs} or {@link Double#isInfinite() infinities}.
* @return this array.
*/
public JSONArray put(double value) {
m_values.add(JSON.checkDouble(value));
return this;
}
/**
* Appends {@code value} to the end of this array.
*
* @return this array.
*/
public JSONArray put(int value) {
m_values.add(value);
return this;
}
/**
* Appends {@code value} to the end of this array.
*
* @return this array.
*/
public JSONArray put(long value) {
m_values.add(value);
return this;
}
/**
* Appends {@code value} to the end of this array.
*
* @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}.
* Unsupported values are not permitted and will cause the array to be in an inconsistent state.
* @return this array.
*/
public JSONArray put(Object value) {
m_values.add(value);
return this;
}
/**
* Same as {@link #put}, with added validity checks.
*/
void checkedPut(Object 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.
*
* @return this array.
*/
public JSONArray put(int index, boolean value) {
return put(index, (Boolean) 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 value
* a finite value. May not be {@link Double#isNaN() NaNs} or {@link Double#isInfinite() infinities}.
* @return this array.
*/
public JSONArray put(int index, double value) {
return put(index, (Double) 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.
*
* @return this array.
*/
public JSONArray put(int index, int value) {
return put(index, (Integer) 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.
*
* @return this array.
*/
public JSONArray put(int index, long value) {
return put(index, (Long) 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 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.
*/
public JSONArray put(int index, Object value) {
if (value instanceof Number) {
// deviate from the original by checking all Numbers, not just floats & doubles
JSON.checkDouble(((Number) value).doubleValue());
}
while (m_values.size() <= index) {
m_values.add(null);
}
m_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}.
*/
public boolean isNull(int index) {
Object value = opt(index);
return value == null || value == JSONObject.NULL;
}
/**
* Returns the value at {@code index}.
*
* @throws JSONException
* 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 = m_values.get(index);
if (value == null) {
throw new JSONException("Value at " + index + " is null.");
}
return value;
}
catch (IndexOutOfBoundsException e) { // NOSONAR
throw new JSONException("Index " + index + " out of range [0.." + m_values.size() + ")");
}
}
/**
* Returns the value at {@code index}, or null if the array has no value at {@code index}.
*/
public Object opt(int index) {
if (index < 0 || index >= m_values.size()) {
return null;
}
return m_values.get(index);
}
/**
* Removes and returns the value at {@code index}, or null if the array has no value at {@code index}.
*/
public Object remove(int index) {
if (index < 0 || index >= m_values.size()) {
return null;
}
return m_values.remove(index);
}
/**
* Returns the value at {@code index} if it exists and is a boolean or can be coerced to a boolean.
*
* @throws JSONException
* 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(index, object, "boolean");
}
return result;
}
/**
* Returns the value at {@code index} if it exists and is a boolean or can be coerced to a boolean. Returns false
* otherwise.
*/
public boolean optBoolean(int index) {
return optBoolean(index, false);
}
/**
* Returns the value at {@code index} if it exists and is a boolean or can be coerced to a boolean. Returns
* {@code fallback} otherwise.
*/
public boolean optBoolean(int index, boolean fallback) {
Object object = opt(index);
Boolean result = JSON.toBoolean(object);
return result != null ? result : fallback;
}
/**
* Returns the value at {@code index} if it exists and is a double or can be coerced to a double.
*
* @throws JSONException
* 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(index, object, "double");
}
return result;
}
/**
* Returns the value at {@code index} if it exists and is a double or can be coerced to a double. Returns {@code NaN}
* otherwise.
*/
public double optDouble(int index) {
return optDouble(index, Double.NaN);
}
/**
* Returns the value at {@code index} if it exists and is a double or can be coerced to a double. Returns
* {@code fallback} otherwise.
*/
public double optDouble(int index, double fallback) {
Object object = opt(index);
Double result = JSON.toDouble(object);
return result != null ? result : fallback;
}
/**
* Returns the value at {@code index} if it exists and is an int or can be coerced to an int.
*
* @throws JSONException
* 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(index, object, "int");
}
return result;
}
/**
* Returns the value at {@code index} if it exists and is an int or can be coerced to an int. Returns 0 otherwise.
*/
public int optInt(int index) {
return optInt(index, 0);
}
/**
* Returns the value at {@code index} if it exists and is an int or can be coerced to an int. Returns {@code fallback}
* otherwise.
*/
public int optInt(int index, int fallback) {
Object object = opt(index);
Integer result = JSON.toInteger(object);
return result != null ? result : fallback;
}
/**
* Returns the value at {@code index} if it exists and is a long or can be coerced to a long.
*
* @throws JSONException
* 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(index, object, "long");
}
return result;
}
/**
* Returns the value at {@code index} if it exists and is a long or can be coerced to a long. Returns 0 otherwise.
*/
public long optLong(int index) {
return optLong(index, 0L);
}
/**
* Returns the value at {@code index} if it exists and is a long or can be coerced to a long. Returns {@code fallback}
* otherwise.
*/
public long optLong(int index, long fallback) {
Object object = opt(index);
Long result = JSON.toLong(object);
return result != null ? result : fallback;
}
/**
* Returns the value at {@code index} if it exists, coercing it if necessary.
*
* @throws JSONException
* 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(index, object, "String");
}
return result;
}
/**
* Returns the value at {@code index} if it exists, coercing it if necessary. Returns the empty string if no such
* value exists.
*/
public String optString(int index) {
return optString(index, "");
}
/**
* Returns the value at {@code index} if it exists, coercing it if necessary. Returns {@code fallback} if no such
* value exists.
*/
public String optString(int index, String fallback) {
Object object = opt(index);
String result = JSON.toString(object);
return result != null ? result : fallback;
}
/**
* Returns the value at {@code index} if it exists and is a {@code JSONArray}.
*
* @throws JSONException
* 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(index, object, "JSONArray");
}
}
/**
* Returns the value at {@code index} if it exists and is a {@code JSONArray}. Returns null otherwise.
*/
public JSONArray optJSONArray(int index) {
Object object = opt(index);
return object instanceof JSONArray ? (JSONArray) object : null;
}
/**
* Returns the value at {@code index} if it exists and is a {@code JSONObject}.
*
* @throws JSONException
* 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(index, object, "JSONObject");
}
}
/**
* Returns the value at {@code index} if it exists and is a {@code JSONObject}. Returns null otherwise.
*/
public JSONObject optJSONObject(int index) {
Object object = opt(index);
return object instanceof JSONObject ? (JSONObject) object : null;
}
/**
* Returns a new object whose values are the values in this array, and whose names are the values in {@code names}.
* Names and values are paired up by index from 0 through to the shorter array's length. Names that are not strings
* will be coerced to strings. This method returns null if either array is empty.
*/
public JSONObject toJSONObject(JSONArray names) {
JSONObject result = new JSONObject();
int length = Math.min(names.length(), m_values.size());
if (length == 0) {
return null;
}
for (int i = 0; i < length; i++) {
String name = JSON.toString(names.opt(i));
result.put(name, opt(i));
}
return result;
}
/**
* Returns a new string by alternating this array's values with {@code separator}. This array's string values are
* quoted and have their special characters escaped. For example, the array containing the strings '12" pizza', 'taco'
* and 'soda' joined on '+' returns this:
*
* <pre>
* &quot;12\&quot; pizza&quot; + &quot;taco&quot; + &quot;soda&quot;
* </pre>
*/
public String join(String separator) {
JSONStringer stringer = new JSONStringer();
stringer.open(Scope.NULL, "");
for (int i = 0, size = m_values.size(); i < size; i++) {
if (i > 0) {
stringer.out.append(separator);
}
stringer.value(m_values.get(i));
}
stringer.close(Scope.NULL, Scope.NULL, "");
return stringer.out.toString();
}
/**
* Encodes this array as a compact JSON string, such as:
*
* <pre>
* [94043,90210]
* </pre>
*/
@Override
@SuppressWarnings({"squid:S1166", "squid:S2225"})
public String toString() {
try {
JSONStringer stringer = new JSONStringer();
writeTo(stringer);
return stringer.toString();
}
catch (JSONException e) {
return null;
}
}
/**
* Encodes this array as a human readable JSON string for debugging, such as:
*
* <pre>
* [
* 94043,
* 90210
* ]
* </pre>
*
* @param indentSpaces
* the number of spaces to indent for each level of nesting.
*/
public String toString(int indentSpaces) {
JSONStringer stringer = new JSONStringer(indentSpaces);
writeTo(stringer);
return stringer.toString();
}
void writeTo(JSONStringer stringer) {
stringer.array();
for (Object value : m_values) {
stringer.value(value);
}
stringer.endArray();
}
@Override
public boolean equals(Object o) { // BSI (changed to be symmetric)
if (o == this) {
return true;
}
if (o == null) {
return false;
}
if (getClass() != o.getClass()) {
return false;
}
return ((JSONArray) o).m_values.equals(m_values);
}
@Override
public int hashCode() {
// diverge from the original, which doesn't implement hashCode
return m_values.hashCode();
}
}