mirror of
https://github.com/stleary/JSON-java.git
synced 2026-01-24 00:03:17 -05:00
Compare commits
159 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
a963115ac2 | ||
|
|
f959baa3cb | ||
|
|
c048b36516 | ||
|
|
f6e5bfa2db | ||
|
|
084b24cbe7 | ||
|
|
22ccf1b8e5 | ||
|
|
133c0cc75f | ||
|
|
0578285036 | ||
|
|
48fb5261fe | ||
|
|
8353b9c3f0 | ||
|
|
fe22b242b5 | ||
|
|
7eca507d13 | ||
|
|
e1eabc9c27 | ||
|
|
0d436d92e2 | ||
|
|
45bcba518f | ||
|
|
47fb49b6a8 | ||
|
|
0df034c9fd | ||
|
|
f0a05e6911 | ||
|
|
24093491a8 | ||
|
|
1275f6809d | ||
|
|
9234eab00a | ||
|
|
72f4c3e646 | ||
|
|
df2d6f8363 | ||
|
|
a6e412bded | ||
|
|
2391d248cc | ||
|
|
401495ae86 | ||
|
|
448e204186 | ||
|
|
eb56704e68 | ||
|
|
651511f500 | ||
|
|
a14cb12c85 | ||
|
|
f566a1d9ee | ||
|
|
5920eca2d7 | ||
|
|
3b097d051a | ||
|
|
4e630e58a4 | ||
|
|
b732188e4e | ||
|
|
5369442671 | ||
|
|
bb1138762a | ||
|
|
6a732ec99d | ||
|
|
c798c76ddd | ||
|
|
23d5e52a53 | ||
|
|
98df35449a | ||
|
|
1be6ee31a7 | ||
|
|
a2d3d3c9b5 | ||
|
|
bf9219386a | ||
|
|
85495facbd | ||
|
|
7aba3ac941 | ||
|
|
d51250f6b0 | ||
|
|
9cb8e153bf | ||
|
|
80c1479ad8 | ||
|
|
444335d12a | ||
|
|
a2c0562e04 | ||
|
|
153972afdf | ||
|
|
4a8ff28fd8 | ||
|
|
fa457a4113 | ||
|
|
b7f708b222 | ||
|
|
12411b7981 | ||
|
|
61801c623e | ||
|
|
1915aab7c4 | ||
|
|
8439039da7 | ||
|
|
a6bdd081eb | ||
|
|
a30d71fdca | ||
|
|
cdd67b0aef | ||
|
|
89f6e7f6a6 | ||
|
|
346fb26f2d | ||
|
|
6daabb43ab | ||
|
|
57f785c42e | ||
|
|
3eecd67a3b | ||
|
|
5a587bbdfd | ||
|
|
e0534b3ec7 | ||
|
|
7dd9e01dae | ||
|
|
88ca19042b | ||
|
|
b4036e6a8e | ||
|
|
89f16ad0af | ||
|
|
6f92a3ab4e | ||
|
|
beae279b21 | ||
|
|
a642329314 | ||
|
|
9abb35ad39 | ||
|
|
c0a1d5f741 | ||
|
|
04a765647e | ||
|
|
4f78ec666a | ||
|
|
7d6b76de37 | ||
|
|
5cfe216ffd | ||
|
|
c33ad9c062 | ||
|
|
7a124d857d | ||
|
|
f1b0210b8a | ||
|
|
8ef8e1463d | ||
|
|
48b6aa3e4c | ||
|
|
cff5cc6c74 | ||
|
|
04e8ea84dc | ||
|
|
812955e39d | ||
|
|
bc623e36d6 | ||
|
|
5dd78bc0b9 | ||
|
|
e638955034 | ||
|
|
3f9b53fee4 | ||
|
|
a0f90b776d | ||
|
|
fafaeb7aa6 | ||
|
|
e356739a2f | ||
|
|
fca7e17b38 | ||
|
|
fb96e870a9 | ||
|
|
638273af7a | ||
|
|
1ffcf3915c | ||
|
|
4565bddcbb | ||
|
|
b5bcb68968 | ||
|
|
7823d3a4f3 | ||
|
|
d6227c83d7 | ||
|
|
f54b5e4b0d | ||
|
|
30b3680050 | ||
|
|
3ed9154f63 | ||
|
|
669316d29e | ||
|
|
30a70c8886 | ||
|
|
93f4b34890 | ||
|
|
9c87d6e214 | ||
|
|
f27e5fe04d | ||
|
|
2528e60b09 | ||
|
|
4e601fd46e | ||
|
|
9f19c22b77 | ||
|
|
9000901a11 | ||
|
|
f03eb56071 | ||
|
|
b4bbc58644 | ||
|
|
cf43419015 | ||
|
|
e896497602 | ||
|
|
d284c81e16 | ||
|
|
a526b41b67 | ||
|
|
9f07853f19 | ||
|
|
e29c541353 | ||
|
|
de5f768cff | ||
|
|
8ca8a80753 | ||
|
|
fb71d9d40f | ||
|
|
b80141a19c | ||
|
|
c0e467c848 | ||
|
|
8680b10716 | ||
|
|
2559114dbb | ||
|
|
bb048e3ffb | ||
|
|
579784d73e | ||
|
|
c03054b1a6 | ||
|
|
4e0faebe62 | ||
|
|
c6089e53f5 | ||
|
|
cfbc306673 | ||
|
|
449ec8745e | ||
|
|
f91d0c8a52 | ||
|
|
4571978840 | ||
|
|
76ec2fe5a2 | ||
|
|
0200c984cc | ||
|
|
8f3e5ade18 | ||
|
|
20b4f85efe | ||
|
|
5bc8dae5d0 | ||
|
|
8fa9be742c | ||
|
|
b48abe6558 | ||
|
|
97023e1098 | ||
|
|
143db39d27 | ||
|
|
75894086e5 | ||
|
|
fa46da45f4 | ||
|
|
f912e5d1c4 | ||
|
|
6bf3d38889 | ||
|
|
c62df81b3e | ||
|
|
e1f69ff3fe | ||
|
|
50d619698f | ||
|
|
29103e3228 | ||
|
|
2630676f36 |
43
.github/workflows/codeql-analysis.yml
vendored
Normal file
43
.github/workflows/codeql-analysis.yml
vendored
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
name: "CodeQL"
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [ master ]
|
||||||
|
pull_request:
|
||||||
|
# The branches below must be a subset of the branches above
|
||||||
|
branches: [ master ]
|
||||||
|
schedule:
|
||||||
|
- cron: '18 18 * * 1'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
analyze:
|
||||||
|
name: Analyze
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
actions: read
|
||||||
|
contents: read
|
||||||
|
security-events: write
|
||||||
|
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
language: [ 'java' ]
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
# Initializes the CodeQL tools for scanning.
|
||||||
|
- name: Initialize CodeQL
|
||||||
|
uses: github/codeql-action/init@v1
|
||||||
|
with:
|
||||||
|
languages: ${{ matrix.language }}
|
||||||
|
# If you wish to specify custom queries, you can do so here or in a config file.
|
||||||
|
# By default, queries listed here will override any specified in a config file.
|
||||||
|
# Prefix the list here with "+" to use these queries and those in the config file.
|
||||||
|
# queries: ./path/to/local/query, your-org/your-repo/queries@main
|
||||||
|
|
||||||
|
- run: "mvn clean compile -Dmaven.test.skip=true -Dmaven.site.skip=true -Dmaven.javadoc.skip=true"
|
||||||
|
|
||||||
|
- name: Perform CodeQL Analysis
|
||||||
|
uses: github/codeql-action/analyze@v1
|
||||||
6
.github/workflows/pipeline.yml
vendored
6
.github/workflows/pipeline.yml
vendored
@@ -12,7 +12,7 @@ on:
|
|||||||
jobs:
|
jobs:
|
||||||
# old-school build and jar method. No tests run or compiled.
|
# old-school build and jar method. No tests run or compiled.
|
||||||
build-1_6:
|
build-1_6:
|
||||||
runs-on: ubuntu-16.04
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
# build for java 1.6, however don't run any tests
|
# build for java 1.6, however don't run any tests
|
||||||
@@ -38,11 +38,11 @@ jobs:
|
|||||||
path: target/org.json.jar
|
path: target/org.json.jar
|
||||||
|
|
||||||
build:
|
build:
|
||||||
runs-on: ubuntu-16.04
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
# build against supported Java LTS versions:
|
# build against supported Java LTS versions:
|
||||||
java: [ 1.7, 8, 11 ]
|
java: [ 8, 11 ]
|
||||||
name: Java ${{ matrix.java }}
|
name: Java ${{ matrix.java }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
|
|||||||
76
CODE_OF_CONDUCT.md
Normal file
76
CODE_OF_CONDUCT.md
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
# Contributor Covenant Code of Conduct
|
||||||
|
|
||||||
|
## Our Pledge
|
||||||
|
|
||||||
|
In the interest of fostering an open and welcoming environment, we as
|
||||||
|
contributors and maintainers pledge to making participation in our project and
|
||||||
|
our community a harassment-free experience for everyone, regardless of age, body
|
||||||
|
size, disability, ethnicity, sex characteristics, gender identity and expression,
|
||||||
|
level of experience, education, socio-economic status, nationality, personal
|
||||||
|
appearance, race, religion, or sexual identity and orientation.
|
||||||
|
|
||||||
|
## Our Standards
|
||||||
|
|
||||||
|
Examples of behavior that contributes to creating a positive environment
|
||||||
|
include:
|
||||||
|
|
||||||
|
* Using welcoming and inclusive language
|
||||||
|
* Being respectful of differing viewpoints and experiences
|
||||||
|
* Gracefully accepting constructive criticism
|
||||||
|
* Focusing on what is best for the community
|
||||||
|
* Showing empathy towards other community members
|
||||||
|
|
||||||
|
Examples of unacceptable behavior by participants include:
|
||||||
|
|
||||||
|
* The use of sexualized language or imagery and unwelcome sexual attention or
|
||||||
|
advances
|
||||||
|
* Trolling, insulting/derogatory comments, and personal or political attacks
|
||||||
|
* Public or private harassment
|
||||||
|
* Publishing others' private information, such as a physical or electronic
|
||||||
|
address, without explicit permission
|
||||||
|
* Other conduct which could reasonably be considered inappropriate in a
|
||||||
|
professional setting
|
||||||
|
|
||||||
|
## Our Responsibilities
|
||||||
|
|
||||||
|
Project maintainers are responsible for clarifying the standards of acceptable
|
||||||
|
behavior and are expected to take appropriate and fair corrective action in
|
||||||
|
response to any instances of unacceptable behavior.
|
||||||
|
|
||||||
|
Project maintainers have the right and responsibility to remove, edit, or
|
||||||
|
reject comments, commits, code, wiki edits, issues, and other contributions
|
||||||
|
that are not aligned to this Code of Conduct, or to ban temporarily or
|
||||||
|
permanently any contributor for other behaviors that they deem inappropriate,
|
||||||
|
threatening, offensive, or harmful.
|
||||||
|
|
||||||
|
## Scope
|
||||||
|
|
||||||
|
This Code of Conduct applies both within project spaces and in public spaces
|
||||||
|
when an individual is representing the project or its community. Examples of
|
||||||
|
representing a project or community include using an official project e-mail
|
||||||
|
address, posting via an official social media account, or acting as an appointed
|
||||||
|
representative at an online or offline event. Representation of a project may be
|
||||||
|
further defined and clarified by project maintainers.
|
||||||
|
|
||||||
|
## Enforcement
|
||||||
|
|
||||||
|
Instances of abusive, harassing, or otherwise unacceptable behavior may be
|
||||||
|
reported by contacting the project team at jsonjava060@gmail.com. All
|
||||||
|
complaints will be reviewed and investigated and will result in a response that
|
||||||
|
is deemed necessary and appropriate to the circumstances. The project team is
|
||||||
|
obligated to maintain confidentiality with regard to the reporter of an incident.
|
||||||
|
Further details of specific enforcement policies may be posted separately.
|
||||||
|
|
||||||
|
Project maintainers who do not follow or enforce the Code of Conduct in good
|
||||||
|
faith may face temporary or permanent repercussions as determined by other
|
||||||
|
members of the project's leadership.
|
||||||
|
|
||||||
|
## Attribution
|
||||||
|
|
||||||
|
This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4,
|
||||||
|
available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html
|
||||||
|
|
||||||
|
[homepage]: https://www.contributor-covenant.org
|
||||||
|
|
||||||
|
For answers to common questions about this code of conduct, see
|
||||||
|
https://www.contributor-covenant.org/faq
|
||||||
22
CONTRIBUTING.md
Normal file
22
CONTRIBUTING.md
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
# Contribution Guidelines
|
||||||
|
|
||||||
|
Feel free to work on any issue with a #hacktoberfest label.
|
||||||
|
|
||||||
|
If you discover an issue you would like to work on, you can add a new issue to the list. If it meets our criteria, a hacktoberfest label will be added.
|
||||||
|
|
||||||
|
# Who is allowed to submit pull requests for this project?
|
||||||
|
|
||||||
|
Anyone can submit pull requests for code, tests, or documentation.
|
||||||
|
|
||||||
|
# How do you decide which pull requests to accept?
|
||||||
|
|
||||||
|
* Does it call out a bug that needs to be fixed? If so, it goes to the top of the list.
|
||||||
|
* Does it fix a major user inconvenience? These are given high priority as well.
|
||||||
|
* Does it align with the specs? If not, it will probably not be accepted. It turns out there are gray areas in the specs. If this is in a gray area, it will likely be given the benefit of the doubt.
|
||||||
|
* Does it break the existing behavior of the lib? If so, it will not be accepted, unless it fixes an egregious bug. This is happening less frequently now.
|
||||||
|
|
||||||
|
# For more guidance, see these links:
|
||||||
|
|
||||||
|
[README.md (includes build instructions)](https://github.com/stleary/JSON-java#readme)
|
||||||
|
|
||||||
|
[FAQ - all your questions answered](https://github.com/stleary/JSON-java/wiki/FAQ)
|
||||||
433
Examples.md
Normal file
433
Examples.md
Normal file
@@ -0,0 +1,433 @@
|
|||||||
|
<h1>Examples</h1>
|
||||||
|
<p>Imports used in the examples: </p>
|
||||||
|
|
||||||
|
```java
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Properties;
|
||||||
|
```
|
||||||
|
|
||||||
|
<p>This document's intention is to explain to new-comers the basics of this project</p>
|
||||||
|
|
||||||
|
|
||||||
|
<h2>Part 1: Creating a JSON document</h2>
|
||||||
|
|
||||||
|
<h3>Using JSONArray</h3>
|
||||||
|
|
||||||
|
```java
|
||||||
|
private static void JSONExampleArray1() {
|
||||||
|
//We create a JSONObject from a String containing an array using JSONArray
|
||||||
|
//Firstly, we declare an Array in a String
|
||||||
|
|
||||||
|
String arrayStr =
|
||||||
|
"["+"true,"+"false,"+ "\"true\","+ "\"false\","+"\"hello\","+"23.45e-4,"+
|
||||||
|
"\"23.45\","+"42,"+"\"43\","+"["+"\"world\""+"],"+
|
||||||
|
"{"+
|
||||||
|
"\"key1\":\"value1\","+
|
||||||
|
"\"key2\":\"value2\","+
|
||||||
|
"\"key3\":\"value3\","+
|
||||||
|
"\"key4\":\"value4\""+
|
||||||
|
"},"+
|
||||||
|
"0,"+"\"-1\""+
|
||||||
|
"]";
|
||||||
|
|
||||||
|
//Then, we initializate the JSONArray thanks to its constructor
|
||||||
|
|
||||||
|
JSONArray array = new JSONArray(arrayStr);
|
||||||
|
System.out.println("Values array: "+ array);
|
||||||
|
|
||||||
|
//We convert that array into a JSONObject, but first, we need the labels, so we need another JSONArray with the labels.
|
||||||
|
//Here we will use an auxiliary function to get one for the example.
|
||||||
|
|
||||||
|
JSONArray list = listNumberArray(array.length());
|
||||||
|
System.out.println("Label Array: "+ list.toString());
|
||||||
|
//Now, we construct the JSONObject using both the value array and the label array.
|
||||||
|
JSONObject object = array.toJSONObject(list);
|
||||||
|
System.out.println("Final JSONOBject: " + object);
|
||||||
|
}
|
||||||
|
|
||||||
|
//This method creates an JSONArray of labels in which those are generated by their positions
|
||||||
|
|
||||||
|
private static JSONArray listNumberArray(int max){
|
||||||
|
JSONArray res = new JSONArray();
|
||||||
|
for (int i=0; i<max;i++) {
|
||||||
|
//The value of the labels must be an String in order to make it work
|
||||||
|
res.put(String.valueOf(i));
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
```java
|
||||||
|
private static void JSONExampleArray2() {
|
||||||
|
|
||||||
|
//We can also create an Array without a String by creating an empty array and adding elements to it
|
||||||
|
|
||||||
|
JSONArray array = new JSONArray();
|
||||||
|
|
||||||
|
//Adding elements with .put()
|
||||||
|
|
||||||
|
array.put("value");
|
||||||
|
array.put(5);
|
||||||
|
array.put(-23.45e67);
|
||||||
|
array.put(true);
|
||||||
|
|
||||||
|
//We convert it to JSONObject providing a label arrray like last time
|
||||||
|
|
||||||
|
JSONArray list = listNumberArray(array.length());
|
||||||
|
JSONObject object = array.toJSONObject(list);
|
||||||
|
System.out.println("Final JSONOBject: " + object);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
<h3>Using JSONStringer</h3>
|
||||||
|
|
||||||
|
```java
|
||||||
|
private static void JSONExampleStringer() {
|
||||||
|
|
||||||
|
//We initializate the JSONStringer
|
||||||
|
|
||||||
|
JSONStringer jsonStringer = new JSONStringer();
|
||||||
|
|
||||||
|
//Now we start the process of adding elements with .object()
|
||||||
|
|
||||||
|
jsonStringer.object();
|
||||||
|
|
||||||
|
//We can now add elements as keys and values with .values () and .key()
|
||||||
|
|
||||||
|
jsonStringer.key("trueValue").value(true);
|
||||||
|
jsonStringer.key("falseValue").value(false);
|
||||||
|
jsonStringer.key("nullValue").value(null);
|
||||||
|
jsonStringer.key("stringValue").value("hello world!");
|
||||||
|
jsonStringer.key("complexStringValue").value("h\be\tllo w\u1234orld!");
|
||||||
|
jsonStringer.key("intValue").value(42);
|
||||||
|
jsonStringer.key("doubleValue").value(-23.45e67);
|
||||||
|
|
||||||
|
//We end this prcedure with .ednObject
|
||||||
|
|
||||||
|
jsonStringer.endObject();
|
||||||
|
|
||||||
|
//Once we have a JSONStringer, we convert it to JSONObject generating a String and using JSONObject's contructor.
|
||||||
|
|
||||||
|
String str = jsonStringer.toString();
|
||||||
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
|
|
||||||
|
System.out.println("Final JSONOBject: " + jsonObject);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
<h3>Using JSONObject</h3>
|
||||||
|
|
||||||
|
```java
|
||||||
|
private static void JSONExampleObject1() {
|
||||||
|
|
||||||
|
//We can create a JSONObject from a String with the class builder
|
||||||
|
|
||||||
|
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||||
|
JSONObject example = new JSONObject(string);
|
||||||
|
System.out.println("Final JSONObject: " + example);
|
||||||
|
|
||||||
|
}
|
||||||
|
```
|
||||||
|
```java
|
||||||
|
private static void JSONExampleObject2() {
|
||||||
|
|
||||||
|
//We can also create a JSONObject directly without messing around with any of the other functions.
|
||||||
|
|
||||||
|
JSONObject example = new JSONObject();
|
||||||
|
|
||||||
|
|
||||||
|
//Now we add the keys and values in a similar way as the Stringer method
|
||||||
|
example.put("key", "value");
|
||||||
|
|
||||||
|
//As you can see, the first entry is the key and the second would be its associeted value.
|
||||||
|
|
||||||
|
example.put("key2", 5);
|
||||||
|
example.put("key3", -23.45e67);
|
||||||
|
example.put("trueValue", true);
|
||||||
|
|
||||||
|
//We can't add null values thougth
|
||||||
|
|
||||||
|
//example.put("nullValue", null); //This is not possible
|
||||||
|
|
||||||
|
System.out.println("Final JSONOBject: " + example);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
```java
|
||||||
|
private static void JSONExampleObject3() {
|
||||||
|
|
||||||
|
//We can also create a JSONObject with a Java Map
|
||||||
|
//YoU will need a Map whose keys are Strings. The values can be whatever you want
|
||||||
|
|
||||||
|
Map<String,Double> map = new HashMap<String, Double>();
|
||||||
|
|
||||||
|
map.put("key1", 1.0);
|
||||||
|
map.put("key2", -23.45e67);
|
||||||
|
|
||||||
|
//We create the JSONObject with the map with its class builder
|
||||||
|
|
||||||
|
JSONObject example = new JSONObject(map);
|
||||||
|
System.out.println("Final JSONOBject: " + example);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
<h3>Using JSONWriter</h3>
|
||||||
|
|
||||||
|
```java
|
||||||
|
private static void JSONExamplWriter() {
|
||||||
|
|
||||||
|
//This method works in a very similar way to Object and Stringer in the construction of the JSON.
|
||||||
|
//The difference is that it needs a Java object called "Appendable" like StringBuilder
|
||||||
|
|
||||||
|
StringBuilder write = new StringBuilder();
|
||||||
|
JSONWriter jsonWriter = new JSONWriter(write);
|
||||||
|
|
||||||
|
//We behave now the same way as Stringer
|
||||||
|
|
||||||
|
jsonWriter.object();
|
||||||
|
|
||||||
|
jsonWriter.key("trueValue").value(true);
|
||||||
|
jsonWriter.key("falseValue").value(false);
|
||||||
|
jsonWriter.key("nullValue").value(null);
|
||||||
|
jsonWriter.key("stringValue").value("hello world!");
|
||||||
|
jsonWriter.key("complexStringValue").value("h\be\tllo w\u1234orld!");
|
||||||
|
jsonWriter.key("intValue").value(42);
|
||||||
|
jsonWriter.key("doubleValue").value(-23.45e67);
|
||||||
|
|
||||||
|
jsonWriter.endObject();
|
||||||
|
|
||||||
|
//The resoult should be in the "write" object
|
||||||
|
|
||||||
|
System.out.println("JSON: " + write.toString());
|
||||||
|
|
||||||
|
//The difference is that we don't get a JSONObject in this one.
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
```
|
||||||
|
```java
|
||||||
|
private static void JSONExampleTokener() {
|
||||||
|
|
||||||
|
//A partir de una String podemos crear un JSONTokener, que lo podemos usar alternativamente para JSONArray,JSONObject
|
||||||
|
|
||||||
|
String string = "this is not a valid JSON string";
|
||||||
|
JSONTokener token = new JSONTokener(string);
|
||||||
|
|
||||||
|
//Now you can use the token in JSONObject and Array the same way as a String
|
||||||
|
|
||||||
|
JSONObject object = new JSONObject(token);
|
||||||
|
JSONArray array = new JSONArray(token);
|
||||||
|
|
||||||
|
}
|
||||||
|
```
|
||||||
|
<h2>Part 2: Conversion methods</h2>
|
||||||
|
<p>We don't need to have a JSON document to work. This project also admits conversions from other type of files.</p>
|
||||||
|
<p>Secondly, we can also convert from JSON to those type of files.</p>
|
||||||
|
|
||||||
|
<h3>Extra: Conversion to JSONArray</h3>
|
||||||
|
|
||||||
|
```java
|
||||||
|
private static void JSONObjectToArray() {
|
||||||
|
//We start with a JSONObject
|
||||||
|
|
||||||
|
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||||
|
|
||||||
|
JSONObject example = new JSONObject(string);
|
||||||
|
|
||||||
|
//We need a list of key strings like the reverse operation
|
||||||
|
|
||||||
|
JSONArray keyStrings = listNumberArray(example.length());
|
||||||
|
|
||||||
|
//Then we convert to the Array using both elelements
|
||||||
|
|
||||||
|
JSONArray array = example.toJSONArray(keyStrings);
|
||||||
|
|
||||||
|
System.out.println("Final JSONArray: " + array);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
<h3>XML Conversions</h3>
|
||||||
|
|
||||||
|
```java
|
||||||
|
private static void XMLToExampleConversion() {
|
||||||
|
|
||||||
|
//We start with a JSONObject
|
||||||
|
|
||||||
|
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||||
|
JSONObject example = new JSONObject(string);
|
||||||
|
|
||||||
|
//We obtain a String with XML format with toString()
|
||||||
|
|
||||||
|
String output = XML.toString(example);
|
||||||
|
System.out.println("Final XML: " + output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
```java
|
||||||
|
private static void XMLFromExampleConversion() {
|
||||||
|
|
||||||
|
//We start with a string with the XML format
|
||||||
|
|
||||||
|
String string = "<0>value</0><1>5</1><2>-2.345E+68</2><3>true</3>";
|
||||||
|
|
||||||
|
//We obtain a JSONObject with toJSONOBject()
|
||||||
|
|
||||||
|
JSONObject output = XML.toJSONObject(string);
|
||||||
|
|
||||||
|
System.out.println("Final JSONObject: " + output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
<h3>Cookie Conversions</h3>
|
||||||
|
|
||||||
|
```java
|
||||||
|
private static void CookieToExampleConversion() {
|
||||||
|
|
||||||
|
//We start with a JSONObject
|
||||||
|
//The JSONOBject needs to entries that gives the cookie a name and gives the field "name" a name too.
|
||||||
|
//The Cokkie format doesn't support booleans
|
||||||
|
|
||||||
|
String string = "{\"name\":\"Cookie-Name\",\"value\":\"name\",\"1\":5,\"2\":-2.345E68,\"3\":'true'}";
|
||||||
|
JSONObject example = new JSONObject(string);
|
||||||
|
|
||||||
|
//We obtain a String with Cookie format with toString()
|
||||||
|
|
||||||
|
String output = Cookie.toString(example);
|
||||||
|
System.out.println("Final Cookie: " + output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
```java
|
||||||
|
private static void CookieFromExampleConversion() {
|
||||||
|
|
||||||
|
//We start with a string with the Cookie format
|
||||||
|
|
||||||
|
String string = "Cookie-Name=name;1=5;2=-2.345E%2b68;3=true";
|
||||||
|
|
||||||
|
//We obtain a JSONObject with toJSONOBject()
|
||||||
|
|
||||||
|
JSONObject output = Cookie.toJSONObject(string);
|
||||||
|
System.out.println("Final JSONObject: " + output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
<h3>HTTP Conversions</h3>
|
||||||
|
|
||||||
|
```java
|
||||||
|
private static void HTTPToExampleConversion() {
|
||||||
|
|
||||||
|
//We start with a JSONObject
|
||||||
|
//The JSONObject must have the minimun header for a HTTP request or header
|
||||||
|
|
||||||
|
String string = "{\"Method\":\"POST\",\"Request-URI\":'/',\"HTTP-Version\":'HTTP/1.1',\"Value1\":true,\"Value2\":2,\"Value3\":-2.345E68}";
|
||||||
|
|
||||||
|
JSONObject example = new JSONObject(string);
|
||||||
|
|
||||||
|
//We obtain a String with HTTP format with toString()
|
||||||
|
|
||||||
|
String output = HTTP.toString(example);
|
||||||
|
System.out.println("Final HTTP: " + output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
```java
|
||||||
|
private static void HTTPFromExampleConversion() {
|
||||||
|
|
||||||
|
//We start with a string with the HTTP format
|
||||||
|
|
||||||
|
String string = "Final HTTP: POST '/' HTTP/1.1 Value3: -2.345E+68 Value1: true Value2: 2";
|
||||||
|
|
||||||
|
//We obtain a JSONObject with toJSONOBject()
|
||||||
|
|
||||||
|
JSONObject output = HTTP.toJSONObject(string);
|
||||||
|
System.out.println("Final JSONObject: " + output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
<h3>CDL Conversions</h3>
|
||||||
|
|
||||||
|
```java
|
||||||
|
private static void CDLToExampleConversion() {
|
||||||
|
|
||||||
|
//We start with some JSONObjects with the same values in the keys but different values in the "values"
|
||||||
|
|
||||||
|
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||||
|
JSONObject example = new JSONObject(string);
|
||||||
|
|
||||||
|
String string2 = "{\"0\":\"value2\",\"1\":6,\"2\":-8.345E68,\"3\":false}";
|
||||||
|
JSONObject example2 = new JSONObject(string2);
|
||||||
|
|
||||||
|
//We need now a JSONArray with those JSONObjects
|
||||||
|
|
||||||
|
JSONArray array = new JSONArray();
|
||||||
|
array.put(example);
|
||||||
|
array.put(example2);
|
||||||
|
|
||||||
|
//We obtain a String with XML format with toString()
|
||||||
|
|
||||||
|
String output = CDL.toString(array);
|
||||||
|
System.out.println("Final CDL: \r\n" + output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
```java
|
||||||
|
private static void CDLFromExampleConversion() {
|
||||||
|
|
||||||
|
//We start wtih a String with the CDL format
|
||||||
|
|
||||||
|
String string = "0,1,2,3\n"
|
||||||
|
+ "value,5,-2.345E+68,true\n"
|
||||||
|
+ "value2,6,-8.345E+68,false";
|
||||||
|
|
||||||
|
//We obtain a JSONArray with toJSONOBject()
|
||||||
|
|
||||||
|
JSONArray output = CDL.toJSONArray(string);
|
||||||
|
System.out.println("Final JSONArray: " + output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
<h3>Properties Conversions</h3>
|
||||||
|
|
||||||
|
```java
|
||||||
|
private static Properties PropertyToExampleConversion() {
|
||||||
|
|
||||||
|
//We start with a JSONObject
|
||||||
|
|
||||||
|
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||||
|
JSONObject example = new JSONObject(string);
|
||||||
|
|
||||||
|
//We obtain a String with Properties format with toString()
|
||||||
|
|
||||||
|
Properties output = Property.toProperties(example);
|
||||||
|
System.out.println("Final Properties: " + output);
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
```java
|
||||||
|
private static void PropertyFromExampleConversion() {
|
||||||
|
|
||||||
|
//We start with a Properties object
|
||||||
|
|
||||||
|
Properties input = PropertyToExampleConversion();
|
||||||
|
|
||||||
|
//We obtain a JSONObject with toJSONOBject()
|
||||||
|
|
||||||
|
JSONObject output = Property.toJSONObject(input);
|
||||||
|
System.out.println("Final JSONObject: " + output);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
<h2>List of all examples methods</h2>
|
||||||
|
|
||||||
|
```java
|
||||||
|
public static void main(String[] args) {
|
||||||
|
//JSONObjectToArray();
|
||||||
|
//JSONExampleArray1();
|
||||||
|
//JSONExampleArray2();
|
||||||
|
//JSONExampleStringer();
|
||||||
|
//JSONExampleObject1();
|
||||||
|
//JSONExampleObject2();
|
||||||
|
//JSONExampleObject3();
|
||||||
|
//JSONExamplWriter();
|
||||||
|
//XMLToExampleConversion();
|
||||||
|
//XMLFromExampleConversion();
|
||||||
|
//CookieToExampleConversion();
|
||||||
|
//CookieFromExampleConversion();
|
||||||
|
//HTTPToExampleConversion();
|
||||||
|
//HTTPFromExampleConversion();
|
||||||
|
//CDLToExampleConversion();
|
||||||
|
//CDLFromExampleConversion();
|
||||||
|
//PropertyToExampleConversion();
|
||||||
|
//PropertyFromExampleConversion();
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
23
LICENSE
23
LICENSE
@@ -1,23 +1,2 @@
|
|||||||
|
Public Domain.
|
||||||
Copyright (c) 2002 JSON.org
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
|
|
||||||
|
|||||||
206
README.md
206
README.md
@@ -1,9 +1,15 @@
|
|||||||
|

|
||||||
|
|
||||||
|
<sub><sup>image credit: Ismael Pérez Ortiz</sup></sub>
|
||||||
|
|
||||||
|
|
||||||
JSON in Java [package org.json]
|
JSON in Java [package org.json]
|
||||||
===============================
|
===============================
|
||||||
|
|
||||||
[](https://mvnrepository.com/artifact/org.json/json)
|
[](https://mvnrepository.com/artifact/org.json/json)
|
||||||
|
|
||||||
**[Click here if you just want the latest release jar file.](https://repo1.maven.org/maven2/org/json/json/20201115/json-20201115.jar)**
|
**[Click here if you just want the latest release jar file.](https://search.maven.org/remotecontent?filepath=org/json/json/20230618/json-20230618.jar)**
|
||||||
|
|
||||||
|
|
||||||
# Overview
|
# Overview
|
||||||
|
|
||||||
@@ -22,9 +28,9 @@ Project goals include:
|
|||||||
|
|
||||||
The files in this package implement JSON encoders and decoders. The package can also convert between JSON and XML, HTTP headers, Cookies, and CDL.
|
The files in this package implement JSON encoders and decoders. The package can also convert between JSON and XML, HTTP headers, Cookies, and CDL.
|
||||||
|
|
||||||
The license includes this restriction: ["The software shall be used for good, not evil."](https://en.wikipedia.org/wiki/Douglas_Crockford#%22Good,_not_Evil%22) If your conscience cannot live with that, then choose a different package.
|
# If you would like to contribute to this project
|
||||||
|
|
||||||
**If you would like to contribute to this project**
|
For more information on contributions, please see [CONTRIBUTING.md](https://github.com/stleary/JSON-java/blob/master/docs/CONTRIBUTING.md)
|
||||||
|
|
||||||
Bug fixes, code improvements, and unit test coverage changes are welcome! Because this project is currently in the maintenance phase, the kinds of changes that can be accepted are limited. For more information, please read the [FAQ](https://github.com/stleary/JSON-java/wiki/FAQ).
|
Bug fixes, code improvements, and unit test coverage changes are welcome! Because this project is currently in the maintenance phase, the kinds of changes that can be accepted are limited. For more information, please read the [FAQ](https://github.com/stleary/JSON-java/wiki/FAQ).
|
||||||
|
|
||||||
@@ -36,7 +42,7 @@ The org.json package can be built from the command line, Maven, and Gradle. The
|
|||||||
|
|
||||||
*Build the class files from the package root directory src/main/java*
|
*Build the class files from the package root directory src/main/java*
|
||||||
````
|
````
|
||||||
javac org\json\*.java
|
javac org/json/*.java
|
||||||
````
|
````
|
||||||
|
|
||||||
*Create the jar file in the current directory*
|
*Create the jar file in the current directory*
|
||||||
@@ -46,7 +52,8 @@ jar cf json-java.jar org/json/*.class
|
|||||||
|
|
||||||
*Compile a program that uses the jar (see example code below)*
|
*Compile a program that uses the jar (see example code below)*
|
||||||
````
|
````
|
||||||
javac -cp .;json-java.jar Test.java
|
javac -cp .;json-java.jar Test.java (Windows)
|
||||||
|
javac -cp .:json-java.jar Test.java (Unix Systems)
|
||||||
````
|
````
|
||||||
|
|
||||||
*Test file contents*
|
*Test file contents*
|
||||||
@@ -63,7 +70,8 @@ public class Test {
|
|||||||
|
|
||||||
*Execute the Test file*
|
*Execute the Test file*
|
||||||
````
|
````
|
||||||
java -cp .;json-java.jar Test
|
java -cp .;json-java.jar Test (Windows)
|
||||||
|
java -cp .:json-java.jar Test (Unix Systems)
|
||||||
````
|
````
|
||||||
|
|
||||||
*Expected output*
|
*Expected output*
|
||||||
@@ -88,192 +96,12 @@ gradlew clean build test
|
|||||||
|
|
||||||
# Notes
|
# Notes
|
||||||
|
|
||||||
**Recent directory structure change**
|
For more information, please see [NOTES.md](https://github.com/stleary/JSON-java/blob/master/docs/NOTES.md)
|
||||||
|
|
||||||
_Due to a recent commit - [#515 Merge tests and pom and code](https://github.com/stleary/JSON-java/pull/515) - the structure of the project has changed from a flat directory containing all of the Java files to a directory structure that includes unit tests and several tools used to build the project jar and run the unit tests. If you have difficulty using the new structure, please open an issue so we can work through it._
|
|
||||||
|
|
||||||
**Implementation notes**
|
|
||||||
|
|
||||||
Numeric types in this package comply with
|
|
||||||
[ECMA-404: The JSON Data Interchange Format](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf) and
|
|
||||||
[RFC 8259: The JavaScript Object Notation (JSON) Data Interchange Format](https://tools.ietf.org/html/rfc8259#section-6).
|
|
||||||
This package fully supports `Integer`, `Long`, and `Double` Java types. Partial support
|
|
||||||
for `BigInteger` and `BigDecimal` values in `JSONObject` and `JSONArray` objects is provided
|
|
||||||
in the form of `get()`, `opt()`, and `put()` API methods.
|
|
||||||
|
|
||||||
Although 1.6 compatibility is currently supported, it is not a project goal and might be
|
|
||||||
removed in some future release.
|
|
||||||
|
|
||||||
In compliance with RFC8259 page 10 section 9, the parser is more lax with what is valid
|
|
||||||
JSON then the Generator. For Example, the tab character (U+0009) is allowed when reading
|
|
||||||
JSON Text strings, but when output by the Generator, the tab is properly converted to \t in
|
|
||||||
the string. Other instances may occur where reading invalid JSON text does not cause an
|
|
||||||
error to be generated. Malformed JSON Texts such as missing end " (quote) on strings or
|
|
||||||
invalid number formats (1.2e6.3) will cause errors as such documents can not be read
|
|
||||||
reliably.
|
|
||||||
|
|
||||||
Some notable exceptions that the JSON Parser in this library accepts are:
|
|
||||||
* Unquoted keys `{ key: "value" }`
|
|
||||||
* Unquoted values `{ "key": value }`
|
|
||||||
* Unescaped literals like "tab" in string values `{ "key": "value with an unescaped tab" }`
|
|
||||||
* Numbers out of range for `Double` or `Long` are parsed as strings
|
|
||||||
|
|
||||||
Recent pull requests added a new method `putAll` on the JSONArray. The `putAll` method
|
|
||||||
works similarly to other `put` methods in that it does not call `JSONObject.wrap` for items
|
|
||||||
added. This can lead to inconsistent object representation in JSONArray structures.
|
|
||||||
|
|
||||||
For example, code like this will create a mixed JSONArray, some items wrapped, others
|
|
||||||
not:
|
|
||||||
|
|
||||||
```java
|
|
||||||
SomeBean[] myArr = new SomeBean[]{ new SomeBean(1), new SomeBean(2) };
|
|
||||||
// these will be wrapped
|
|
||||||
JSONArray jArr = new JSONArray(myArr);
|
|
||||||
// these will not be wrapped
|
|
||||||
jArr.putAll(new SomeBean[]{ new SomeBean(3), new SomeBean(4) });
|
|
||||||
```
|
|
||||||
|
|
||||||
For structure consistency, it would be recommended that the above code is changed
|
|
||||||
to look like 1 of 2 ways.
|
|
||||||
|
|
||||||
Option 1:
|
|
||||||
```Java
|
|
||||||
SomeBean[] myArr = new SomeBean[]{ new SomeBean(1), new SomeBean(2) };
|
|
||||||
JSONArray jArr = new JSONArray();
|
|
||||||
// these will not be wrapped
|
|
||||||
jArr.putAll(myArr);
|
|
||||||
// these will not be wrapped
|
|
||||||
jArr.putAll(new SomeBean[]{ new SomeBean(3), new SomeBean(4) });
|
|
||||||
// our jArr is now consistent.
|
|
||||||
```
|
|
||||||
|
|
||||||
Option 2:
|
|
||||||
```Java
|
|
||||||
SomeBean[] myArr = new SomeBean[]{ new SomeBean(1), new SomeBean(2) };
|
|
||||||
// these will be wrapped
|
|
||||||
JSONArray jArr = new JSONArray(myArr);
|
|
||||||
// these will be wrapped
|
|
||||||
jArr.putAll(new JSONArray(new SomeBean[]{ new SomeBean(3), new SomeBean(4) }));
|
|
||||||
// our jArr is now consistent.
|
|
||||||
```
|
|
||||||
|
|
||||||
**Unit Test Conventions**
|
|
||||||
|
|
||||||
Test filenames should consist of the name of the module being tested, with the suffix "Test".
|
|
||||||
For example, <b>Cookie.java</b> is tested by <b>CookieTest.java</b>.
|
|
||||||
|
|
||||||
<b>The fundamental issues with JSON-Java testing are:</b><br>
|
|
||||||
* <b>JSONObjects</b> are unordered, making simple string comparison ineffective.
|
|
||||||
* Comparisons via **equals()** is not currently supported. Neither <b>JSONArray</b> nor <b>JSONObject</b> override <b>hashCode()</b> or <b>equals()</b>, so comparison defaults to the <b>Object</b> equals(), which is not useful.
|
|
||||||
* Access to the <b>JSONArray</b> and <b>JSONObject</b> internal containers for comparison is not currently available.
|
|
||||||
|
|
||||||
<b>General issues with unit testing are:</b><br>
|
|
||||||
* Just writing tests to make coverage goals tends to result in poor tests.
|
|
||||||
* Unit tests are a form of documentation - how a given method works is demonstrated by the test. So for a code reviewer or future developer looking at code a good test helps explain how a function is supposed to work according to the original author. This can be difficult if you are not the original developer.
|
|
||||||
* It is difficult to evaluate unit tests in a vacuum. You also need to see the code being tested to understand if a test is good.
|
|
||||||
* Without unit tests, it is hard to feel confident about the quality of the code, especially when fixing bugs or refactoring. Good tests prevent regressions and keep the intent of the code correct.
|
|
||||||
* If you have unit test results along with pull requests, the reviewer has an easier time understanding your code and determining if it works as intended.
|
|
||||||
|
|
||||||
|
|
||||||
# Files
|
# Files
|
||||||
|
|
||||||
**JSONObject.java**: The `JSONObject` can parse text from a `String` or a `JSONTokener`
|
For more information on files, please see [FILES.md](https://github.com/stleary/JSON-java/blob/master/docs/FILES.md)
|
||||||
to produce a map-like object. The object provides methods for manipulating its
|
|
||||||
contents, and for producing a JSON compliant object serialization.
|
|
||||||
|
|
||||||
**JSONArray.java**: The `JSONArray` can parse text from a String or a `JSONTokener`
|
|
||||||
to produce a vector-like object. The object provides methods for manipulating
|
|
||||||
its contents, and for producing a JSON compliant array serialization.
|
|
||||||
|
|
||||||
**JSONTokener.java**: The `JSONTokener` breaks a text into a sequence of individual
|
|
||||||
tokens. It can be constructed from a `String`, `Reader`, or `InputStream`. It also can
|
|
||||||
parse text from a `String`, `Number`, `Boolean` or `null` like `"hello"`, `42`, `true`,
|
|
||||||
`null` to produce a simple json object.
|
|
||||||
|
|
||||||
**JSONException.java**: The `JSONException` is the standard exception type thrown
|
|
||||||
by this package.
|
|
||||||
|
|
||||||
**JSONPointer.java**: Implementation of
|
|
||||||
[JSON Pointer (RFC 6901)](https://tools.ietf.org/html/rfc6901). Supports
|
|
||||||
JSON Pointers both in the form of string representation and URI fragment
|
|
||||||
representation.
|
|
||||||
|
|
||||||
**JSONPropertyIgnore.java**: Annotation class that can be used on Java Bean getter methods.
|
|
||||||
When used on a bean method that would normally be serialized into a `JSONObject`, it
|
|
||||||
overrides the getter-to-key-name logic and forces the property to be excluded from the
|
|
||||||
resulting `JSONObject`.
|
|
||||||
|
|
||||||
**JSONPropertyName.java**: Annotation class that can be used on Java Bean getter methods.
|
|
||||||
When used on a bean method that would normally be serialized into a `JSONObject`, it
|
|
||||||
overrides the getter-to-key-name logic and uses the value of the annotation. The Bean
|
|
||||||
processor will look through the class hierarchy. This means you can use the annotation on
|
|
||||||
a base class or interface and the value of the annotation will be used even if the getter
|
|
||||||
is overridden in a child class.
|
|
||||||
|
|
||||||
**JSONString.java**: The `JSONString` interface requires a `toJSONString` method,
|
|
||||||
allowing an object to provide its own serialization.
|
|
||||||
|
|
||||||
**JSONStringer.java**: The `JSONStringer` provides a convenient facility for
|
|
||||||
building JSON strings.
|
|
||||||
|
|
||||||
**JSONWriter.java**: The `JSONWriter` provides a convenient facility for building
|
|
||||||
JSON text through a writer.
|
|
||||||
|
|
||||||
|
|
||||||
**CDL.java**: `CDL` provides support for converting between JSON and comma
|
|
||||||
delimited lists.
|
|
||||||
|
|
||||||
**Cookie.java**: `Cookie` provides support for converting between JSON and cookies.
|
|
||||||
|
|
||||||
**CookieList.java**: `CookieList` provides support for converting between JSON and
|
|
||||||
cookie lists.
|
|
||||||
|
|
||||||
**HTTP.java**: `HTTP` provides support for converting between JSON and HTTP headers.
|
|
||||||
|
|
||||||
**HTTPTokener.java**: `HTTPTokener` extends `JSONTokener` for parsing HTTP headers.
|
|
||||||
|
|
||||||
**XML.java**: `XML` provides support for converting between JSON and XML.
|
|
||||||
|
|
||||||
**JSONML.java**: `JSONML` provides support for converting between JSONML and XML.
|
|
||||||
|
|
||||||
**XMLTokener.java**: `XMLTokener` extends `JSONTokener` for parsing XML text.
|
|
||||||
|
|
||||||
|
|
||||||
# Release history:
|
# Release history:
|
||||||
|
|
||||||
JSON-java releases can be found by searching the Maven repository for groupId "org.json"
|
For the release history, please see [RELEASES.md](https://github.com/stleary/JSON-java/blob/master/docs/RELEASES.md)
|
||||||
and artifactId "json". For example:
|
|
||||||
[https://search.maven.org/search?q=g:org.json%20AND%20a:json&core=gav](https://search.maven.org/search?q=g:org.json%20AND%20a:json&core=gav)
|
|
||||||
|
|
||||||
~~~
|
|
||||||
20210307 Recent commits and potentially breaking fix to JSONPointer
|
|
||||||
|
|
||||||
20201115 Recent commits and first release after project structure change
|
|
||||||
|
|
||||||
20200518 Recent commits and snapshot before project structure change
|
|
||||||
|
|
||||||
20190722 Recent commits
|
|
||||||
|
|
||||||
20180813 POM change to include Automatic-Module-Name (#431)
|
|
||||||
|
|
||||||
20180130 Recent commits
|
|
||||||
|
|
||||||
20171018 Checkpoint for recent commits.
|
|
||||||
|
|
||||||
20170516 Roll up recent commits.
|
|
||||||
|
|
||||||
20160810 Revert code that was breaking opt*() methods.
|
|
||||||
|
|
||||||
20160807 This release contains a bug in the JSONObject.opt*() and JSONArray.opt*() methods,
|
|
||||||
it is not recommended for use.
|
|
||||||
Java 1.6 compatability fixed, JSONArray.toList() and JSONObject.toMap(),
|
|
||||||
RFC4180 compatibility, JSONPointer, some exception fixes, optional XML type conversion.
|
|
||||||
Contains the latest code as of 7 Aug 2016
|
|
||||||
|
|
||||||
20160212 Java 1.6 compatibility, OSGi bundle. Contains the latest code as of 12 Feb 2016.
|
|
||||||
|
|
||||||
20151123 JSONObject and JSONArray initialization with generics. Contains the latest code as of 23 Nov 2015.
|
|
||||||
|
|
||||||
20150729 Checkpoint for Maven central repository release. Contains the latest code
|
|
||||||
as of 29 July 2015.
|
|
||||||
~~~
|
|
||||||
|
|||||||
5
SECURITY.md
Normal file
5
SECURITY.md
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
# Security Policy
|
||||||
|
|
||||||
|
## Reporting a Vulnerability
|
||||||
|
|
||||||
|
Please follow the instructions in the ["How are vulnerabilities and exploits handled?"](https://github.com/stleary/JSON-java/wiki/FAQ#how-are-vulnerabilities-and-exploits-handled) section in the FAQ.
|
||||||
@@ -30,7 +30,7 @@ subprojects {
|
|||||||
}
|
}
|
||||||
|
|
||||||
group = 'org.json'
|
group = 'org.json'
|
||||||
version = 'v20200429-SNAPSHOT'
|
version = 'v20211205-SNAPSHOT'
|
||||||
description = 'JSON in Java'
|
description = 'JSON in Java'
|
||||||
sourceCompatibility = '1.7'
|
sourceCompatibility = '1.7'
|
||||||
|
|
||||||
|
|||||||
22
docs/CONTRIBUTING.md
Normal file
22
docs/CONTRIBUTING.md
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
# Contribution Guidelines
|
||||||
|
|
||||||
|
Feel free to work on any issue with a #hacktoberfest label.
|
||||||
|
|
||||||
|
If you discover an issue you would like to work on, you can add a new issue to the list. If it meets our criteria, a hacktoberfest label will be added.
|
||||||
|
|
||||||
|
# Who is allowed to submit pull requests for this project?
|
||||||
|
|
||||||
|
Anyone can submit pull requests for code, tests, or documentation.
|
||||||
|
|
||||||
|
# How do you decide which pull requests to accept?
|
||||||
|
|
||||||
|
* Does it call out a bug that needs to be fixed? If so, it goes to the top of the list.
|
||||||
|
* Does it fix a major user inconvenience? These are given high priority as well.
|
||||||
|
* Does it align with the specs? If not, it will probably not be accepted. It turns out there are gray areas in the specs. If this is in a gray area, it will likely be given the benefit of the doubt.
|
||||||
|
* Does it break the existing behavior of the lib? If so, it will not be accepted, unless it fixes an egregious bug. This is happening less frequently now.
|
||||||
|
|
||||||
|
# For more guidance, see these links:
|
||||||
|
|
||||||
|
[README.md (includes build instructions)](https://github.com/stleary/JSON-java#readme)
|
||||||
|
|
||||||
|
[FAQ - all your questions answered](https://github.com/stleary/JSON-java/wiki/FAQ)
|
||||||
62
docs/FILES.md
Normal file
62
docs/FILES.md
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
# Files
|
||||||
|
|
||||||
|
**JSONObject.java**: The `JSONObject` can parse text from a `String` or a `JSONTokener`
|
||||||
|
to produce a map-like object. The object provides methods for manipulating its
|
||||||
|
contents, and for producing a JSON compliant object serialization.
|
||||||
|
|
||||||
|
**JSONArray.java**: The `JSONArray` can parse text from a String or a `JSONTokener`
|
||||||
|
to produce a vector-like object. The object provides methods for manipulating
|
||||||
|
its contents, and for producing a JSON compliant array serialization.
|
||||||
|
|
||||||
|
**JSONTokener.java**: The `JSONTokener` breaks a text into a sequence of individual
|
||||||
|
tokens. It can be constructed from a `String`, `Reader`, or `InputStream`. It also can
|
||||||
|
parse text from a `String`, `Number`, `Boolean` or `null` like `"hello"`, `42`, `true`,
|
||||||
|
`null` to produce a simple json object.
|
||||||
|
|
||||||
|
**JSONException.java**: The `JSONException` is the standard exception type thrown
|
||||||
|
by this package.
|
||||||
|
|
||||||
|
**JSONPointer.java**: Implementation of
|
||||||
|
[JSON Pointer (RFC 6901)](https://tools.ietf.org/html/rfc6901). Supports
|
||||||
|
JSON Pointers both in the form of string representation and URI fragment
|
||||||
|
representation.
|
||||||
|
|
||||||
|
**JSONPropertyIgnore.java**: Annotation class that can be used on Java Bean getter methods.
|
||||||
|
When used on a bean method that would normally be serialized into a `JSONObject`, it
|
||||||
|
overrides the getter-to-key-name logic and forces the property to be excluded from the
|
||||||
|
resulting `JSONObject`.
|
||||||
|
|
||||||
|
**JSONPropertyName.java**: Annotation class that can be used on Java Bean getter methods.
|
||||||
|
When used on a bean method that would normally be serialized into a `JSONObject`, it
|
||||||
|
overrides the getter-to-key-name logic and uses the value of the annotation. The Bean
|
||||||
|
processor will look through the class hierarchy. This means you can use the annotation on
|
||||||
|
a base class or interface and the value of the annotation will be used even if the getter
|
||||||
|
is overridden in a child class.
|
||||||
|
|
||||||
|
**JSONString.java**: The `JSONString` interface requires a `toJSONString` method,
|
||||||
|
allowing an object to provide its own serialization.
|
||||||
|
|
||||||
|
**JSONStringer.java**: The `JSONStringer` provides a convenient facility for
|
||||||
|
building JSON strings.
|
||||||
|
|
||||||
|
**JSONWriter.java**: The `JSONWriter` provides a convenient facility for building
|
||||||
|
JSON text through a writer.
|
||||||
|
|
||||||
|
|
||||||
|
**CDL.java**: `CDL` provides support for converting between JSON and comma
|
||||||
|
delimited lists.
|
||||||
|
|
||||||
|
**Cookie.java**: `Cookie` provides support for converting between JSON and cookies.
|
||||||
|
|
||||||
|
**CookieList.java**: `CookieList` provides support for converting between JSON and
|
||||||
|
cookie lists.
|
||||||
|
|
||||||
|
**HTTP.java**: `HTTP` provides support for converting between JSON and HTTP headers.
|
||||||
|
|
||||||
|
**HTTPTokener.java**: `HTTPTokener` extends `JSONTokener` for parsing HTTP headers.
|
||||||
|
|
||||||
|
**XML.java**: `XML` provides support for converting between JSON and XML.
|
||||||
|
|
||||||
|
**JSONML.java**: `JSONML` provides support for converting between JSONML and XML.
|
||||||
|
|
||||||
|
**XMLTokener.java**: `XMLTokener` extends `JSONTokener` for parsing XML text.
|
||||||
87
docs/NOTES.md
Normal file
87
docs/NOTES.md
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
# Notes
|
||||||
|
|
||||||
|
**Recent directory structure change**
|
||||||
|
|
||||||
|
_Due to a recent commit - [#515 Merge tests and pom and code](https://github.com/stleary/JSON-java/pull/515) - the structure of the project has changed from a flat directory containing all of the Java files to a directory structure that includes unit tests and several tools used to build the project jar and run the unit tests. If you have difficulty using the new structure, please open an issue so we can work through it._
|
||||||
|
|
||||||
|
**Implementation notes**
|
||||||
|
|
||||||
|
Numeric types in this package comply with
|
||||||
|
[ECMA-404: The JSON Data Interchange Format](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf) and
|
||||||
|
[RFC 8259: The JavaScript Object Notation (JSON) Data Interchange Format](https://tools.ietf.org/html/rfc8259#section-6).
|
||||||
|
This package fully supports `Integer`, `Long`, and `Double` Java types. Partial support
|
||||||
|
for `BigInteger` and `BigDecimal` values in `JSONObject` and `JSONArray` objects is provided
|
||||||
|
in the form of `get()`, `opt()`, and `put()` API methods.
|
||||||
|
|
||||||
|
Although 1.6 compatibility is currently supported, it is not a project goal and might be
|
||||||
|
removed in some future release.
|
||||||
|
|
||||||
|
In compliance with RFC8259 page 10 section 9, the parser is more lax with what is valid
|
||||||
|
JSON then the Generator. For Example, the tab character (U+0009) is allowed when reading
|
||||||
|
JSON Text strings, but when output by the Generator, the tab is properly converted to \t in
|
||||||
|
the string. Other instances may occur where reading invalid JSON text does not cause an
|
||||||
|
error to be generated. Malformed JSON Texts such as missing end " (quote) on strings or
|
||||||
|
invalid number formats (1.2e6.3) will cause errors as such documents can not be read
|
||||||
|
reliably.
|
||||||
|
|
||||||
|
Some notable exceptions that the JSON Parser in this library accepts are:
|
||||||
|
* Unquoted keys `{ key: "value" }`
|
||||||
|
* Unquoted values `{ "key": value }`
|
||||||
|
* Unescaped literals like "tab" in string values `{ "key": "value with an unescaped tab" }`
|
||||||
|
* Numbers out of range for `Double` or `Long` are parsed as strings
|
||||||
|
|
||||||
|
Recent pull requests added a new method `putAll` on the JSONArray. The `putAll` method
|
||||||
|
works similarly to other `put` methods in that it does not call `JSONObject.wrap` for items
|
||||||
|
added. This can lead to inconsistent object representation in JSONArray structures.
|
||||||
|
|
||||||
|
For example, code like this will create a mixed JSONArray, some items wrapped, others
|
||||||
|
not:
|
||||||
|
|
||||||
|
```java
|
||||||
|
SomeBean[] myArr = new SomeBean[]{ new SomeBean(1), new SomeBean(2) };
|
||||||
|
// these will be wrapped
|
||||||
|
JSONArray jArr = new JSONArray(myArr);
|
||||||
|
// these will not be wrapped
|
||||||
|
jArr.putAll(new SomeBean[]{ new SomeBean(3), new SomeBean(4) });
|
||||||
|
```
|
||||||
|
|
||||||
|
For structure consistency, it would be recommended that the above code is changed
|
||||||
|
to look like 1 of 2 ways.
|
||||||
|
|
||||||
|
Option 1:
|
||||||
|
```Java
|
||||||
|
SomeBean[] myArr = new SomeBean[]{ new SomeBean(1), new SomeBean(2) };
|
||||||
|
JSONArray jArr = new JSONArray();
|
||||||
|
// these will not be wrapped
|
||||||
|
jArr.putAll(myArr);
|
||||||
|
// these will not be wrapped
|
||||||
|
jArr.putAll(new SomeBean[]{ new SomeBean(3), new SomeBean(4) });
|
||||||
|
// our jArr is now consistent.
|
||||||
|
```
|
||||||
|
|
||||||
|
Option 2:
|
||||||
|
```Java
|
||||||
|
SomeBean[] myArr = new SomeBean[]{ new SomeBean(1), new SomeBean(2) };
|
||||||
|
// these will be wrapped
|
||||||
|
JSONArray jArr = new JSONArray(myArr);
|
||||||
|
// these will be wrapped
|
||||||
|
jArr.putAll(new JSONArray(new SomeBean[]{ new SomeBean(3), new SomeBean(4) }));
|
||||||
|
// our jArr is now consistent.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Unit Test Conventions**
|
||||||
|
|
||||||
|
Test filenames should consist of the name of the module being tested, with the suffix "Test".
|
||||||
|
For example, <b>Cookie.java</b> is tested by <b>CookieTest.java</b>.
|
||||||
|
|
||||||
|
<b>The fundamental issues with JSON-Java testing are:</b><br>
|
||||||
|
* <b>JSONObjects</b> are unordered, making simple string comparison ineffective.
|
||||||
|
* Comparisons via **equals()** is not currently supported. Neither <b>JSONArray</b> nor <b>JSONObject</b> override <b>hashCode()</b> or <b>equals()</b>, so comparison defaults to the <b>Object</b> equals(), which is not useful.
|
||||||
|
* Access to the <b>JSONArray</b> and <b>JSONObject</b> internal containers for comparison is not currently available.
|
||||||
|
|
||||||
|
<b>General issues with unit testing are:</b><br>
|
||||||
|
* Just writing tests to make coverage goals tends to result in poor tests.
|
||||||
|
* Unit tests are a form of documentation - how a given method works is demonstrated by the test. So for a code reviewer or future developer looking at code a good test helps explain how a function is supposed to work according to the original author. This can be difficult if you are not the original developer.
|
||||||
|
* It is difficult to evaluate unit tests in a vacuum. You also need to see the code being tested to understand if a test is good.
|
||||||
|
* Without unit tests, it is hard to feel confident about the quality of the code, especially when fixing bugs or refactoring. Good tests prevent regressions and keep the intent of the code correct.
|
||||||
|
* If you have unit test results along with pull requests, the reviewer has an easier time understanding your code and determining if it works as intended.
|
||||||
49
docs/RELEASES.md
Normal file
49
docs/RELEASES.md
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
# Release history:
|
||||||
|
|
||||||
|
JSON-java releases can be found by searching the Maven repository for groupId "org.json"
|
||||||
|
and artifactId "json". For example:
|
||||||
|
[https://search.maven.org/search?q=g:org.json%20AND%20a:json&core=gav](https://search.maven.org/search?q=g:org.json%20AND%20a:json&core=gav)
|
||||||
|
|
||||||
|
~~~
|
||||||
|
20230618 Final release with Java 1.6 compatibility. Future releases will require Java 1.8 or greater.
|
||||||
|
|
||||||
|
20230227 Fix for CVE-2022-45688 and recent commits
|
||||||
|
|
||||||
|
20220924 New License - public domain, and some minor updates
|
||||||
|
|
||||||
|
20220320 Wrap StackOverflow with JSONException
|
||||||
|
|
||||||
|
20211205 Recent commits and some bug fixes for similar()
|
||||||
|
|
||||||
|
20210307 Recent commits and potentially breaking fix to JSONPointer
|
||||||
|
|
||||||
|
20201115 Recent commits and first release after project structure change
|
||||||
|
|
||||||
|
20200518 Recent commits and snapshot before project structure change
|
||||||
|
|
||||||
|
20190722 Recent commits
|
||||||
|
|
||||||
|
20180813 POM change to include Automatic-Module-Name (#431)
|
||||||
|
JSONObject(Map) now throws an exception if any of a map keys are null (#405)
|
||||||
|
|
||||||
|
20180130 Recent commits
|
||||||
|
|
||||||
|
20171018 Checkpoint for recent commits.
|
||||||
|
|
||||||
|
20170516 Roll up recent commits.
|
||||||
|
|
||||||
|
20160810 Revert code that was breaking opt*() methods.
|
||||||
|
|
||||||
|
20160807 This release contains a bug in the JSONObject.opt*() and JSONArray.opt*() methods,
|
||||||
|
it is not recommended for use.
|
||||||
|
Java 1.6 compatability fixed, JSONArray.toList() and JSONObject.toMap(),
|
||||||
|
RFC4180 compatibility, JSONPointer, some exception fixes, optional XML type conversion.
|
||||||
|
Contains the latest code as of 7 Aug 2016
|
||||||
|
|
||||||
|
20160212 Java 1.6 compatibility, OSGi bundle. Contains the latest code as of 12 Feb 2016.
|
||||||
|
|
||||||
|
20151123 JSONObject and JSONArray initialization with generics. Contains the latest code as of 23 Nov 2015.
|
||||||
|
|
||||||
|
20150729 Checkpoint for Maven central repository release. Contains the latest code
|
||||||
|
as of 29 July 2015.
|
||||||
|
~~~
|
||||||
5
docs/SECURITY.md
Normal file
5
docs/SECURITY.md
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
# Security Policy
|
||||||
|
|
||||||
|
## Reporting a Vulnerability
|
||||||
|
|
||||||
|
Please follow the instructions in the ["How are vulnerabilities and exploits handled?"](https://github.com/stleary/JSON-java/wiki/FAQ#how-are-vulnerabilities-and-exploits-handled) section in the FAQ.
|
||||||
BIN
images/JsonJava.png
Normal file
BIN
images/JsonJava.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 15 KiB |
35
pom.xml
35
pom.xml
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
<groupId>org.json</groupId>
|
<groupId>org.json</groupId>
|
||||||
<artifactId>json</artifactId>
|
<artifactId>json</artifactId>
|
||||||
<version>20210307</version>
|
<version>20230618</version>
|
||||||
<packaging>bundle</packaging>
|
<packaging>bundle</packaging>
|
||||||
|
|
||||||
<name>JSON in Java</name>
|
<name>JSON in Java</name>
|
||||||
@@ -18,10 +18,6 @@
|
|||||||
This is a reference implementation. There is a large number of JSON packages
|
This is a reference implementation. There is a large number of JSON packages
|
||||||
in Java. Perhaps someday the Java community will standardize on one. Until
|
in Java. Perhaps someday the Java community will standardize on one. Until
|
||||||
then, choose carefully.
|
then, choose carefully.
|
||||||
|
|
||||||
The license includes this restriction: "The software shall be used for good,
|
|
||||||
not evil." If your conscience cannot live with that, then choose a different
|
|
||||||
package.
|
|
||||||
</description>
|
</description>
|
||||||
<url>https://github.com/douglascrockford/JSON-java</url>
|
<url>https://github.com/douglascrockford/JSON-java</url>
|
||||||
|
|
||||||
@@ -39,28 +35,9 @@
|
|||||||
|
|
||||||
<licenses>
|
<licenses>
|
||||||
<license>
|
<license>
|
||||||
<name>The JSON License</name>
|
<name>Public Domain</name>
|
||||||
<url>http://json.org/license.html</url>
|
<url>https://github.com/stleary/JSON-java/blob/master/LICENSE</url>
|
||||||
<distribution>repo</distribution>
|
<distribution>repo</distribution>
|
||||||
<comments>Copyright (c) 2002 JSON.org
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and
|
|
||||||
associated documentation files (the "Software"), to deal in the Software without restriction, including
|
|
||||||
without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the
|
|
||||||
following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all copies or substantial
|
|
||||||
portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT
|
|
||||||
LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
||||||
NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
||||||
WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
||||||
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
||||||
</comments>
|
|
||||||
</license>
|
</license>
|
||||||
</licenses>
|
</licenses>
|
||||||
|
|
||||||
@@ -162,6 +139,12 @@
|
|||||||
<goals>
|
<goals>
|
||||||
<goal>sign</goal>
|
<goal>sign</goal>
|
||||||
</goals>
|
</goals>
|
||||||
|
<configuration>
|
||||||
|
<gpgArguments>
|
||||||
|
<arg>--pinentry-mode</arg>
|
||||||
|
<arg>loopback</arg>
|
||||||
|
</gpgArguments>
|
||||||
|
</configuration>
|
||||||
</execution>
|
</execution>
|
||||||
</executions>
|
</executions>
|
||||||
</plugin>
|
</plugin>
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -3,27 +3,7 @@ package org.json;
|
|||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -109,7 +89,7 @@ public class Cookie {
|
|||||||
// parse the remaining cookie attributes
|
// parse the remaining cookie attributes
|
||||||
while (x.more()) {
|
while (x.more()) {
|
||||||
name = unescape(x.nextTo("=;")).trim().toLowerCase(Locale.ROOT);
|
name = unescape(x.nextTo("=;")).trim().toLowerCase(Locale.ROOT);
|
||||||
// don't allow a cookies attributes to overwrite it's name or value.
|
// don't allow a cookies attributes to overwrite its name or value.
|
||||||
if("name".equalsIgnoreCase(name)) {
|
if("name".equalsIgnoreCase(name)) {
|
||||||
throw new JSONException("Illegal attribute name: 'name'");
|
throw new JSONException("Illegal attribute name: 'name'");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
@@ -288,7 +268,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
.equalsIgnoreCase("true"))) {
|
.equalsIgnoreCase("true"))) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
throw wrongValueFormatException(index, "boolean", null);
|
throw wrongValueFormatException(index, "boolean", object, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -309,7 +289,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
try {
|
try {
|
||||||
return Double.parseDouble(object.toString());
|
return Double.parseDouble(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(index, "double", e);
|
throw wrongValueFormatException(index, "double", object, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -326,12 +306,12 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
public float getFloat(int index) throws JSONException {
|
public float getFloat(int index) throws JSONException {
|
||||||
final Object object = this.get(index);
|
final Object object = this.get(index);
|
||||||
if(object instanceof Number) {
|
if(object instanceof Number) {
|
||||||
return ((Float)object).floatValue();
|
return ((Number)object).floatValue();
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
return Float.parseFloat(object.toString());
|
return Float.parseFloat(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(index, "float", e);
|
throw wrongValueFormatException(index, "float", object, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -353,7 +333,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
}
|
}
|
||||||
return JSONObject.stringToNumber(object.toString());
|
return JSONObject.stringToNumber(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(index, "number", e);
|
throw wrongValueFormatException(index, "number", object, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -378,14 +358,14 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
// If it did, I would re-implement this with the Enum.valueOf
|
// If it did, I would re-implement this with the Enum.valueOf
|
||||||
// method and place any thrown exception in the JSONException
|
// method and place any thrown exception in the JSONException
|
||||||
throw wrongValueFormatException(index, "enum of type "
|
throw wrongValueFormatException(index, "enum of type "
|
||||||
+ JSONObject.quote(clazz.getSimpleName()), null);
|
+ JSONObject.quote(clazz.getSimpleName()), opt(index), null);
|
||||||
}
|
}
|
||||||
return val;
|
return val;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the BigDecimal value associated with an index. If the value is float
|
* Get the BigDecimal value associated with an index. If the value is float
|
||||||
* or double, the the {@link BigDecimal#BigDecimal(double)} constructor
|
* or double, the {@link BigDecimal#BigDecimal(double)} constructor
|
||||||
* will be used. See notes on the constructor for conversion issues that
|
* will be used. See notes on the constructor for conversion issues that
|
||||||
* may arise.
|
* may arise.
|
||||||
*
|
*
|
||||||
@@ -441,7 +421,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
try {
|
try {
|
||||||
return Integer.parseInt(object.toString());
|
return Integer.parseInt(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(index, "int", e);
|
throw wrongValueFormatException(index, "int", object, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -460,7 +440,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
if (object instanceof JSONArray) {
|
if (object instanceof JSONArray) {
|
||||||
return (JSONArray) object;
|
return (JSONArray) object;
|
||||||
}
|
}
|
||||||
throw wrongValueFormatException(index, "JSONArray", null);
|
throw wrongValueFormatException(index, "JSONArray", object, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -478,7 +458,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
if (object instanceof JSONObject) {
|
if (object instanceof JSONObject) {
|
||||||
return (JSONObject) object;
|
return (JSONObject) object;
|
||||||
}
|
}
|
||||||
throw wrongValueFormatException(index, "JSONObject", null);
|
throw wrongValueFormatException(index, "JSONObject", object, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -499,7 +479,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
try {
|
try {
|
||||||
return Long.parseLong(object.toString());
|
return Long.parseLong(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(index, "long", e);
|
throw wrongValueFormatException(index, "long", object, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -517,7 +497,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
if (object instanceof String) {
|
if (object instanceof String) {
|
||||||
return (String) object;
|
return (String) object;
|
||||||
}
|
}
|
||||||
throw wrongValueFormatException(index, "String", null);
|
throw wrongValueFormatException(index, "String", object, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -792,7 +772,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* Get the optional BigDecimal value associated with an index. The
|
* Get the optional BigDecimal value associated with an index. The
|
||||||
* defaultValue is returned if there is no value for the index, or if the
|
* defaultValue is returned if there is no value for the index, or if the
|
||||||
* value is not a number and cannot be converted to a number. If the value
|
* value is not a number and cannot be converted to a number. If the value
|
||||||
* is float or double, the the {@link BigDecimal#BigDecimal(double)}
|
* is float or double, the {@link BigDecimal#BigDecimal(double)}
|
||||||
* constructor will be used. See notes on the constructor for conversion
|
* constructor will be used. See notes on the constructor for conversion
|
||||||
* issues that may arise.
|
* issues that may arise.
|
||||||
*
|
*
|
||||||
@@ -1157,7 +1137,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* The Map value.
|
* The Map value.
|
||||||
* @return this.
|
* @return this.
|
||||||
* @throws JSONException
|
* @throws JSONException
|
||||||
* If the index is negative or if the the value is an invalid
|
* If the index is negative or if the value is an invalid
|
||||||
* number.
|
* number.
|
||||||
* @throws NullPointerException
|
* @throws NullPointerException
|
||||||
* If a key in the map is <code>null</code>
|
* If a key in the map is <code>null</code>
|
||||||
@@ -1180,7 +1160,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* String, or the JSONObject.NULL object.
|
* String, or the JSONObject.NULL object.
|
||||||
* @return this.
|
* @return this.
|
||||||
* @throws JSONException
|
* @throws JSONException
|
||||||
* If the index is negative or if the the value is an invalid
|
* If the index is negative or if the value is an invalid
|
||||||
* number.
|
* number.
|
||||||
*/
|
*/
|
||||||
public JSONArray put(int index, Object value) throws JSONException {
|
public JSONArray put(int index, Object value) throws JSONException {
|
||||||
@@ -1383,7 +1363,13 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
} else if (valueThis instanceof Number && valueOther instanceof Number) {
|
} else if (valueThis instanceof Number && valueOther instanceof Number) {
|
||||||
return JSONObject.isNumberSimilar((Number)valueThis, (Number)valueOther);
|
if (!JSONObject.isNumberSimilar((Number)valueThis, (Number)valueOther)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else if (valueThis instanceof JSONString && valueOther instanceof JSONString) {
|
||||||
|
if (!((JSONString) valueThis).toJSONString().equals(((JSONString) valueOther).toJSONString())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
} else if (!valueThis.equals(valueOther)) {
|
} else if (!valueThis.equals(valueOther)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -1462,6 +1448,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* <small>(right bracket)</small>.
|
* <small>(right bracket)</small>.
|
||||||
* @throws JSONException if a called function fails
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings("resource")
|
||||||
public String toString(int indentFactor) throws JSONException {
|
public String toString(int indentFactor) throws JSONException {
|
||||||
StringWriter sw = new StringWriter();
|
StringWriter sw = new StringWriter();
|
||||||
synchronized (sw.getBuffer()) {
|
synchronized (sw.getBuffer()) {
|
||||||
@@ -1511,6 +1498,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* @return The writer.
|
* @return The writer.
|
||||||
* @throws JSONException if a called function fails or unable to write
|
* @throws JSONException if a called function fails or unable to write
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings("resource")
|
||||||
public Writer write(Writer writer, int indentFactor, int indent)
|
public Writer write(Writer writer, int indentFactor, int indent)
|
||||||
throws JSONException {
|
throws JSONException {
|
||||||
try {
|
try {
|
||||||
@@ -1678,22 +1666,6 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new JSONException in a common format for incorrect conversions.
|
|
||||||
* @param idx index of the item
|
|
||||||
* @param valueType the type of value being coerced to
|
|
||||||
* @param cause optional cause of the coercion failure
|
|
||||||
* @return JSONException that can be thrown.
|
|
||||||
*/
|
|
||||||
private static JSONException wrongValueFormatException(
|
|
||||||
int idx,
|
|
||||||
String valueType,
|
|
||||||
Throwable cause) {
|
|
||||||
return new JSONException(
|
|
||||||
"JSONArray[" + idx + "] is not a " + valueType + "."
|
|
||||||
, cause);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new JSONException in a common format for incorrect conversions.
|
* Create a new JSONException in a common format for incorrect conversions.
|
||||||
* @param idx index of the item
|
* @param idx index of the item
|
||||||
@@ -1706,8 +1678,19 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
String valueType,
|
String valueType,
|
||||||
Object value,
|
Object value,
|
||||||
Throwable cause) {
|
Throwable cause) {
|
||||||
|
if(value == null) {
|
||||||
return new JSONException(
|
return new JSONException(
|
||||||
"JSONArray[" + idx + "] is not a " + valueType + " (" + value + ")."
|
"JSONArray[" + idx + "] is not a " + valueType + " (null)."
|
||||||
|
, cause);
|
||||||
|
}
|
||||||
|
// don't try to toString collections or known object types that could be large.
|
||||||
|
if(value instanceof Map || value instanceof Iterable || value instanceof JSONObject) {
|
||||||
|
return new JSONException(
|
||||||
|
"JSONArray[" + idx + "] is not a " + valueType + " (" + value.getClass() + ")."
|
||||||
|
, cause);
|
||||||
|
}
|
||||||
|
return new JSONException(
|
||||||
|
"JSONArray[" + idx + "] is not a " + valueType + " (" + value.getClass() + " : " + value + ")."
|
||||||
, cause);
|
, cause);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2008 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -47,7 +27,32 @@ public class JSONML {
|
|||||||
XMLTokener x,
|
XMLTokener x,
|
||||||
boolean arrayForm,
|
boolean arrayForm,
|
||||||
JSONArray ja,
|
JSONArray ja,
|
||||||
boolean keepStrings
|
boolean keepStrings,
|
||||||
|
int currentNestingDepth
|
||||||
|
) throws JSONException {
|
||||||
|
return parse(x,arrayForm, ja,
|
||||||
|
keepStrings ? JSONMLParserConfiguration.KEEP_STRINGS : JSONMLParserConfiguration.ORIGINAL,
|
||||||
|
currentNestingDepth);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse XML values and store them in a JSONArray.
|
||||||
|
* @param x The XMLTokener containing the source string.
|
||||||
|
* @param arrayForm true if array form, false if object form.
|
||||||
|
* @param ja The JSONArray that is containing the current tag or null
|
||||||
|
* if we are at the outermost level.
|
||||||
|
* @param config The parser configuration:
|
||||||
|
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||||
|
* JSONMLParserConfiguration.KEEP_STRINGS means Don't type-convert text nodes and attribute values.
|
||||||
|
* @return A JSONArray if the value is the outermost tag, otherwise null.
|
||||||
|
* @throws JSONException if a parsing error occurs
|
||||||
|
*/
|
||||||
|
private static Object parse(
|
||||||
|
XMLTokener x,
|
||||||
|
boolean arrayForm,
|
||||||
|
JSONArray ja,
|
||||||
|
JSONMLParserConfiguration config,
|
||||||
|
int currentNestingDepth
|
||||||
) throws JSONException {
|
) throws JSONException {
|
||||||
String attribute;
|
String attribute;
|
||||||
char c;
|
char c;
|
||||||
@@ -172,7 +177,7 @@ public class JSONML {
|
|||||||
if (!(token instanceof String)) {
|
if (!(token instanceof String)) {
|
||||||
throw x.syntaxError("Missing value");
|
throw x.syntaxError("Missing value");
|
||||||
}
|
}
|
||||||
newjo.accumulate(attribute, keepStrings ? ((String)token) :XML.stringToValue((String)token));
|
newjo.accumulate(attribute, config.isKeepStrings() ? ((String)token) :XML.stringToValue((String)token));
|
||||||
token = null;
|
token = null;
|
||||||
} else {
|
} else {
|
||||||
newjo.accumulate(attribute, "");
|
newjo.accumulate(attribute, "");
|
||||||
@@ -201,7 +206,12 @@ public class JSONML {
|
|||||||
if (token != XML.GT) {
|
if (token != XML.GT) {
|
||||||
throw x.syntaxError("Misshaped tag");
|
throw x.syntaxError("Misshaped tag");
|
||||||
}
|
}
|
||||||
closeTag = (String)parse(x, arrayForm, newja, keepStrings);
|
|
||||||
|
if (currentNestingDepth == config.getMaxNestingDepth()) {
|
||||||
|
throw x.syntaxError("Maximum nesting depth of " + config.getMaxNestingDepth() + " reached");
|
||||||
|
}
|
||||||
|
|
||||||
|
closeTag = (String)parse(x, arrayForm, newja, config, currentNestingDepth + 1);
|
||||||
if (closeTag != null) {
|
if (closeTag != null) {
|
||||||
if (!closeTag.equals(tagName)) {
|
if (!closeTag.equals(tagName)) {
|
||||||
throw x.syntaxError("Mismatched '" + tagName +
|
throw x.syntaxError("Mismatched '" + tagName +
|
||||||
@@ -223,7 +233,7 @@ public class JSONML {
|
|||||||
} else {
|
} else {
|
||||||
if (ja != null) {
|
if (ja != null) {
|
||||||
ja.put(token instanceof String
|
ja.put(token instanceof String
|
||||||
? keepStrings ? XML.unescape((String)token) :XML.stringToValue((String)token)
|
? (config.isKeepStrings() ? XML.unescape((String)token) : XML.stringToValue((String)token))
|
||||||
: token);
|
: token);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -244,7 +254,7 @@ public class JSONML {
|
|||||||
* @throws JSONException Thrown on error converting to a JSONArray
|
* @throws JSONException Thrown on error converting to a JSONArray
|
||||||
*/
|
*/
|
||||||
public static JSONArray toJSONArray(String string) throws JSONException {
|
public static JSONArray toJSONArray(String string) throws JSONException {
|
||||||
return (JSONArray)parse(new XMLTokener(string), true, null, false);
|
return (JSONArray)parse(new XMLTokener(string), true, null, JSONMLParserConfiguration.ORIGINAL, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -266,7 +276,56 @@ public class JSONML {
|
|||||||
* @throws JSONException Thrown on error converting to a JSONArray
|
* @throws JSONException Thrown on error converting to a JSONArray
|
||||||
*/
|
*/
|
||||||
public static JSONArray toJSONArray(String string, boolean keepStrings) throws JSONException {
|
public static JSONArray toJSONArray(String string, boolean keepStrings) throws JSONException {
|
||||||
return (JSONArray)parse(new XMLTokener(string), true, null, keepStrings);
|
return (JSONArray)parse(new XMLTokener(string), true, null, keepStrings, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONArray using the JsonML transform. Each XML tag is represented as
|
||||||
|
* a JSONArray in which the first element is the tag name. If the tag has
|
||||||
|
* attributes, then the second element will be JSONObject containing the
|
||||||
|
* name/value pairs. If the tag contains children, then strings and
|
||||||
|
* JSONArrays will represent the child tags.
|
||||||
|
* As opposed to toJSONArray this method does not attempt to convert
|
||||||
|
* any text node or attribute value to any type
|
||||||
|
* but just leaves it as a string.
|
||||||
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
|
* @param string The source string.
|
||||||
|
* @param config The parser configuration:
|
||||||
|
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||||
|
* JSONMLParserConfiguration.KEEP_STRINGS means values will not be coerced into boolean
|
||||||
|
* or numeric values and will instead be left as strings
|
||||||
|
* @return A JSONArray containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown on error converting to a JSONArray
|
||||||
|
*/
|
||||||
|
public static JSONArray toJSONArray(String string, JSONMLParserConfiguration config) throws JSONException {
|
||||||
|
return (JSONArray)parse(new XMLTokener(string), true, null, config, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONArray using the JsonML transform. Each XML tag is represented as
|
||||||
|
* a JSONArray in which the first element is the tag name. If the tag has
|
||||||
|
* attributes, then the second element will be JSONObject containing the
|
||||||
|
* name/value pairs. If the tag contains children, then strings and
|
||||||
|
* JSONArrays will represent the child content and tags.
|
||||||
|
* As opposed to toJSONArray this method does not attempt to convert
|
||||||
|
* any text node or attribute value to any type
|
||||||
|
* but just leaves it as a string.
|
||||||
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
|
* @param x An XMLTokener.
|
||||||
|
* @param config The parser configuration:
|
||||||
|
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||||
|
* JSONMLParserConfiguration.KEEP_STRINGS means values will not be coerced into boolean
|
||||||
|
* or numeric values and will instead be left as strings
|
||||||
|
* @return A JSONArray containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown on error converting to a JSONArray
|
||||||
|
*/
|
||||||
|
public static JSONArray toJSONArray(XMLTokener x, JSONMLParserConfiguration config) throws JSONException {
|
||||||
|
return (JSONArray)parse(x, true, null, config, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -288,7 +347,7 @@ public class JSONML {
|
|||||||
* @throws JSONException Thrown on error converting to a JSONArray
|
* @throws JSONException Thrown on error converting to a JSONArray
|
||||||
*/
|
*/
|
||||||
public static JSONArray toJSONArray(XMLTokener x, boolean keepStrings) throws JSONException {
|
public static JSONArray toJSONArray(XMLTokener x, boolean keepStrings) throws JSONException {
|
||||||
return (JSONArray)parse(x, true, null, keepStrings);
|
return (JSONArray)parse(x, true, null, keepStrings, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -305,7 +364,7 @@ public class JSONML {
|
|||||||
* @throws JSONException Thrown on error converting to a JSONArray
|
* @throws JSONException Thrown on error converting to a JSONArray
|
||||||
*/
|
*/
|
||||||
public static JSONArray toJSONArray(XMLTokener x) throws JSONException {
|
public static JSONArray toJSONArray(XMLTokener x) throws JSONException {
|
||||||
return (JSONArray)parse(x, true, null, false);
|
return (JSONArray)parse(x, true, null, false, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -323,7 +382,7 @@ public class JSONML {
|
|||||||
* @throws JSONException Thrown on error converting to a JSONObject
|
* @throws JSONException Thrown on error converting to a JSONObject
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(String string) throws JSONException {
|
public static JSONObject toJSONObject(String string) throws JSONException {
|
||||||
return (JSONObject)parse(new XMLTokener(string), false, null, false);
|
return (JSONObject)parse(new XMLTokener(string), false, null, false, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -343,7 +402,29 @@ public class JSONML {
|
|||||||
* @throws JSONException Thrown on error converting to a JSONObject
|
* @throws JSONException Thrown on error converting to a JSONObject
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(String string, boolean keepStrings) throws JSONException {
|
public static JSONObject toJSONObject(String string, boolean keepStrings) throws JSONException {
|
||||||
return (JSONObject)parse(new XMLTokener(string), false, null, keepStrings);
|
return (JSONObject)parse(new XMLTokener(string), false, null, keepStrings, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONObject using the JsonML transform. Each XML tag is represented as
|
||||||
|
* a JSONObject with a "tagName" property. If the tag has attributes, then
|
||||||
|
* the attributes will be in the JSONObject as properties. If the tag
|
||||||
|
* contains children, the object will have a "childNodes" property which
|
||||||
|
* will be an array of strings and JsonML JSONObjects.
|
||||||
|
|
||||||
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
|
* @param string The XML source text.
|
||||||
|
* @param config The parser configuration:
|
||||||
|
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||||
|
* JSONMLParserConfiguration.KEEP_STRINGS means values will not be coerced into boolean
|
||||||
|
* or numeric values and will instead be left as strings
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown on error converting to a JSONObject
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(String string, JSONMLParserConfiguration config) throws JSONException {
|
||||||
|
return (JSONObject)parse(new XMLTokener(string), false, null, config, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -361,7 +442,7 @@ public class JSONML {
|
|||||||
* @throws JSONException Thrown on error converting to a JSONObject
|
* @throws JSONException Thrown on error converting to a JSONObject
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(XMLTokener x) throws JSONException {
|
public static JSONObject toJSONObject(XMLTokener x) throws JSONException {
|
||||||
return (JSONObject)parse(x, false, null, false);
|
return (JSONObject)parse(x, false, null, false, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -381,7 +462,29 @@ public class JSONML {
|
|||||||
* @throws JSONException Thrown on error converting to a JSONObject
|
* @throws JSONException Thrown on error converting to a JSONObject
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(XMLTokener x, boolean keepStrings) throws JSONException {
|
public static JSONObject toJSONObject(XMLTokener x, boolean keepStrings) throws JSONException {
|
||||||
return (JSONObject)parse(x, false, null, keepStrings);
|
return (JSONObject)parse(x, false, null, keepStrings, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONObject using the JsonML transform. Each XML tag is represented as
|
||||||
|
* a JSONObject with a "tagName" property. If the tag has attributes, then
|
||||||
|
* the attributes will be in the JSONObject as properties. If the tag
|
||||||
|
* contains children, the object will have a "childNodes" property which
|
||||||
|
* will be an array of strings and JsonML JSONObjects.
|
||||||
|
|
||||||
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
|
* @param x An XMLTokener of the XML source text.
|
||||||
|
* @param config The parser configuration:
|
||||||
|
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||||
|
* JSONMLParserConfiguration.KEEP_STRINGS means values will not be coerced into boolean
|
||||||
|
* or numeric values and will instead be left as strings
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown on error converting to a JSONObject
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(XMLTokener x, JSONMLParserConfiguration config) throws JSONException {
|
||||||
|
return (JSONObject)parse(x, false, null, config, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -462,6 +565,7 @@ public class JSONML {
|
|||||||
return sb.toString();
|
return sb.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reverse the JSONML transformation, making an XML text from a JSONObject.
|
* Reverse the JSONML transformation, making an XML text from a JSONObject.
|
||||||
* The JSONObject must contain a "tagName" property. If it has children,
|
* The JSONObject must contain a "tagName" property. If it has children,
|
||||||
|
|||||||
67
src/main/java/org/json/JSONMLParserConfiguration.java
Normal file
67
src/main/java/org/json/JSONMLParserConfiguration.java
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
package org.json;
|
||||||
|
/*
|
||||||
|
Public Domain.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configuration object for the XML to JSONML parser. The configuration is immutable.
|
||||||
|
*/
|
||||||
|
@SuppressWarnings({""})
|
||||||
|
public class JSONMLParserConfiguration extends ParserConfiguration {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We can override the default maximum nesting depth if needed.
|
||||||
|
*/
|
||||||
|
public static final int DEFAULT_MAXIMUM_NESTING_DEPTH = ParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH;
|
||||||
|
|
||||||
|
/** Original Configuration of the XML to JSONML Parser. */
|
||||||
|
public static final JSONMLParserConfiguration ORIGINAL
|
||||||
|
= new JSONMLParserConfiguration();
|
||||||
|
/** Original configuration of the XML to JSONML Parser except that values are kept as strings. */
|
||||||
|
public static final JSONMLParserConfiguration KEEP_STRINGS
|
||||||
|
= new JSONMLParserConfiguration().withKeepStrings(true);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Default parser configuration. Does not keep strings (tries to implicitly convert values).
|
||||||
|
*/
|
||||||
|
public JSONMLParserConfiguration() {
|
||||||
|
super();
|
||||||
|
this.maxNestingDepth = DEFAULT_MAXIMUM_NESTING_DEPTH;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configure the parser string processing and use the default CDATA Tag Name as "content".
|
||||||
|
* @param keepStrings <code>true</code> to parse all values as string.
|
||||||
|
* <code>false</code> to try and convert XML string values into a JSON value.
|
||||||
|
* @param maxNestingDepth <code>int</code> to limit the nesting depth
|
||||||
|
*/
|
||||||
|
protected JSONMLParserConfiguration(final boolean keepStrings, final int maxNestingDepth) {
|
||||||
|
super(keepStrings, maxNestingDepth);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides a new instance of the same configuration.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected JSONMLParserConfiguration clone() {
|
||||||
|
// future modifications to this method should always ensure a "deep"
|
||||||
|
// clone in the case of collections. i.e. if a Map is added as a configuration
|
||||||
|
// item, a new map instance should be created and if possible each value in the
|
||||||
|
// map should be cloned as well. If the values of the map are known to also
|
||||||
|
// be immutable, then a shallow clone of the map is acceptable.
|
||||||
|
return new JSONMLParserConfiguration(
|
||||||
|
this.keepStrings,
|
||||||
|
this.maxNestingDepth
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public JSONMLParserConfiguration withKeepStrings(final boolean newVal) {
|
||||||
|
return super.withKeepStrings(newVal);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public JSONMLParserConfiguration withMaxNestingDepth(int maxNestingDepth) {
|
||||||
|
return super.withMaxNestingDepth(maxNestingDepth);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,31 +1,10 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
import java.io.Closeable;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
*/
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
|
import java.io.Closeable;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.StringWriter;
|
import java.io.StringWriter;
|
||||||
import java.io.Writer;
|
import java.io.Writer;
|
||||||
@@ -37,8 +16,10 @@ import java.lang.reflect.Modifier;
|
|||||||
import java.math.BigDecimal;
|
import java.math.BigDecimal;
|
||||||
import java.math.BigInteger;
|
import java.math.BigInteger;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
|
import java.util.Collections;
|
||||||
import java.util.Enumeration;
|
import java.util.Enumeration;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.IdentityHashMap;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
@@ -128,6 +109,7 @@ public class JSONObject {
|
|||||||
* null.
|
* null.
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
|
@SuppressWarnings("lgtm[java/unchecked-cast-in-equals]")
|
||||||
public boolean equals(Object object) {
|
public boolean equals(Object object) {
|
||||||
return object == null || object == this;
|
return object == null || object == this;
|
||||||
}
|
}
|
||||||
@@ -163,6 +145,10 @@ public class JSONObject {
|
|||||||
*/
|
*/
|
||||||
private final Map<String, Object> map;
|
private final Map<String, Object> map;
|
||||||
|
|
||||||
|
public Class<? extends Map> getMapType() {
|
||||||
|
return map.getClass();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* It is sometimes more convenient and less ambiguous to have a
|
* It is sometimes more convenient and less ambiguous to have a
|
||||||
* <code>NULL</code> object than to use Java's <code>null</code> value.
|
* <code>NULL</code> object than to use Java's <code>null</code> value.
|
||||||
@@ -222,12 +208,19 @@ public class JSONObject {
|
|||||||
throw x.syntaxError("A JSONObject text must begin with '{'");
|
throw x.syntaxError("A JSONObject text must begin with '{'");
|
||||||
}
|
}
|
||||||
for (;;) {
|
for (;;) {
|
||||||
|
char prev = x.getPrevious();
|
||||||
c = x.nextClean();
|
c = x.nextClean();
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case 0:
|
case 0:
|
||||||
throw x.syntaxError("A JSONObject text must end with '}'");
|
throw x.syntaxError("A JSONObject text must end with '}'");
|
||||||
case '}':
|
case '}':
|
||||||
return;
|
return;
|
||||||
|
case '{':
|
||||||
|
case '[':
|
||||||
|
if(prev=='{') {
|
||||||
|
throw x.syntaxError("A JSON Object can not directly nest another JSON Object or JSON Array.");
|
||||||
|
}
|
||||||
|
// fall through
|
||||||
default:
|
default:
|
||||||
x.back();
|
x.back();
|
||||||
key = x.nextValue().toString();
|
key = x.nextValue().toString();
|
||||||
@@ -364,6 +357,11 @@ public class JSONObject {
|
|||||||
this.populateMap(bean);
|
this.populateMap(bean);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private JSONObject(Object bean, Set<Object> objectsRecord) {
|
||||||
|
this();
|
||||||
|
this.populateMap(bean, objectsRecord);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a JSONObject from an Object, using reflection to find the
|
* Construct a JSONObject from an Object, using reflection to find the
|
||||||
* public members. The resulting JSONObject's keys will be the strings from
|
* public members. The resulting JSONObject's keys will be the strings from
|
||||||
@@ -594,7 +592,7 @@ public class JSONObject {
|
|||||||
// JSONException should really take a throwable argument.
|
// JSONException should really take a throwable argument.
|
||||||
// If it did, I would re-implement this with the Enum.valueOf
|
// If it did, I would re-implement this with the Enum.valueOf
|
||||||
// method and place any thrown exception in the JSONException
|
// method and place any thrown exception in the JSONException
|
||||||
throw wrongValueFormatException(key, "enum of type " + quote(clazz.getSimpleName()), null);
|
throw wrongValueFormatException(key, "enum of type " + quote(clazz.getSimpleName()), opt(key), null);
|
||||||
}
|
}
|
||||||
return val;
|
return val;
|
||||||
}
|
}
|
||||||
@@ -620,7 +618,7 @@ public class JSONObject {
|
|||||||
.equalsIgnoreCase("true"))) {
|
.equalsIgnoreCase("true"))) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
throw wrongValueFormatException(key, "Boolean", null);
|
throw wrongValueFormatException(key, "Boolean", object, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -644,7 +642,7 @@ public class JSONObject {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the BigDecimal value associated with a key. If the value is float or
|
* Get the BigDecimal value associated with a key. If the value is float or
|
||||||
* double, the the {@link BigDecimal#BigDecimal(double)} constructor will
|
* double, the {@link BigDecimal#BigDecimal(double)} constructor will
|
||||||
* be used. See notes on the constructor for conversion issues that may
|
* be used. See notes on the constructor for conversion issues that may
|
||||||
* arise.
|
* arise.
|
||||||
*
|
*
|
||||||
@@ -682,7 +680,7 @@ public class JSONObject {
|
|||||||
try {
|
try {
|
||||||
return Double.parseDouble(object.toString());
|
return Double.parseDouble(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(key, "double", e);
|
throw wrongValueFormatException(key, "double", object, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -704,7 +702,7 @@ public class JSONObject {
|
|||||||
try {
|
try {
|
||||||
return Float.parseFloat(object.toString());
|
return Float.parseFloat(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(key, "float", e);
|
throw wrongValueFormatException(key, "float", object, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -726,7 +724,7 @@ public class JSONObject {
|
|||||||
}
|
}
|
||||||
return stringToNumber(object.toString());
|
return stringToNumber(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(key, "number", e);
|
throw wrongValueFormatException(key, "number", object, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -748,7 +746,7 @@ public class JSONObject {
|
|||||||
try {
|
try {
|
||||||
return Integer.parseInt(object.toString());
|
return Integer.parseInt(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(key, "int", e);
|
throw wrongValueFormatException(key, "int", object, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -766,7 +764,7 @@ public class JSONObject {
|
|||||||
if (object instanceof JSONArray) {
|
if (object instanceof JSONArray) {
|
||||||
return (JSONArray) object;
|
return (JSONArray) object;
|
||||||
}
|
}
|
||||||
throw wrongValueFormatException(key, "JSONArray", null);
|
throw wrongValueFormatException(key, "JSONArray", object, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -783,7 +781,7 @@ public class JSONObject {
|
|||||||
if (object instanceof JSONObject) {
|
if (object instanceof JSONObject) {
|
||||||
return (JSONObject) object;
|
return (JSONObject) object;
|
||||||
}
|
}
|
||||||
throw wrongValueFormatException(key, "JSONObject", null);
|
throw wrongValueFormatException(key, "JSONObject", object, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -804,7 +802,7 @@ public class JSONObject {
|
|||||||
try {
|
try {
|
||||||
return Long.parseLong(object.toString());
|
return Long.parseLong(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(key, "long", e);
|
throw wrongValueFormatException(key, "long", object, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -860,7 +858,7 @@ public class JSONObject {
|
|||||||
if (object instanceof String) {
|
if (object instanceof String) {
|
||||||
return (String) object;
|
return (String) object;
|
||||||
}
|
}
|
||||||
throw wrongValueFormatException(key, "string", null);
|
throw wrongValueFormatException(key, "string", object, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1159,6 +1157,18 @@ public class JSONObject {
|
|||||||
* to convert.
|
* to convert.
|
||||||
*/
|
*/
|
||||||
static BigDecimal objectToBigDecimal(Object val, BigDecimal defaultValue) {
|
static BigDecimal objectToBigDecimal(Object val, BigDecimal defaultValue) {
|
||||||
|
return objectToBigDecimal(val, defaultValue, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param val value to convert
|
||||||
|
* @param defaultValue default value to return is the conversion doesn't work or is null.
|
||||||
|
* @param exact When <code>true</code>, then {@link Double} and {@link Float} values will be converted exactly.
|
||||||
|
* When <code>false</code>, they will be converted to {@link String} values before converting to {@link BigDecimal}.
|
||||||
|
* @return BigDecimal conversion of the original value, or the defaultValue if unable
|
||||||
|
* to convert.
|
||||||
|
*/
|
||||||
|
static BigDecimal objectToBigDecimal(Object val, BigDecimal defaultValue, boolean exact) {
|
||||||
if (NULL.equals(val)) {
|
if (NULL.equals(val)) {
|
||||||
return defaultValue;
|
return defaultValue;
|
||||||
}
|
}
|
||||||
@@ -1172,7 +1182,13 @@ public class JSONObject {
|
|||||||
if (!numberIsFinite((Number)val)) {
|
if (!numberIsFinite((Number)val)) {
|
||||||
return defaultValue;
|
return defaultValue;
|
||||||
}
|
}
|
||||||
return new BigDecimal(((Number) val).doubleValue());
|
if (exact) {
|
||||||
|
return new BigDecimal(((Number)val).doubleValue());
|
||||||
|
}
|
||||||
|
// use the string constructor so that we maintain "nice" values for doubles and floats
|
||||||
|
// the double constructor will translate doubles to "exact" values instead of the likely
|
||||||
|
// intended representation
|
||||||
|
return new BigDecimal(val.toString());
|
||||||
}
|
}
|
||||||
if (val instanceof Long || val instanceof Integer
|
if (val instanceof Long || val instanceof Integer
|
||||||
|| val instanceof Short || val instanceof Byte){
|
|| val instanceof Short || val instanceof Byte){
|
||||||
@@ -1274,11 +1290,7 @@ public class JSONObject {
|
|||||||
if (val == null) {
|
if (val == null) {
|
||||||
return defaultValue;
|
return defaultValue;
|
||||||
}
|
}
|
||||||
final double doubleValue = val.doubleValue();
|
return val.doubleValue();
|
||||||
// if (Double.isNaN(doubleValue) || Double.isInfinite(doubleValue)) {
|
|
||||||
// return defaultValue;
|
|
||||||
// }
|
|
||||||
return doubleValue;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1370,9 +1382,21 @@ public class JSONObject {
|
|||||||
* A key string.
|
* A key string.
|
||||||
* @return A JSONObject which is the value.
|
* @return A JSONObject which is the value.
|
||||||
*/
|
*/
|
||||||
public JSONObject optJSONObject(String key) {
|
public JSONObject optJSONObject(String key) { return this.optJSONObject(key, null); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an optional JSONObject associated with a key, or the default if there
|
||||||
|
* is no such key or if the value is not a JSONObject.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default.
|
||||||
|
* @return An JSONObject which is the value.
|
||||||
|
*/
|
||||||
|
public JSONObject optJSONObject(String key, JSONObject defaultValue) {
|
||||||
Object object = this.opt(key);
|
Object object = this.opt(key);
|
||||||
return object instanceof JSONObject ? (JSONObject) object : null;
|
return object instanceof JSONObject ? (JSONObject) object : defaultValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1488,6 +1512,10 @@ public class JSONObject {
|
|||||||
* the bean
|
* the bean
|
||||||
*/
|
*/
|
||||||
private void populateMap(Object bean) {
|
private void populateMap(Object bean) {
|
||||||
|
populateMap(bean, Collections.newSetFromMap(new IdentityHashMap<Object, Boolean>()));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void populateMap(Object bean, Set<Object> objectsRecord) {
|
||||||
Class<?> klass = bean.getClass();
|
Class<?> klass = bean.getClass();
|
||||||
|
|
||||||
// If klass is a System class then set includeSuperClass to false.
|
// If klass is a System class then set includeSuperClass to false.
|
||||||
@@ -1508,7 +1536,19 @@ public class JSONObject {
|
|||||||
try {
|
try {
|
||||||
final Object result = method.invoke(bean);
|
final Object result = method.invoke(bean);
|
||||||
if (result != null) {
|
if (result != null) {
|
||||||
this.map.put(key, wrap(result));
|
// check cyclic dependency and throw error if needed
|
||||||
|
// the wrap and populateMap combination method is
|
||||||
|
// itself DFS recursive
|
||||||
|
if (objectsRecord.contains(result)) {
|
||||||
|
throw recursivelyDefinedObjectException(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
objectsRecord.add(result);
|
||||||
|
|
||||||
|
this.map.put(key, wrap(result, objectsRecord));
|
||||||
|
|
||||||
|
objectsRecord.remove(result);
|
||||||
|
|
||||||
// we don't use the result anywhere outside of wrap
|
// we don't use the result anywhere outside of wrap
|
||||||
// if it's a resource we should be sure to close it
|
// if it's a resource we should be sure to close it
|
||||||
// after calling toString
|
// after calling toString
|
||||||
@@ -1581,7 +1621,7 @@ public class JSONObject {
|
|||||||
* @param annotationClass
|
* @param annotationClass
|
||||||
* annotation to look for
|
* annotation to look for
|
||||||
* @return the {@link Annotation} if the annotation exists on the current method
|
* @return the {@link Annotation} if the annotation exists on the current method
|
||||||
* or one of it's super class definitions
|
* or one of its super class definitions
|
||||||
*/
|
*/
|
||||||
private static <A extends Annotation> A getAnnotation(final Method m, final Class<A> annotationClass) {
|
private static <A extends Annotation> A getAnnotation(final Method m, final Class<A> annotationClass) {
|
||||||
// if we have invalid data the result is null
|
// if we have invalid data the result is null
|
||||||
@@ -1627,9 +1667,6 @@ public class JSONObject {
|
|||||||
* implementations and interfaces has the annotation. Returns the depth of the
|
* implementations and interfaces has the annotation. Returns the depth of the
|
||||||
* annotation in the hierarchy.
|
* annotation in the hierarchy.
|
||||||
*
|
*
|
||||||
* @param <A>
|
|
||||||
* type of the annotation
|
|
||||||
*
|
|
||||||
* @param m
|
* @param m
|
||||||
* method to check
|
* method to check
|
||||||
* @param annotationClass
|
* @param annotationClass
|
||||||
@@ -1962,6 +1999,7 @@ public class JSONObject {
|
|||||||
* A String
|
* A String
|
||||||
* @return A String correctly formatted for insertion in a JSON text.
|
* @return A String correctly formatted for insertion in a JSON text.
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings("resource")
|
||||||
public static String quote(String string) {
|
public static String quote(String string) {
|
||||||
StringWriter sw = new StringWriter();
|
StringWriter sw = new StringWriter();
|
||||||
synchronized (sw.getBuffer()) {
|
synchronized (sw.getBuffer()) {
|
||||||
@@ -2080,7 +2118,13 @@ public class JSONObject {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
} else if (valueThis instanceof Number && valueOther instanceof Number) {
|
} else if (valueThis instanceof Number && valueOther instanceof Number) {
|
||||||
return isNumberSimilar((Number)valueThis, (Number)valueOther);
|
if (!isNumberSimilar((Number)valueThis, (Number)valueOther)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else if (valueThis instanceof JSONString && valueOther instanceof JSONString) {
|
||||||
|
if (!((JSONString) valueThis).toJSONString().equals(((JSONString) valueOther).toJSONString())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
} else if (!valueThis.equals(valueOther)) {
|
} else if (!valueThis.equals(valueOther)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -2123,8 +2167,8 @@ public class JSONObject {
|
|||||||
// BigDecimal should be able to handle all of our number types that we support through
|
// BigDecimal should be able to handle all of our number types that we support through
|
||||||
// documentation. Convert to BigDecimal first, then use the Compare method to
|
// documentation. Convert to BigDecimal first, then use the Compare method to
|
||||||
// decide equality.
|
// decide equality.
|
||||||
final BigDecimal lBigDecimal = objectToBigDecimal(l, null);
|
final BigDecimal lBigDecimal = objectToBigDecimal(l, null, false);
|
||||||
final BigDecimal rBigDecimal = objectToBigDecimal(r, null);
|
final BigDecimal rBigDecimal = objectToBigDecimal(r, null, false);
|
||||||
if (lBigDecimal == null || rBigDecimal == null) {
|
if (lBigDecimal == null || rBigDecimal == null) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -2205,7 +2249,7 @@ public class JSONObject {
|
|||||||
// This will narrow any values to the smallest reasonable Object representation
|
// This will narrow any values to the smallest reasonable Object representation
|
||||||
// (Integer, Long, or BigInteger)
|
// (Integer, Long, or BigInteger)
|
||||||
|
|
||||||
// BigInteger down conversion: We use a similar bitLenth compare as
|
// BigInteger down conversion: We use a similar bitLength compare as
|
||||||
// BigInteger#intValueExact uses. Increases GC, but objects hold
|
// BigInteger#intValueExact uses. Increases GC, but objects hold
|
||||||
// only what they need. i.e. Less runtime overhead if the value is
|
// only what they need. i.e. Less runtime overhead if the value is
|
||||||
// long lived.
|
// long lived.
|
||||||
@@ -2348,6 +2392,7 @@ public class JSONObject {
|
|||||||
* @throws JSONException
|
* @throws JSONException
|
||||||
* If the object contains an invalid number.
|
* If the object contains an invalid number.
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings("resource")
|
||||||
public String toString(int indentFactor) throws JSONException {
|
public String toString(int indentFactor) throws JSONException {
|
||||||
StringWriter w = new StringWriter();
|
StringWriter w = new StringWriter();
|
||||||
synchronized (w.getBuffer()) {
|
synchronized (w.getBuffer()) {
|
||||||
@@ -2400,6 +2445,10 @@ public class JSONObject {
|
|||||||
* @return The wrapped value
|
* @return The wrapped value
|
||||||
*/
|
*/
|
||||||
public static Object wrap(Object object) {
|
public static Object wrap(Object object) {
|
||||||
|
return wrap(object, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Object wrap(Object object, Set<Object> objectsRecord) {
|
||||||
try {
|
try {
|
||||||
if (NULL.equals(object)) {
|
if (NULL.equals(object)) {
|
||||||
return NULL;
|
return NULL;
|
||||||
@@ -2434,7 +2483,13 @@ public class JSONObject {
|
|||||||
|| object.getClass().getClassLoader() == null) {
|
|| object.getClass().getClassLoader() == null) {
|
||||||
return object.toString();
|
return object.toString();
|
||||||
}
|
}
|
||||||
|
if (objectsRecord != null) {
|
||||||
|
return new JSONObject(object, objectsRecord);
|
||||||
|
}
|
||||||
return new JSONObject(object);
|
return new JSONObject(object);
|
||||||
|
}
|
||||||
|
catch (JSONException exception) {
|
||||||
|
throw exception;
|
||||||
} catch (Exception exception) {
|
} catch (Exception exception) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -2454,6 +2509,7 @@ public class JSONObject {
|
|||||||
return this.write(writer, 0, 0);
|
return this.write(writer, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("resource")
|
||||||
static final Writer writeValue(Writer writer, Object value,
|
static final Writer writeValue(Writer writer, Object value,
|
||||||
int indentFactor, int indent) throws JSONException, IOException {
|
int indentFactor, int indent) throws JSONException, IOException {
|
||||||
if (value == null || value.equals(null)) {
|
if (value == null || value.equals(null)) {
|
||||||
@@ -2531,6 +2587,7 @@ public class JSONObject {
|
|||||||
* @throws JSONException if a called function has an error or a write error
|
* @throws JSONException if a called function has an error or a write error
|
||||||
* occurs
|
* occurs
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings("resource")
|
||||||
public Writer write(Writer writer, int indentFactor, int indent)
|
public Writer write(Writer writer, int indentFactor, int indent)
|
||||||
throws JSONException {
|
throws JSONException {
|
||||||
try {
|
try {
|
||||||
@@ -2623,26 +2680,33 @@ public class JSONObject {
|
|||||||
private static JSONException wrongValueFormatException(
|
private static JSONException wrongValueFormatException(
|
||||||
String key,
|
String key,
|
||||||
String valueType,
|
String valueType,
|
||||||
|
Object value,
|
||||||
Throwable cause) {
|
Throwable cause) {
|
||||||
|
if(value == null) {
|
||||||
|
|
||||||
return new JSONException(
|
return new JSONException(
|
||||||
"JSONObject[" + quote(key) + "] is not a " + valueType + "."
|
"JSONObject[" + quote(key) + "] is not a " + valueType + " (null)."
|
||||||
|
, cause);
|
||||||
|
}
|
||||||
|
// don't try to toString collections or known object types that could be large.
|
||||||
|
if(value instanceof Map || value instanceof Iterable || value instanceof JSONObject) {
|
||||||
|
return new JSONException(
|
||||||
|
"JSONObject[" + quote(key) + "] is not a " + valueType + " (" + value.getClass() + ")."
|
||||||
|
, cause);
|
||||||
|
}
|
||||||
|
return new JSONException(
|
||||||
|
"JSONObject[" + quote(key) + "] is not a " + valueType + " (" + value.getClass() + " : " + value + ")."
|
||||||
, cause);
|
, cause);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new JSONException in a common format for incorrect conversions.
|
* Create a new JSONException in a common format for recursive object definition.
|
||||||
* @param key name of the key
|
* @param key name of the key
|
||||||
* @param valueType the type of value being coerced to
|
|
||||||
* @param cause optional cause of the coercion failure
|
|
||||||
* @return JSONException that can be thrown.
|
* @return JSONException that can be thrown.
|
||||||
*/
|
*/
|
||||||
private static JSONException wrongValueFormatException(
|
private static JSONException recursivelyDefinedObjectException(String key) {
|
||||||
String key,
|
|
||||||
String valueType,
|
|
||||||
Object value,
|
|
||||||
Throwable cause) {
|
|
||||||
return new JSONException(
|
return new JSONException(
|
||||||
"JSONObject[" + quote(key) + "] is not a " + valueType + " (" + value + ")."
|
"JavaBean object contains recursively defined member variable of key " + quote(key)
|
||||||
, cause);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,27 +10,7 @@ import java.util.Collections;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -188,7 +168,7 @@ public class JSONPointer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @see https://tools.ietf.org/html/rfc6901#section-3
|
* @see <a href="https://tools.ietf.org/html/rfc6901#section-3">rfc6901 section 3</a>
|
||||||
*/
|
*/
|
||||||
private static String unescape(String token) {
|
private static String unescape(String token) {
|
||||||
return token.replace("~1", "/").replace("~0", "~");
|
return token.replace("~1", "/").replace("~0", "~");
|
||||||
@@ -268,7 +248,7 @@ public class JSONPointer {
|
|||||||
* @param token the JSONPointer segment value to be escaped
|
* @param token the JSONPointer segment value to be escaped
|
||||||
* @return the escaped value for the token
|
* @return the escaped value for the token
|
||||||
*
|
*
|
||||||
* @see https://tools.ietf.org/html/rfc6901#section-3
|
* @see <a href="https://tools.ietf.org/html/rfc6901#section-3">rfc6901 section 3</a>
|
||||||
*/
|
*/
|
||||||
private static String escape(String token) {
|
private static String escape(String token) {
|
||||||
return token.replace("~", "~0")
|
return token.replace("~", "~0")
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2018 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static java.lang.annotation.ElementType.METHOD;
|
import static java.lang.annotation.ElementType.METHOD;
|
||||||
@@ -31,13 +11,13 @@ import java.lang.annotation.Documented;
|
|||||||
import java.lang.annotation.Retention;
|
import java.lang.annotation.Retention;
|
||||||
import java.lang.annotation.Target;
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
@Documented
|
|
||||||
@Retention(RUNTIME)
|
|
||||||
@Target({METHOD})
|
|
||||||
/**
|
/**
|
||||||
* Use this annotation on a getter method to override the Bean name
|
* Use this annotation on a getter method to override the Bean name
|
||||||
* parser for Bean -> JSONObject mapping. If this annotation is
|
* parser for Bean -> JSONObject mapping. If this annotation is
|
||||||
* present at any level in the class hierarchy, then the method will
|
* present at any level in the class hierarchy, then the method will
|
||||||
* not be serialized from the bean into the JSONObject.
|
* not be serialized from the bean into the JSONObject.
|
||||||
*/
|
*/
|
||||||
|
@Documented
|
||||||
|
@Retention(RUNTIME)
|
||||||
|
@Target({METHOD})
|
||||||
public @interface JSONPropertyIgnore { }
|
public @interface JSONPropertyIgnore { }
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2018 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static java.lang.annotation.ElementType.METHOD;
|
import static java.lang.annotation.ElementType.METHOD;
|
||||||
@@ -31,14 +11,14 @@ import java.lang.annotation.Documented;
|
|||||||
import java.lang.annotation.Retention;
|
import java.lang.annotation.Retention;
|
||||||
import java.lang.annotation.Target;
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
@Documented
|
|
||||||
@Retention(RUNTIME)
|
|
||||||
@Target({METHOD})
|
|
||||||
/**
|
/**
|
||||||
* Use this annotation on a getter method to override the Bean name
|
* Use this annotation on a getter method to override the Bean name
|
||||||
* parser for Bean -> JSONObject mapping. A value set to empty string <code>""</code>
|
* parser for Bean -> JSONObject mapping. A value set to empty string <code>""</code>
|
||||||
* will have the Bean parser fall back to the default field name processing.
|
* will have the Bean parser fall back to the default field name processing.
|
||||||
*/
|
*/
|
||||||
|
@Documented
|
||||||
|
@Retention(RUNTIME)
|
||||||
|
@Target({METHOD})
|
||||||
public @interface JSONPropertyName {
|
public @interface JSONPropertyName {
|
||||||
/**
|
/**
|
||||||
* @return The name of the property as to be used in the JSON Object.
|
* @return The name of the property as to be used in the JSON Object.
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2006 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.io.StringWriter;
|
import java.io.StringWriter;
|
||||||
@@ -50,7 +30,7 @@ import java.io.StringWriter;
|
|||||||
* <p>
|
* <p>
|
||||||
* The first method called must be <code>array</code> or <code>object</code>.
|
* The first method called must be <code>array</code> or <code>object</code>.
|
||||||
* There are no methods for adding commas or colons. JSONStringer adds them for
|
* There are no methods for adding commas or colons. JSONStringer adds them for
|
||||||
* you. Objects and arrays can be nested up to 20 levels deep.
|
* you. Objects and arrays can be nested up to 200 levels deep.
|
||||||
* <p>
|
* <p>
|
||||||
* This can sometimes be easier than using a JSONObject to build a string.
|
* This can sometimes be easier than using a JSONObject to build a string.
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
|
|||||||
@@ -1,34 +1,10 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
import java.io.BufferedReader;
|
import java.io.*;
|
||||||
import java.io.IOException;
|
import java.nio.charset.Charset;
|
||||||
import java.io.InputStream;
|
|
||||||
import java.io.InputStreamReader;
|
|
||||||
import java.io.Reader;
|
|
||||||
import java.io.StringReader;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -81,7 +57,7 @@ public class JSONTokener {
|
|||||||
* @param inputStream The source.
|
* @param inputStream The source.
|
||||||
*/
|
*/
|
||||||
public JSONTokener(InputStream inputStream) {
|
public JSONTokener(InputStream inputStream) {
|
||||||
this(new InputStreamReader(inputStream));
|
this(new InputStreamReader(inputStream, Charset.forName("UTF-8")));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -210,6 +186,12 @@ public class JSONTokener {
|
|||||||
return this.previous;
|
return this.previous;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the last character read from the input or '\0' if nothing has been read yet.
|
||||||
|
* @return the last character read from the input.
|
||||||
|
*/
|
||||||
|
protected char getPrevious() { return this.previous;}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Increments the internal indexes according to the previous character
|
* Increments the internal indexes according to the previous character
|
||||||
* read and the character passed as the current character.
|
* read and the character passed as the current character.
|
||||||
@@ -428,10 +410,18 @@ public class JSONTokener {
|
|||||||
return this.nextString(c);
|
return this.nextString(c);
|
||||||
case '{':
|
case '{':
|
||||||
this.back();
|
this.back();
|
||||||
|
try {
|
||||||
return new JSONObject(this);
|
return new JSONObject(this);
|
||||||
|
} catch (StackOverflowError e) {
|
||||||
|
throw new JSONException("JSON Array or Object depth too large to process.", e);
|
||||||
|
}
|
||||||
case '[':
|
case '[':
|
||||||
this.back();
|
this.back();
|
||||||
|
try {
|
||||||
return new JSONArray(this);
|
return new JSONArray(this);
|
||||||
|
} catch (StackOverflowError e) {
|
||||||
|
throw new JSONException("JSON Array or Object depth too large to process.", e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -528,4 +518,10 @@ public class JSONTokener {
|
|||||||
return " at " + this.index + " [character " + this.character + " line " +
|
return " at " + this.index + " [character " + this.character + " line " +
|
||||||
this.line + "]";
|
this.line + "]";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void close() throws IOException {
|
||||||
|
if(reader!=null){
|
||||||
|
reader.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,27 +5,7 @@ import java.util.Collection;
|
|||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2006 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
112
src/main/java/org/json/ParserConfiguration.java
Normal file
112
src/main/java/org/json/ParserConfiguration.java
Normal file
@@ -0,0 +1,112 @@
|
|||||||
|
package org.json;
|
||||||
|
/*
|
||||||
|
Public Domain.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configuration base object for parsers. The configuration is immutable.
|
||||||
|
*/
|
||||||
|
@SuppressWarnings({""})
|
||||||
|
public class ParserConfiguration {
|
||||||
|
/**
|
||||||
|
* Used to indicate there's no defined limit to the maximum nesting depth when parsing a document.
|
||||||
|
*/
|
||||||
|
public static final int UNDEFINED_MAXIMUM_NESTING_DEPTH = -1;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The default maximum nesting depth when parsing a document.
|
||||||
|
*/
|
||||||
|
public static final int DEFAULT_MAXIMUM_NESTING_DEPTH = 512;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies if values should be kept as strings (<code>true</code>), or if
|
||||||
|
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||||
|
*/
|
||||||
|
protected boolean keepStrings;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The maximum nesting depth when parsing a document.
|
||||||
|
*/
|
||||||
|
protected int maxNestingDepth;
|
||||||
|
|
||||||
|
public ParserConfiguration() {
|
||||||
|
this.keepStrings = false;
|
||||||
|
this.maxNestingDepth = DEFAULT_MAXIMUM_NESTING_DEPTH;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected ParserConfiguration(final boolean keepStrings, final int maxNestingDepth) {
|
||||||
|
this.keepStrings = keepStrings;
|
||||||
|
this.maxNestingDepth = maxNestingDepth;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides a new instance of the same configuration.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected ParserConfiguration clone() {
|
||||||
|
// future modifications to this method should always ensure a "deep"
|
||||||
|
// clone in the case of collections. i.e. if a Map is added as a configuration
|
||||||
|
// item, a new map instance should be created and if possible each value in the
|
||||||
|
// map should be cloned as well. If the values of the map are known to also
|
||||||
|
// be immutable, then a shallow clone of the map is acceptable.
|
||||||
|
return new ParserConfiguration(
|
||||||
|
this.keepStrings,
|
||||||
|
this.maxNestingDepth
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When parsing the XML into JSONML, specifies if values should be kept as strings (<code>true</code>), or if
|
||||||
|
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||||
|
*
|
||||||
|
* @return The <code>keepStrings</code> configuration value.
|
||||||
|
*/
|
||||||
|
public boolean isKeepStrings() {
|
||||||
|
return this.keepStrings;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When parsing the XML into JSONML, specifies if values should be kept as strings (<code>true</code>), or if
|
||||||
|
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||||
|
*
|
||||||
|
* @param newVal
|
||||||
|
* new value to use for the <code>keepStrings</code> configuration option.
|
||||||
|
*
|
||||||
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
|
*/
|
||||||
|
public <T extends ParserConfiguration> T withKeepStrings(final boolean newVal) {
|
||||||
|
T newConfig = (T)this.clone();
|
||||||
|
newConfig.keepStrings = newVal;
|
||||||
|
return newConfig;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The maximum nesting depth that the parser will descend before throwing an exception
|
||||||
|
* when parsing the XML into JSONML.
|
||||||
|
* @return the maximum nesting depth set for this configuration
|
||||||
|
*/
|
||||||
|
public int getMaxNestingDepth() {
|
||||||
|
return maxNestingDepth;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines the maximum nesting depth that the parser will descend before throwing an exception
|
||||||
|
* when parsing the XML into JSONML. The default max nesting depth is 512, which means the parser
|
||||||
|
* will throw a JsonException if the maximum depth is reached.
|
||||||
|
* Using any negative value as a parameter is equivalent to setting no limit to the nesting depth,
|
||||||
|
* which means the parses will go as deep as the maximum call stack size allows.
|
||||||
|
* @param maxNestingDepth the maximum nesting depth allowed to the XML parser
|
||||||
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
|
*/
|
||||||
|
public <T extends ParserConfiguration> T withMaxNestingDepth(int maxNestingDepth) {
|
||||||
|
T newConfig = (T)this.clone();
|
||||||
|
|
||||||
|
if (maxNestingDepth > UNDEFINED_MAXIMUM_NESTING_DEPTH) {
|
||||||
|
newConfig.maxNestingDepth = maxNestingDepth;
|
||||||
|
} else {
|
||||||
|
newConfig.maxNestingDepth = UNDEFINED_MAXIMUM_NESTING_DEPTH;
|
||||||
|
}
|
||||||
|
|
||||||
|
return newConfig;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Enumeration;
|
import java.util.Enumeration;
|
||||||
|
|||||||
@@ -1,32 +1,11 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2015 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.io.Reader;
|
import java.io.Reader;
|
||||||
import java.io.StringReader;
|
import java.io.StringReader;
|
||||||
import java.lang.reflect.Method;
|
|
||||||
import java.math.BigDecimal;
|
import java.math.BigDecimal;
|
||||||
import java.math.BigInteger;
|
import java.math.BigInteger;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
@@ -250,10 +229,14 @@ public class XML {
|
|||||||
* The JSONObject that will include the new material.
|
* The JSONObject that will include the new material.
|
||||||
* @param name
|
* @param name
|
||||||
* The tag name.
|
* The tag name.
|
||||||
|
* @param config
|
||||||
|
* The XML parser configuration.
|
||||||
|
* @param currentNestingDepth
|
||||||
|
* The current nesting depth.
|
||||||
* @return true if the close tag is processed.
|
* @return true if the close tag is processed.
|
||||||
* @throws JSONException
|
* @throws JSONException Thrown if any parsing error occurs.
|
||||||
*/
|
*/
|
||||||
private static boolean parse(XMLTokener x, JSONObject context, String name, XMLParserConfiguration config)
|
private static boolean parse(XMLTokener x, JSONObject context, String name, XMLParserConfiguration config, int currentNestingDepth)
|
||||||
throws JSONException {
|
throws JSONException {
|
||||||
char c;
|
char c;
|
||||||
int i;
|
int i;
|
||||||
@@ -380,6 +363,16 @@ public class XML {
|
|||||||
if (x.nextToken() != GT) {
|
if (x.nextToken() != GT) {
|
||||||
throw x.syntaxError("Misshaped tag");
|
throw x.syntaxError("Misshaped tag");
|
||||||
}
|
}
|
||||||
|
if (config.getForceList().contains(tagName)) {
|
||||||
|
// Force the value to be an array
|
||||||
|
if (nilAttributeFound) {
|
||||||
|
context.append(tagName, JSONObject.NULL);
|
||||||
|
} else if (jsonObject.length() > 0) {
|
||||||
|
context.append(tagName, jsonObject);
|
||||||
|
} else {
|
||||||
|
context.put(tagName, new JSONArray());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
if (nilAttributeFound) {
|
if (nilAttributeFound) {
|
||||||
context.accumulate(tagName, JSONObject.NULL);
|
context.accumulate(tagName, JSONObject.NULL);
|
||||||
} else if (jsonObject.length() > 0) {
|
} else if (jsonObject.length() > 0) {
|
||||||
@@ -387,6 +380,7 @@ public class XML {
|
|||||||
} else {
|
} else {
|
||||||
context.accumulate(tagName, "");
|
context.accumulate(tagName, "");
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
} else if (token == GT) {
|
} else if (token == GT) {
|
||||||
@@ -412,7 +406,22 @@ public class XML {
|
|||||||
|
|
||||||
} else if (token == LT) {
|
} else if (token == LT) {
|
||||||
// Nested element
|
// Nested element
|
||||||
if (parse(x, jsonObject, tagName, config)) {
|
if (currentNestingDepth == config.getMaxNestingDepth()) {
|
||||||
|
throw x.syntaxError("Maximum nesting depth of " + config.getMaxNestingDepth() + " reached");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parse(x, jsonObject, tagName, config, currentNestingDepth + 1)) {
|
||||||
|
if (config.getForceList().contains(tagName)) {
|
||||||
|
// Force the value to be an array
|
||||||
|
if (jsonObject.length() == 0) {
|
||||||
|
context.put(tagName, new JSONArray());
|
||||||
|
} else if (jsonObject.length() == 1
|
||||||
|
&& jsonObject.opt(config.getcDataTagName()) != null) {
|
||||||
|
context.append(tagName, jsonObject.opt(config.getcDataTagName()));
|
||||||
|
} else {
|
||||||
|
context.append(tagName, jsonObject);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
if (jsonObject.length() == 0) {
|
if (jsonObject.length() == 0) {
|
||||||
context.accumulate(tagName, "");
|
context.accumulate(tagName, "");
|
||||||
} else if (jsonObject.length() == 1
|
} else if (jsonObject.length() == 1
|
||||||
@@ -421,6 +430,8 @@ public class XML {
|
|||||||
} else {
|
} else {
|
||||||
context.accumulate(tagName, jsonObject);
|
context.accumulate(tagName, jsonObject);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -532,7 +543,7 @@ public class XML {
|
|||||||
// This will narrow any values to the smallest reasonable Object representation
|
// This will narrow any values to the smallest reasonable Object representation
|
||||||
// (Integer, Long, or BigInteger)
|
// (Integer, Long, or BigInteger)
|
||||||
|
|
||||||
// BigInteger down conversion: We use a similar bitLenth compare as
|
// BigInteger down conversion: We use a similar bitLength compare as
|
||||||
// BigInteger#intValueExact uses. Increases GC, but objects hold
|
// BigInteger#intValueExact uses. Increases GC, but objects hold
|
||||||
// only what they need. i.e. Less runtime overhead if the value is
|
// only what they need. i.e. Less runtime overhead if the value is
|
||||||
// long lived.
|
// long lived.
|
||||||
@@ -652,7 +663,7 @@ public class XML {
|
|||||||
while (x.more()) {
|
while (x.more()) {
|
||||||
x.skipPast("<");
|
x.skipPast("<");
|
||||||
if(x.more()) {
|
if(x.more()) {
|
||||||
parse(x, jo, null, config);
|
parse(x, jo, null, config, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return jo;
|
return jo;
|
||||||
@@ -749,6 +760,28 @@ public class XML {
|
|||||||
*/
|
*/
|
||||||
public static String toString(final Object object, final String tagName, final XMLParserConfiguration config)
|
public static String toString(final Object object, final String tagName, final XMLParserConfiguration config)
|
||||||
throws JSONException {
|
throws JSONException {
|
||||||
|
return toString(object, tagName, config, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a JSONObject into a well-formed, element-normal XML string,
|
||||||
|
* either pretty print or single-lined depending on indent factor.
|
||||||
|
*
|
||||||
|
* @param object
|
||||||
|
* A JSONObject.
|
||||||
|
* @param tagName
|
||||||
|
* The optional name of the enclosing tag.
|
||||||
|
* @param config
|
||||||
|
* Configuration that can control output to XML.
|
||||||
|
* @param indentFactor
|
||||||
|
* The number of spaces to add to each level of indentation.
|
||||||
|
* @param indent
|
||||||
|
* The current ident level in spaces.
|
||||||
|
* @return
|
||||||
|
* @throws JSONException
|
||||||
|
*/
|
||||||
|
private static String toString(final Object object, final String tagName, final XMLParserConfiguration config, int indentFactor, int indent)
|
||||||
|
throws JSONException {
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
JSONArray ja;
|
JSONArray ja;
|
||||||
JSONObject jo;
|
JSONObject jo;
|
||||||
@@ -758,9 +791,14 @@ public class XML {
|
|||||||
|
|
||||||
// Emit <tagName>
|
// Emit <tagName>
|
||||||
if (tagName != null) {
|
if (tagName != null) {
|
||||||
|
sb.append(indent(indent));
|
||||||
sb.append('<');
|
sb.append('<');
|
||||||
sb.append(tagName);
|
sb.append(tagName);
|
||||||
sb.append('>');
|
sb.append('>');
|
||||||
|
if(indentFactor > 0){
|
||||||
|
sb.append("\n");
|
||||||
|
indent += indentFactor;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Loop thru the keys.
|
// Loop thru the keys.
|
||||||
@@ -803,31 +841,39 @@ public class XML {
|
|||||||
sb.append('<');
|
sb.append('<');
|
||||||
sb.append(key);
|
sb.append(key);
|
||||||
sb.append('>');
|
sb.append('>');
|
||||||
sb.append(toString(val, null, config));
|
sb.append(toString(val, null, config, indentFactor, indent));
|
||||||
sb.append("</");
|
sb.append("</");
|
||||||
sb.append(key);
|
sb.append(key);
|
||||||
sb.append('>');
|
sb.append('>');
|
||||||
} else {
|
} else {
|
||||||
sb.append(toString(val, key, config));
|
sb.append(toString(val, key, config, indentFactor, indent));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if ("".equals(value)) {
|
} else if ("".equals(value)) {
|
||||||
|
sb.append(indent(indent));
|
||||||
sb.append('<');
|
sb.append('<');
|
||||||
sb.append(key);
|
sb.append(key);
|
||||||
sb.append("/>");
|
sb.append("/>");
|
||||||
|
if(indentFactor > 0){
|
||||||
|
sb.append("\n");
|
||||||
|
}
|
||||||
|
|
||||||
// Emit a new tag <k>
|
// Emit a new tag <k>
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
sb.append(toString(value, key, config));
|
sb.append(toString(value, key, config, indentFactor, indent));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (tagName != null) {
|
if (tagName != null) {
|
||||||
|
|
||||||
// Emit the </tagName> close tag
|
// Emit the </tagName> close tag
|
||||||
|
sb.append(indent(indent - indentFactor));
|
||||||
sb.append("</");
|
sb.append("</");
|
||||||
sb.append(tagName);
|
sb.append(tagName);
|
||||||
sb.append('>');
|
sb.append('>');
|
||||||
|
if(indentFactor > 0){
|
||||||
|
sb.append("\n");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
|
|
||||||
@@ -846,15 +892,85 @@ public class XML {
|
|||||||
// XML does not have good support for arrays. If an array
|
// XML does not have good support for arrays. If an array
|
||||||
// appears in a place where XML is lacking, synthesize an
|
// appears in a place where XML is lacking, synthesize an
|
||||||
// <array> element.
|
// <array> element.
|
||||||
sb.append(toString(val, tagName == null ? "array" : tagName, config));
|
sb.append(toString(val, tagName == null ? "array" : tagName, config, indentFactor, indent));
|
||||||
}
|
}
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
string = (object == null) ? "null" : escape(object.toString());
|
|
||||||
return (tagName == null) ? "\"" + string + "\""
|
|
||||||
: (string.length() == 0) ? "<" + tagName + "/>" : "<" + tagName
|
|
||||||
+ ">" + string + "</" + tagName + ">";
|
|
||||||
|
|
||||||
|
string = (object == null) ? "null" : escape(object.toString());
|
||||||
|
|
||||||
|
if(tagName == null){
|
||||||
|
return indent(indent) + "\"" + string + "\"" + ((indentFactor > 0) ? "\n" : "");
|
||||||
|
} else if(string.length() == 0){
|
||||||
|
return indent(indent) + "<" + tagName + "/>" + ((indentFactor > 0) ? "\n" : "");
|
||||||
|
} else {
|
||||||
|
return indent(indent) + "<" + tagName
|
||||||
|
+ ">" + string + "</" + tagName + ">" + ((indentFactor > 0) ? "\n" : "");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a JSONObject into a well-formed, pretty printed element-normal XML string.
|
||||||
|
*
|
||||||
|
* @param object
|
||||||
|
* A JSONObject.
|
||||||
|
* @param indentFactor
|
||||||
|
* The number of spaces to add to each level of indentation.
|
||||||
|
* @return A string.
|
||||||
|
* @throws JSONException Thrown if there is an error parsing the string
|
||||||
|
*/
|
||||||
|
public static String toString(Object object, int indentFactor){
|
||||||
|
return toString(object, null, XMLParserConfiguration.ORIGINAL, indentFactor);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a JSONObject into a well-formed, pretty printed element-normal XML string.
|
||||||
|
*
|
||||||
|
* @param object
|
||||||
|
* A JSONObject.
|
||||||
|
* @param tagName
|
||||||
|
* The optional name of the enclosing tag.
|
||||||
|
* @param indentFactor
|
||||||
|
* The number of spaces to add to each level of indentation.
|
||||||
|
* @return A string.
|
||||||
|
* @throws JSONException Thrown if there is an error parsing the string
|
||||||
|
*/
|
||||||
|
public static String toString(final Object object, final String tagName, int indentFactor) {
|
||||||
|
return toString(object, tagName, XMLParserConfiguration.ORIGINAL, indentFactor);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a JSONObject into a well-formed, pretty printed element-normal XML string.
|
||||||
|
*
|
||||||
|
* @param object
|
||||||
|
* A JSONObject.
|
||||||
|
* @param tagName
|
||||||
|
* The optional name of the enclosing tag.
|
||||||
|
* @param config
|
||||||
|
* Configuration that can control output to XML.
|
||||||
|
* @param indentFactor
|
||||||
|
* The number of spaces to add to each level of indentation.
|
||||||
|
* @return A string.
|
||||||
|
* @throws JSONException Thrown if there is an error parsing the string
|
||||||
|
*/
|
||||||
|
public static String toString(final Object object, final String tagName, final XMLParserConfiguration config, int indentFactor)
|
||||||
|
throws JSONException {
|
||||||
|
return toString(object, tagName, config, indentFactor, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return a String consisting of a number of space characters specified by indent
|
||||||
|
*
|
||||||
|
* @param indent
|
||||||
|
* The number of spaces to be appended to the String.
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
private static final String indent(int indent) {
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
for (int i = 0; i < indent; i++) {
|
||||||
|
sb.append(' ');
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,31 +1,13 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -33,7 +15,13 @@ import java.util.Map;
|
|||||||
* @author AylwardJ
|
* @author AylwardJ
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings({""})
|
@SuppressWarnings({""})
|
||||||
public class XMLParserConfiguration {
|
public class XMLParserConfiguration extends ParserConfiguration {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The default maximum nesting depth when parsing a XML document to JSON.
|
||||||
|
*/
|
||||||
|
// public static final int DEFAULT_MAXIMUM_NESTING_DEPTH = 512; // We could override
|
||||||
|
|
||||||
/** Original Configuration of the XML Parser. */
|
/** Original Configuration of the XML Parser. */
|
||||||
public static final XMLParserConfiguration ORIGINAL
|
public static final XMLParserConfiguration ORIGINAL
|
||||||
= new XMLParserConfiguration();
|
= new XMLParserConfiguration();
|
||||||
@@ -41,12 +29,6 @@ public class XMLParserConfiguration {
|
|||||||
public static final XMLParserConfiguration KEEP_STRINGS
|
public static final XMLParserConfiguration KEEP_STRINGS
|
||||||
= new XMLParserConfiguration().withKeepStrings(true);
|
= new XMLParserConfiguration().withKeepStrings(true);
|
||||||
|
|
||||||
/**
|
|
||||||
* When parsing the XML into JSON, specifies if values should be kept as strings (<code>true</code>), or if
|
|
||||||
* they should try to be guessed into JSON values (numeric, boolean, string)
|
|
||||||
*/
|
|
||||||
private boolean keepStrings;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The name of the key in a JSON Object that indicates a CDATA section. Historically this has
|
* The name of the key in a JSON Object that indicates a CDATA section. Historically this has
|
||||||
* been the value "content" but can be changed. Use <code>null</code> to indicate no CDATA
|
* been the value "content" but can be changed. Use <code>null</code> to indicate no CDATA
|
||||||
@@ -66,15 +48,22 @@ public class XMLParserConfiguration {
|
|||||||
*/
|
*/
|
||||||
private Map<String, XMLXsiTypeConverter<?>> xsiTypeMap;
|
private Map<String, XMLXsiTypeConverter<?>> xsiTypeMap;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When parsing the XML into JSON, specifies the tags whose values should be converted
|
||||||
|
* to arrays
|
||||||
|
*/
|
||||||
|
private Set<String> forceList;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Default parser configuration. Does not keep strings (tries to implicitly convert
|
* Default parser configuration. Does not keep strings (tries to implicitly convert
|
||||||
* values), and the CDATA Tag Name is "content".
|
* values), and the CDATA Tag Name is "content".
|
||||||
*/
|
*/
|
||||||
public XMLParserConfiguration () {
|
public XMLParserConfiguration () {
|
||||||
this.keepStrings = false;
|
super();
|
||||||
this.cDataTagName = "content";
|
this.cDataTagName = "content";
|
||||||
this.convertNilAttributeToNull = false;
|
this.convertNilAttributeToNull = false;
|
||||||
this.xsiTypeMap = Collections.emptyMap();
|
this.xsiTypeMap = Collections.emptyMap();
|
||||||
|
this.forceList = Collections.emptySet();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -94,7 +83,7 @@ public class XMLParserConfiguration {
|
|||||||
* Configure the parser string processing to try and convert XML values to JSON values and
|
* Configure the parser string processing to try and convert XML values to JSON values and
|
||||||
* use the passed CDATA Tag Name the processing value. Pass <code>null</code> to
|
* use the passed CDATA Tag Name the processing value. Pass <code>null</code> to
|
||||||
* disable CDATA processing
|
* disable CDATA processing
|
||||||
* @param cDataTagName<code>null</code> to disable CDATA processing. Any other value
|
* @param cDataTagName <code>null</code> to disable CDATA processing. Any other value
|
||||||
* to use that value as the JSONObject key name to process as CDATA.
|
* to use that value as the JSONObject key name to process as CDATA.
|
||||||
* @deprecated This constructor has been deprecated in favor of using the new builder
|
* @deprecated This constructor has been deprecated in favor of using the new builder
|
||||||
* pattern for the configuration.
|
* pattern for the configuration.
|
||||||
@@ -109,7 +98,7 @@ public class XMLParserConfiguration {
|
|||||||
* Configure the parser to use custom settings.
|
* Configure the parser to use custom settings.
|
||||||
* @param keepStrings <code>true</code> to parse all values as string.
|
* @param keepStrings <code>true</code> to parse all values as string.
|
||||||
* <code>false</code> to try and convert XML string values into a JSON value.
|
* <code>false</code> to try and convert XML string values into a JSON value.
|
||||||
* @param cDataTagName<code>null</code> to disable CDATA processing. Any other value
|
* @param cDataTagName <code>null</code> to disable CDATA processing. Any other value
|
||||||
* to use that value as the JSONObject key name to process as CDATA.
|
* to use that value as the JSONObject key name to process as CDATA.
|
||||||
* @deprecated This constructor has been deprecated in favor of using the new builder
|
* @deprecated This constructor has been deprecated in favor of using the new builder
|
||||||
* pattern for the configuration.
|
* pattern for the configuration.
|
||||||
@@ -117,7 +106,7 @@ public class XMLParserConfiguration {
|
|||||||
*/
|
*/
|
||||||
@Deprecated
|
@Deprecated
|
||||||
public XMLParserConfiguration (final boolean keepStrings, final String cDataTagName) {
|
public XMLParserConfiguration (final boolean keepStrings, final String cDataTagName) {
|
||||||
this.keepStrings = keepStrings;
|
super(keepStrings, DEFAULT_MAXIMUM_NESTING_DEPTH);
|
||||||
this.cDataTagName = cDataTagName;
|
this.cDataTagName = cDataTagName;
|
||||||
this.convertNilAttributeToNull = false;
|
this.convertNilAttributeToNull = false;
|
||||||
}
|
}
|
||||||
@@ -136,7 +125,7 @@ public class XMLParserConfiguration {
|
|||||||
*/
|
*/
|
||||||
@Deprecated
|
@Deprecated
|
||||||
public XMLParserConfiguration (final boolean keepStrings, final String cDataTagName, final boolean convertNilAttributeToNull) {
|
public XMLParserConfiguration (final boolean keepStrings, final String cDataTagName, final boolean convertNilAttributeToNull) {
|
||||||
this.keepStrings = keepStrings;
|
super(keepStrings, DEFAULT_MAXIMUM_NESTING_DEPTH);
|
||||||
this.cDataTagName = cDataTagName;
|
this.cDataTagName = cDataTagName;
|
||||||
this.convertNilAttributeToNull = convertNilAttributeToNull;
|
this.convertNilAttributeToNull = convertNilAttributeToNull;
|
||||||
}
|
}
|
||||||
@@ -151,13 +140,17 @@ public class XMLParserConfiguration {
|
|||||||
* <code>false</code> to parse values with attribute xsi:nil="true" as {"xsi:nil":true}.
|
* <code>false</code> to parse values with attribute xsi:nil="true" as {"xsi:nil":true}.
|
||||||
* @param xsiTypeMap <code>new HashMap<String, XMLXsiTypeConverter<?>>()</code> to parse values with attribute
|
* @param xsiTypeMap <code>new HashMap<String, XMLXsiTypeConverter<?>>()</code> to parse values with attribute
|
||||||
* xsi:type="integer" as integer, xsi:type="string" as string
|
* xsi:type="integer" as integer, xsi:type="string" as string
|
||||||
|
* @param forceList <code>new HashSet<String>()</code> to parse the provided tags' values as arrays
|
||||||
|
* @param maxNestingDepth <code>int</code> to limit the nesting depth
|
||||||
*/
|
*/
|
||||||
private XMLParserConfiguration (final boolean keepStrings, final String cDataTagName,
|
private XMLParserConfiguration (final boolean keepStrings, final String cDataTagName,
|
||||||
final boolean convertNilAttributeToNull, final Map<String, XMLXsiTypeConverter<?>> xsiTypeMap ) {
|
final boolean convertNilAttributeToNull, final Map<String, XMLXsiTypeConverter<?>> xsiTypeMap, final Set<String> forceList,
|
||||||
this.keepStrings = keepStrings;
|
final int maxNestingDepth) {
|
||||||
|
super(keepStrings, maxNestingDepth);
|
||||||
this.cDataTagName = cDataTagName;
|
this.cDataTagName = cDataTagName;
|
||||||
this.convertNilAttributeToNull = convertNilAttributeToNull;
|
this.convertNilAttributeToNull = convertNilAttributeToNull;
|
||||||
this.xsiTypeMap = Collections.unmodifiableMap(xsiTypeMap);
|
this.xsiTypeMap = Collections.unmodifiableMap(xsiTypeMap);
|
||||||
|
this.forceList = Collections.unmodifiableSet(forceList);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -174,33 +167,24 @@ public class XMLParserConfiguration {
|
|||||||
this.keepStrings,
|
this.keepStrings,
|
||||||
this.cDataTagName,
|
this.cDataTagName,
|
||||||
this.convertNilAttributeToNull,
|
this.convertNilAttributeToNull,
|
||||||
this.xsiTypeMap
|
this.xsiTypeMap,
|
||||||
|
this.forceList,
|
||||||
|
this.maxNestingDepth
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* When parsing the XML into JSON, specifies if values should be kept as strings (<code>true</code>), or if
|
|
||||||
* they should try to be guessed into JSON values (numeric, boolean, string)
|
|
||||||
*
|
|
||||||
* @return The {@link #keepStrings} configuration value.
|
|
||||||
*/
|
|
||||||
public boolean isKeepStrings() {
|
|
||||||
return this.keepStrings;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* When parsing the XML into JSON, specifies if values should be kept as strings (<code>true</code>), or if
|
* When parsing the XML into JSON, specifies if values should be kept as strings (<code>true</code>), or if
|
||||||
* they should try to be guessed into JSON values (numeric, boolean, string)
|
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||||
*
|
*
|
||||||
* @param newVal
|
* @param newVal
|
||||||
* new value to use for the {@link #keepStrings} configuration option.
|
* new value to use for the <code>keepStrings</code> configuration option.
|
||||||
*
|
*
|
||||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public XMLParserConfiguration withKeepStrings(final boolean newVal) {
|
public XMLParserConfiguration withKeepStrings(final boolean newVal) {
|
||||||
XMLParserConfiguration newConfig = this.clone();
|
return super.withKeepStrings(newVal);
|
||||||
newConfig.keepStrings = newVal;
|
|
||||||
return newConfig;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -208,7 +192,7 @@ public class XMLParserConfiguration {
|
|||||||
* been the value "content" but can be changed. Use <code>null</code> to indicate no CDATA
|
* been the value "content" but can be changed. Use <code>null</code> to indicate no CDATA
|
||||||
* processing.
|
* processing.
|
||||||
*
|
*
|
||||||
* @return The {@link #cDataTagName} configuration value.
|
* @return The <code>cDataTagName</code> configuration value.
|
||||||
*/
|
*/
|
||||||
public String getcDataTagName() {
|
public String getcDataTagName() {
|
||||||
return this.cDataTagName;
|
return this.cDataTagName;
|
||||||
@@ -220,7 +204,7 @@ public class XMLParserConfiguration {
|
|||||||
* processing.
|
* processing.
|
||||||
*
|
*
|
||||||
* @param newVal
|
* @param newVal
|
||||||
* new value to use for the {@link #cDataTagName} configuration option.
|
* new value to use for the <code>cDataTagName</code> configuration option.
|
||||||
*
|
*
|
||||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
*/
|
*/
|
||||||
@@ -235,7 +219,7 @@ public class XMLParserConfiguration {
|
|||||||
* should be kept as attribute(<code>false</code>), or they should be converted to
|
* should be kept as attribute(<code>false</code>), or they should be converted to
|
||||||
* <code>null</code>(<code>true</code>)
|
* <code>null</code>(<code>true</code>)
|
||||||
*
|
*
|
||||||
* @return The {@link #convertNilAttributeToNull} configuration value.
|
* @return The <code>convertNilAttributeToNull</code> configuration value.
|
||||||
*/
|
*/
|
||||||
public boolean isConvertNilAttributeToNull() {
|
public boolean isConvertNilAttributeToNull() {
|
||||||
return this.convertNilAttributeToNull;
|
return this.convertNilAttributeToNull;
|
||||||
@@ -247,7 +231,7 @@ public class XMLParserConfiguration {
|
|||||||
* <code>null</code>(<code>true</code>)
|
* <code>null</code>(<code>true</code>)
|
||||||
*
|
*
|
||||||
* @param newVal
|
* @param newVal
|
||||||
* new value to use for the {@link #convertNilAttributeToNull} configuration option.
|
* new value to use for the <code>convertNilAttributeToNull</code> configuration option.
|
||||||
*
|
*
|
||||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
*/
|
*/
|
||||||
@@ -262,7 +246,7 @@ public class XMLParserConfiguration {
|
|||||||
* will be converted to target type defined to client in this configuration
|
* will be converted to target type defined to client in this configuration
|
||||||
* {@code Map<String, XMLXsiTypeConverter<?>>} to parse values with attribute
|
* {@code Map<String, XMLXsiTypeConverter<?>>} to parse values with attribute
|
||||||
* xsi:type="integer" as integer, xsi:type="string" as string
|
* xsi:type="integer" as integer, xsi:type="string" as string
|
||||||
* @return {@link #xsiTypeMap} unmodifiable configuration map.
|
* @return <code>xsiTypeMap</code> unmodifiable configuration map.
|
||||||
*/
|
*/
|
||||||
public Map<String, XMLXsiTypeConverter<?>> getXsiTypeMap() {
|
public Map<String, XMLXsiTypeConverter<?>> getXsiTypeMap() {
|
||||||
return this.xsiTypeMap;
|
return this.xsiTypeMap;
|
||||||
@@ -283,4 +267,40 @@ public class XMLParserConfiguration {
|
|||||||
newConfig.xsiTypeMap = Collections.unmodifiableMap(cloneXsiTypeMap);
|
newConfig.xsiTypeMap = Collections.unmodifiableMap(cloneXsiTypeMap);
|
||||||
return newConfig;
|
return newConfig;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When parsing the XML into JSON, specifies that tags that will be converted to arrays
|
||||||
|
* in this configuration {@code Set<String>} to parse the provided tags' values as arrays
|
||||||
|
* @return <code>forceList</code> unmodifiable configuration set.
|
||||||
|
*/
|
||||||
|
public Set<String> getForceList() {
|
||||||
|
return this.forceList;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When parsing the XML into JSON, specifies that tags that will be converted to arrays
|
||||||
|
* in this configuration {@code Set<String>} to parse the provided tags' values as arrays
|
||||||
|
* @param forceList {@code new HashSet<String>()} to parse the provided tags' values as arrays
|
||||||
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration withForceList(final Set<String> forceList) {
|
||||||
|
XMLParserConfiguration newConfig = this.clone();
|
||||||
|
Set<String> cloneForceList = new HashSet<String>(forceList);
|
||||||
|
newConfig.forceList = Collections.unmodifiableSet(cloneForceList);
|
||||||
|
return newConfig;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines the maximum nesting depth that the parser will descend before throwing an exception
|
||||||
|
* when parsing the XML into JSON. The default max nesting depth is 512, which means the parser
|
||||||
|
* will throw a JsonException if the maximum depth is reached.
|
||||||
|
* Using any negative value as a parameter is equivalent to setting no limit to the nesting depth,
|
||||||
|
* which means the parses will go as deep as the maximum call stack size allows.
|
||||||
|
* @param maxNestingDepth the maximum nesting depth allowed to the XML parser
|
||||||
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public XMLParserConfiguration withMaxNestingDepth(int maxNestingDepth) {
|
||||||
|
return super.withMaxNestingDepth(maxNestingDepth);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.io.Reader;
|
import java.io.Reader;
|
||||||
|
|||||||
@@ -1,26 +1,6 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
@@ -190,7 +170,7 @@ public class CDLTest {
|
|||||||
CDL.toJSONArray(badLine);
|
CDL.toJSONArray(badLine);
|
||||||
fail("Expecting an exception");
|
fail("Expecting an exception");
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
System.out.println("Message" + e.getMessage());
|
//System.out.println("Message" + e.getMessage());
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"Bad character 'V' (86). at 20 [character 9 line 2]",
|
"Bad character 'V' (86). at 20 [character 9 line 2]",
|
||||||
e.getMessage());
|
e.getMessage());
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
@@ -93,7 +73,7 @@ public class EnumTest {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* To serialize an enum by its set of allowed values, use getNames()
|
* To serialize an enum by its set of allowed values, use getNames()
|
||||||
* and the the JSONObject Object with names constructor.
|
* and the JSONObject Object with names constructor.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void jsonObjectFromEnumWithNames() {
|
public void jsonObjectFromEnumWithNames() {
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
@@ -29,8 +9,10 @@ import static org.junit.Assert.assertFalse;
|
|||||||
import static org.junit.Assert.assertNotNull;
|
import static org.junit.Assert.assertNotNull;
|
||||||
import static org.junit.Assert.assertNull;
|
import static org.junit.Assert.assertNull;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
import java.io.StringWriter;
|
import java.io.StringWriter;
|
||||||
import java.math.BigDecimal;
|
import java.math.BigDecimal;
|
||||||
import java.math.BigInteger;
|
import java.math.BigInteger;
|
||||||
@@ -47,6 +29,9 @@ import org.json.JSONArray;
|
|||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
import org.json.JSONObject;
|
import org.json.JSONObject;
|
||||||
import org.json.JSONPointerException;
|
import org.json.JSONPointerException;
|
||||||
|
import org.json.JSONString;
|
||||||
|
import org.json.JSONTokener;
|
||||||
|
import org.json.junit.data.MyJsonString;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
import com.jayway.jsonpath.Configuration;
|
import com.jayway.jsonpath.Configuration;
|
||||||
@@ -87,6 +72,7 @@ public class JSONArrayTest {
|
|||||||
@Test
|
@Test
|
||||||
public void verifySimilar() {
|
public void verifySimilar() {
|
||||||
final String string1 = "HasSameRef";
|
final String string1 = "HasSameRef";
|
||||||
|
final String string2 = "HasDifferentRef";
|
||||||
JSONArray obj1 = new JSONArray()
|
JSONArray obj1 = new JSONArray()
|
||||||
.put("abc")
|
.put("abc")
|
||||||
.put(string1)
|
.put(string1)
|
||||||
@@ -102,9 +88,19 @@ public class JSONArrayTest {
|
|||||||
.put(new String(string1))
|
.put(new String(string1))
|
||||||
.put(2);
|
.put(2);
|
||||||
|
|
||||||
assertFalse("Should eval to false", obj1.similar(obj2));
|
JSONArray obj4 = new JSONArray()
|
||||||
|
.put("abc")
|
||||||
|
.put(2.0)
|
||||||
|
.put(new String(string1));
|
||||||
|
|
||||||
assertTrue("Should eval to true", obj1.similar(obj3));
|
JSONArray obj5 = new JSONArray()
|
||||||
|
.put("abc")
|
||||||
|
.put(2.0)
|
||||||
|
.put(new String(string2));
|
||||||
|
|
||||||
|
assertFalse("obj1-obj2 Should eval to false", obj1.similar(obj2));
|
||||||
|
assertTrue("obj1-obj3 Should eval to true", obj1.similar(obj3));
|
||||||
|
assertFalse("obj4-obj5 Should eval to false", obj4.similar(obj5));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -223,6 +219,10 @@ public class JSONArrayTest {
|
|||||||
assertTrue(
|
assertTrue(
|
||||||
"The RAW Collection should give me the same as the Typed Collection",
|
"The RAW Collection should give me the same as the Typed Collection",
|
||||||
expected.similar(jaObj));
|
expected.similar(jaObj));
|
||||||
|
Util.checkJSONArrayMaps(expected);
|
||||||
|
Util.checkJSONArrayMaps(jaObj);
|
||||||
|
Util.checkJSONArrayMaps(jaRaw);
|
||||||
|
Util.checkJSONArrayMaps(jaInt);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -261,6 +261,7 @@ public class JSONArrayTest {
|
|||||||
myList.get(i),
|
myList.get(i),
|
||||||
jsonArray.getString(myInts.length + i));
|
jsonArray.getString(myInts.length + i));
|
||||||
}
|
}
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -294,6 +295,9 @@ public class JSONArrayTest {
|
|||||||
assertTrue(
|
assertTrue(
|
||||||
"The RAW Collection should give me the same as the Typed Collection",
|
"The RAW Collection should give me the same as the Typed Collection",
|
||||||
expected.similar(jaInt));
|
expected.similar(jaInt));
|
||||||
|
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
||||||
|
jaRaw, jaObj, jaInt
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -337,6 +341,9 @@ public class JSONArrayTest {
|
|||||||
assertTrue(
|
assertTrue(
|
||||||
"The RAW Collection should give me the same as the Typed Collection",
|
"The RAW Collection should give me the same as the Typed Collection",
|
||||||
expected.similar(jaObjObj));
|
expected.similar(jaObjObj));
|
||||||
|
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
||||||
|
expected, jaRaw, jaStrObj, jaStrInt, jaObjObj
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -364,6 +371,8 @@ public class JSONArrayTest {
|
|||||||
new Double(23.45e-4).equals(jsonArray.getDouble(5)));
|
new Double(23.45e-4).equals(jsonArray.getDouble(5)));
|
||||||
assertTrue("Array string double",
|
assertTrue("Array string double",
|
||||||
new Double(23.45).equals(jsonArray.getDouble(6)));
|
new Double(23.45).equals(jsonArray.getDouble(6)));
|
||||||
|
assertTrue("Array double can be float",
|
||||||
|
new Float(23.45e-4f).equals(jsonArray.getFloat(5)));
|
||||||
// ints
|
// ints
|
||||||
assertTrue("Array value int",
|
assertTrue("Array value int",
|
||||||
new Integer(42).equals(jsonArray.getInt(7)));
|
new Integer(42).equals(jsonArray.getInt(7)));
|
||||||
@@ -381,6 +390,7 @@ public class JSONArrayTest {
|
|||||||
new Long(-1).equals(jsonArray.getLong(12)));
|
new Long(-1).equals(jsonArray.getLong(12)));
|
||||||
|
|
||||||
assertTrue("Array value null", jsonArray.isNull(-1));
|
assertTrue("Array value null", jsonArray.isNull(-1));
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -396,7 +406,7 @@ public class JSONArrayTest {
|
|||||||
assertTrue("expected getBoolean to fail", false);
|
assertTrue("expected getBoolean to fail", false);
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expected an exception message",
|
assertEquals("Expected an exception message",
|
||||||
"JSONArray[4] is not a boolean.",e.getMessage());
|
"JSONArray[4] is not a boolean (class java.lang.String : hello).",e.getMessage());
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
jsonArray.get(-1);
|
jsonArray.get(-1);
|
||||||
@@ -410,43 +420,44 @@ public class JSONArrayTest {
|
|||||||
assertTrue("expected getDouble to fail", false);
|
assertTrue("expected getDouble to fail", false);
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expected an exception message",
|
assertEquals("Expected an exception message",
|
||||||
"JSONArray[4] is not a double.",e.getMessage());
|
"JSONArray[4] is not a double (class java.lang.String : hello).",e.getMessage());
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
jsonArray.getInt(4);
|
jsonArray.getInt(4);
|
||||||
assertTrue("expected getInt to fail", false);
|
assertTrue("expected getInt to fail", false);
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expected an exception message",
|
assertEquals("Expected an exception message",
|
||||||
"JSONArray[4] is not a int.",e.getMessage());
|
"JSONArray[4] is not a int (class java.lang.String : hello).",e.getMessage());
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
jsonArray.getJSONArray(4);
|
jsonArray.getJSONArray(4);
|
||||||
assertTrue("expected getJSONArray to fail", false);
|
assertTrue("expected getJSONArray to fail", false);
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expected an exception message",
|
assertEquals("Expected an exception message",
|
||||||
"JSONArray[4] is not a JSONArray.",e.getMessage());
|
"JSONArray[4] is not a JSONArray (class java.lang.String : hello).",e.getMessage());
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
jsonArray.getJSONObject(4);
|
jsonArray.getJSONObject(4);
|
||||||
assertTrue("expected getJSONObject to fail", false);
|
assertTrue("expected getJSONObject to fail", false);
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expected an exception message",
|
assertEquals("Expected an exception message",
|
||||||
"JSONArray[4] is not a JSONObject.",e.getMessage());
|
"JSONArray[4] is not a JSONObject (class java.lang.String : hello).",e.getMessage());
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
jsonArray.getLong(4);
|
jsonArray.getLong(4);
|
||||||
assertTrue("expected getLong to fail", false);
|
assertTrue("expected getLong to fail", false);
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expected an exception message",
|
assertEquals("Expected an exception message",
|
||||||
"JSONArray[4] is not a long.",e.getMessage());
|
"JSONArray[4] is not a long (class java.lang.String : hello).",e.getMessage());
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
jsonArray.getString(5);
|
jsonArray.getString(5);
|
||||||
assertTrue("expected getString to fail", false);
|
assertTrue("expected getString to fail", false);
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expected an exception message",
|
assertEquals("Expected an exception message",
|
||||||
"JSONArray[5] is not a String.",e.getMessage());
|
"JSONArray[5] is not a String (class java.math.BigDecimal : 0.002345).",e.getMessage());
|
||||||
}
|
}
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -483,6 +494,7 @@ public class JSONArrayTest {
|
|||||||
assertTrue("expected value4", "value4".equals(jsonArray.query("/10/key4")));
|
assertTrue("expected value4", "value4".equals(jsonArray.query("/10/key4")));
|
||||||
assertTrue("expected 0", Integer.valueOf(0).equals(jsonArray.query("/11")));
|
assertTrue("expected 0", Integer.valueOf(0).equals(jsonArray.query("/11")));
|
||||||
assertTrue("expected \"-1\"", "-1".equals(jsonArray.query("/12")));
|
assertTrue("expected \"-1\"", "-1".equals(jsonArray.query("/12")));
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -496,6 +508,9 @@ public class JSONArrayTest {
|
|||||||
assertTrue("expected JSONArray length 13. instead found "+jsonArray.length(), jsonArray.length() == 13);
|
assertTrue("expected JSONArray length 13. instead found "+jsonArray.length(), jsonArray.length() == 13);
|
||||||
JSONArray nestedJsonArray = jsonArray.getJSONArray(9);
|
JSONArray nestedJsonArray = jsonArray.getJSONArray(9);
|
||||||
assertTrue("expected JSONArray length 1", nestedJsonArray.length() == 1);
|
assertTrue("expected JSONArray length 1", nestedJsonArray.length() == 1);
|
||||||
|
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
||||||
|
jsonArray, nestedJsonArray
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -571,6 +586,10 @@ public class JSONArrayTest {
|
|||||||
"hello".equals(jsonArray.optString(4)));
|
"hello".equals(jsonArray.optString(4)));
|
||||||
assertTrue("Array opt string default implicit",
|
assertTrue("Array opt string default implicit",
|
||||||
"".equals(jsonArray.optString(-1)));
|
"".equals(jsonArray.optString(-1)));
|
||||||
|
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
||||||
|
jsonArray, nestedJsonArray
|
||||||
|
)));
|
||||||
|
Util.checkJSONObjectMaps(nestedJsonObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -585,7 +604,9 @@ public class JSONArrayTest {
|
|||||||
assertTrue("unexpected optLong value",ja.optLong(0,0)==123);
|
assertTrue("unexpected optLong value",ja.optLong(0,0)==123);
|
||||||
assertTrue("unexpected optDouble value",ja.optDouble(0,0.0)==123.0);
|
assertTrue("unexpected optDouble value",ja.optDouble(0,0.0)==123.0);
|
||||||
assertTrue("unexpected optBigInteger value",ja.optBigInteger(0,BigInteger.ZERO).compareTo(new BigInteger("123"))==0);
|
assertTrue("unexpected optBigInteger value",ja.optBigInteger(0,BigInteger.ZERO).compareTo(new BigInteger("123"))==0);
|
||||||
assertTrue("unexpected optBigDecimal value",ja.optBigDecimal(0,BigDecimal.ZERO).compareTo(new BigDecimal("123"))==0); }
|
assertTrue("unexpected optBigDecimal value",ja.optBigDecimal(0,BigDecimal.ZERO).compareTo(new BigDecimal("123"))==0);
|
||||||
|
Util.checkJSONArrayMaps(ja);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Exercise the JSONArray.put(value) method with various parameters
|
* Exercise the JSONArray.put(value) method with various parameters
|
||||||
@@ -661,6 +682,8 @@ public class JSONArrayTest {
|
|||||||
assertTrue("expected 2 items in [9]", ((List<?>)(JsonPath.read(doc, "$[9]"))).size() == 2);
|
assertTrue("expected 2 items in [9]", ((List<?>)(JsonPath.read(doc, "$[9]"))).size() == 2);
|
||||||
assertTrue("expected 1", Integer.valueOf(1).equals(jsonArray.query("/9/0")));
|
assertTrue("expected 1", Integer.valueOf(1).equals(jsonArray.query("/9/0")));
|
||||||
assertTrue("expected 2", Integer.valueOf(2).equals(jsonArray.query("/9/1")));
|
assertTrue("expected 2", Integer.valueOf(2).equals(jsonArray.query("/9/1")));
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
|
Util.checkJSONObjectMaps(jsonObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -740,6 +763,8 @@ public class JSONArrayTest {
|
|||||||
assertTrue("expected 2", Integer.valueOf(2).equals(jsonArray.query("/9/1")));
|
assertTrue("expected 2", Integer.valueOf(2).equals(jsonArray.query("/9/1")));
|
||||||
assertTrue("expected 1 item in [10]", ((Map<?,?>)(JsonPath.read(doc, "$[10]"))).size() == 1);
|
assertTrue("expected 1 item in [10]", ((Map<?,?>)(JsonPath.read(doc, "$[10]"))).size() == 1);
|
||||||
assertTrue("expected v1", "v1".equals(jsonArray.query("/10/k1")));
|
assertTrue("expected v1", "v1".equals(jsonArray.query("/10/k1")));
|
||||||
|
Util.checkJSONObjectMaps(jsonObject);
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -756,6 +781,7 @@ public class JSONArrayTest {
|
|||||||
jsonArray.remove(0);
|
jsonArray.remove(0);
|
||||||
assertTrue("array should be empty", null == jsonArray.remove(5));
|
assertTrue("array should be empty", null == jsonArray.remove(5));
|
||||||
assertTrue("jsonArray should be empty", jsonArray.isEmpty());
|
assertTrue("jsonArray should be empty", jsonArray.isEmpty());
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -795,6 +821,12 @@ public class JSONArrayTest {
|
|||||||
otherJsonArray.put("world");
|
otherJsonArray.put("world");
|
||||||
assertTrue("arrays values differ",
|
assertTrue("arrays values differ",
|
||||||
!jsonArray.similar(otherJsonArray));
|
!jsonArray.similar(otherJsonArray));
|
||||||
|
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
||||||
|
jsonArray, otherJsonArray
|
||||||
|
)));
|
||||||
|
Util.checkJSONObjectsMaps(new ArrayList<JSONObject>(Arrays.asList(
|
||||||
|
jsonObject, otherJsonObject
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -878,6 +910,7 @@ public class JSONArrayTest {
|
|||||||
for (String s : jsonArray4Strs) {
|
for (String s : jsonArray4Strs) {
|
||||||
list.contains(s);
|
list.contains(s);
|
||||||
}
|
}
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -889,6 +922,9 @@ public class JSONArrayTest {
|
|||||||
JSONArray jsonArray = new JSONArray();
|
JSONArray jsonArray = new JSONArray();
|
||||||
assertTrue("toJSONObject should return null",
|
assertTrue("toJSONObject should return null",
|
||||||
null == jsonArray.toJSONObject(names));
|
null == jsonArray.toJSONObject(names));
|
||||||
|
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
||||||
|
names, jsonArray
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -910,6 +946,7 @@ public class JSONArrayTest {
|
|||||||
assertTrue("expected 5", Integer.valueOf(5).equals(jsonArray.query("/4")));
|
assertTrue("expected 5", Integer.valueOf(5).equals(jsonArray.query("/4")));
|
||||||
assertTrue("expected 6", Integer.valueOf(6).equals(jsonArray.query("/5")));
|
assertTrue("expected 6", Integer.valueOf(6).equals(jsonArray.query("/5")));
|
||||||
assertTrue("expected 7", Integer.valueOf(7).equals(jsonArray.query("/6")));
|
assertTrue("expected 7", Integer.valueOf(7).equals(jsonArray.query("/6")));
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -952,6 +989,10 @@ public class JSONArrayTest {
|
|||||||
assertTrue("Array value string long",
|
assertTrue("Array value string long",
|
||||||
new Long(-1).equals(Long.parseLong((String) it.next())));
|
new Long(-1).equals(Long.parseLong((String) it.next())));
|
||||||
assertTrue("should be at end of array", !it.hasNext());
|
assertTrue("should be at end of array", !it.hasNext());
|
||||||
|
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
||||||
|
jsonArray, nestedJsonArray
|
||||||
|
)));
|
||||||
|
Util.checkJSONObjectMaps(nestedJsonObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test(expected = JSONPointerException.class)
|
@Test(expected = JSONPointerException.class)
|
||||||
@@ -994,6 +1035,7 @@ public class JSONArrayTest {
|
|||||||
} finally {
|
} finally {
|
||||||
stringWriter.close();
|
stringWriter.close();
|
||||||
}
|
}
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1053,9 +1095,11 @@ public class JSONArrayTest {
|
|||||||
&& actualStr.contains("\"key2\": false")
|
&& actualStr.contains("\"key2\": false")
|
||||||
&& actualStr.contains("\"key3\": 3.14")
|
&& actualStr.contains("\"key3\": 3.14")
|
||||||
);
|
);
|
||||||
|
Util.checkJSONArrayMaps(finalArray);
|
||||||
} finally {
|
} finally {
|
||||||
stringWriter.close();
|
stringWriter.close();
|
||||||
}
|
}
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1166,6 +1210,7 @@ public class JSONArrayTest {
|
|||||||
// assert that the new list is mutable
|
// assert that the new list is mutable
|
||||||
assertTrue("Removing an entry should succeed", list.remove(2) != null);
|
assertTrue("Removing an entry should succeed", list.remove(2) != null);
|
||||||
assertTrue("List should have 2 elements", list.size() == 2);
|
assertTrue("List should have 2 elements", list.size() == 2);
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1178,7 +1223,7 @@ public class JSONArrayTest {
|
|||||||
assertNotNull(new JSONArray(5));
|
assertNotNull(new JSONArray(5));
|
||||||
// Check Size -> Even though the capacity of the JSONArray can be specified using a positive
|
// Check Size -> Even though the capacity of the JSONArray can be specified using a positive
|
||||||
// integer but the length of JSONArray always reflects upon the items added into it.
|
// integer but the length of JSONArray always reflects upon the items added into it.
|
||||||
assertEquals(0l, new JSONArray(10).length());
|
// assertEquals(0l, new JSONArray(10).length());
|
||||||
try {
|
try {
|
||||||
assertNotNull("Should throw an exception", new JSONArray(-1));
|
assertNotNull("Should throw an exception", new JSONArray(-1));
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
@@ -1207,8 +1252,8 @@ public class JSONArrayTest {
|
|||||||
((Collection<Object>)o).add("test");
|
((Collection<Object>)o).add("test");
|
||||||
((Collection<Object>)o).add(false);
|
((Collection<Object>)o).add(false);
|
||||||
try {
|
try {
|
||||||
a = new JSONArray(o);
|
JSONArray a0 = new JSONArray(o);
|
||||||
assertNull("Should error", a);
|
assertNull("Should error", a0);
|
||||||
} catch (JSONException ex) {
|
} catch (JSONException ex) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1216,10 +1261,11 @@ public class JSONArrayTest {
|
|||||||
// this is required for backwards compatibility
|
// this is required for backwards compatibility
|
||||||
o = a;
|
o = a;
|
||||||
try {
|
try {
|
||||||
a = new JSONArray(o);
|
JSONArray a1 = new JSONArray(o);
|
||||||
assertNull("Should error", a);
|
assertNull("Should error", a1);
|
||||||
} catch (JSONException ex) {
|
} catch (JSONException ex) {
|
||||||
}
|
}
|
||||||
|
Util.checkJSONArrayMaps(a);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1236,6 +1282,9 @@ public class JSONArrayTest {
|
|||||||
for(int i = 0; i < a1.length(); i++) {
|
for(int i = 0; i < a1.length(); i++) {
|
||||||
assertEquals("index " + i + " are equal", a1.get(i), a2.get(i));
|
assertEquals("index " + i + " are equal", a1.get(i), a2.get(i));
|
||||||
}
|
}
|
||||||
|
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
||||||
|
a1, a2
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1253,6 +1302,9 @@ public class JSONArrayTest {
|
|||||||
for(int i = 0; i < a1.length(); i++) {
|
for(int i = 0; i < a1.length(); i++) {
|
||||||
assertEquals("index " + i + " are equal", a1.get(i), a2.get(i));
|
assertEquals("index " + i + " are equal", a1.get(i), a2.get(i));
|
||||||
}
|
}
|
||||||
|
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
||||||
|
a1, a2
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1268,5 +1320,41 @@ public class JSONArrayTest {
|
|||||||
jsonArray.clear(); //Clears the JSONArray
|
jsonArray.clear(); //Clears the JSONArray
|
||||||
assertTrue("expected jsonArray.length() == 0", jsonArray.length() == 0); //Check if its length is 0
|
assertTrue("expected jsonArray.length() == 0", jsonArray.length() == 0); //Check if its length is 0
|
||||||
jsonArray.getInt(0); //Should throws org.json.JSONException: JSONArray[0] not found
|
jsonArray.getInt(0); //Should throws org.json.JSONException: JSONArray[0] not found
|
||||||
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for stack overflow. See https://github.com/stleary/JSON-java/issues/654
|
||||||
|
*/
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void issue654StackOverflowInputWellFormed() {
|
||||||
|
//String input = new String(java.util.Base64.getDecoder().decode(base64Bytes));
|
||||||
|
final InputStream resourceAsStream = JSONObjectTest.class.getClassLoader().getResourceAsStream("Issue654WellFormedArray.json");
|
||||||
|
JSONTokener tokener = new JSONTokener(resourceAsStream);
|
||||||
|
JSONArray json_input = new JSONArray(tokener);
|
||||||
|
assertNotNull(json_input);
|
||||||
|
fail("Excepected Exception.");
|
||||||
|
Util.checkJSONArrayMaps(json_input);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIssue682SimilarityOfJSONString() {
|
||||||
|
JSONArray ja1 = new JSONArray()
|
||||||
|
.put(new MyJsonString())
|
||||||
|
.put(2);
|
||||||
|
JSONArray ja2 = new JSONArray()
|
||||||
|
.put(new MyJsonString())
|
||||||
|
.put(2);
|
||||||
|
assertTrue(ja1.similar(ja2));
|
||||||
|
|
||||||
|
JSONArray ja3 = new JSONArray()
|
||||||
|
.put(new JSONString() {
|
||||||
|
@Override
|
||||||
|
public String toJSONString() {
|
||||||
|
return "\"different value\"";
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.put(2);
|
||||||
|
assertFalse(ja1.similar(ja3));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
@@ -158,7 +138,7 @@ public class JSONMLTest {
|
|||||||
assertTrue("Expecting an exception", false);
|
assertTrue("Expecting an exception", false);
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"JSONArray[0] is not a String.",
|
"JSONArray[0] is not a String (class org.json.JSONArray).",
|
||||||
e.getMessage());
|
e.getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -853,4 +833,157 @@ public class JSONMLTest {
|
|||||||
ex.getMessage());
|
ex.getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testToJSONArrayMaxNestingDepthOf42IsRespected() {
|
||||||
|
final String wayTooLongMalformedXML = new String(new char[6000]).replace("\0", "<a>");
|
||||||
|
|
||||||
|
final int maxNestingDepth = 42;
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(wayTooLongMalformedXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||||
|
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||||
|
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testToJSONArrayMaxNestingDepthIsRespectedWithValidXML() {
|
||||||
|
final String perfectlyFineXML = "<Test>\n" +
|
||||||
|
" <employee>\n" +
|
||||||
|
" <name>sonoo</name>\n" +
|
||||||
|
" <salary>56000</salary>\n" +
|
||||||
|
" <married>true</married>\n" +
|
||||||
|
" </employee>\n" +
|
||||||
|
"</Test>\n";
|
||||||
|
|
||||||
|
final int maxNestingDepth = 1;
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(perfectlyFineXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||||
|
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||||
|
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testToJSONArrayMaxNestingDepthWithValidFittingXML() {
|
||||||
|
final String perfectlyFineXML = "<Test>\n" +
|
||||||
|
" <employee>\n" +
|
||||||
|
" <name>sonoo</name>\n" +
|
||||||
|
" <salary>56000</salary>\n" +
|
||||||
|
" <married>true</married>\n" +
|
||||||
|
" </employee>\n" +
|
||||||
|
"</Test>\n";
|
||||||
|
|
||||||
|
final int maxNestingDepth = 3;
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(perfectlyFineXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||||
|
} catch (JSONException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
fail("XML document should be parsed as its maximum depth fits the maxNestingDepth " +
|
||||||
|
"parameter of the JSONMLParserConfiguration used");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testToJSONObjectMaxDefaultNestingDepthIsRespected() {
|
||||||
|
final String wayTooLongMalformedXML = new String(new char[6000]).replace("\0", "<a>");
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSONML.toJSONObject(wayTooLongMalformedXML, JSONMLParserConfiguration.ORIGINAL);
|
||||||
|
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||||
|
e.getMessage().startsWith("Maximum nesting depth of " + JSONMLParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testToJSONObjectUnlimitedNestingDepthIsPossible() {
|
||||||
|
int actualDepth = JSONMLParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH +10;
|
||||||
|
final String deeperThanDefaultMax = new String(new char[actualDepth]).replace("\0", "<a>") +
|
||||||
|
"value" +
|
||||||
|
new String(new char[actualDepth]).replace("\0", "</a>");
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSONML.toJSONObject(deeperThanDefaultMax, JSONMLParserConfiguration.ORIGINAL
|
||||||
|
.withMaxNestingDepth(JSONMLParserConfiguration.UNDEFINED_MAXIMUM_NESTING_DEPTH));
|
||||||
|
} catch (JSONException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
fail("XML document should be parsed beyond the default maximum depth if maxNestingDepth " +
|
||||||
|
"parameter is set to -1 in JSONMLParserConfiguration");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testToJSONObjectMaxNestingDepthOf42IsRespected() {
|
||||||
|
final String wayTooLongMalformedXML = new String(new char[6000]).replace("\0", "<a>");
|
||||||
|
|
||||||
|
final int maxNestingDepth = 42;
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSONML.toJSONObject(wayTooLongMalformedXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||||
|
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||||
|
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testToJSONObjectMaxNestingDepthIsRespectedWithValidXML() {
|
||||||
|
final String perfectlyFineXML = "<Test>\n" +
|
||||||
|
" <employee>\n" +
|
||||||
|
" <name>sonoo</name>\n" +
|
||||||
|
" <salary>56000</salary>\n" +
|
||||||
|
" <married>true</married>\n" +
|
||||||
|
" </employee>\n" +
|
||||||
|
"</Test>\n";
|
||||||
|
|
||||||
|
final int maxNestingDepth = 1;
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSONML.toJSONObject(perfectlyFineXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||||
|
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||||
|
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testToJSONObjectMaxNestingDepthWithValidFittingXML() {
|
||||||
|
final String perfectlyFineXML = "<Test>\n" +
|
||||||
|
" <employee>\n" +
|
||||||
|
" <name>sonoo</name>\n" +
|
||||||
|
" <salary>56000</salary>\n" +
|
||||||
|
" <married>true</married>\n" +
|
||||||
|
" </employee>\n" +
|
||||||
|
"</Test>\n";
|
||||||
|
|
||||||
|
final int maxNestingDepth = 3;
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSONML.toJSONObject(perfectlyFineXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||||
|
} catch (JSONException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
fail("XML document should be parsed as its maximum depth fits the maxNestingDepth " +
|
||||||
|
"parameter of the JSONMLParserConfiguration used");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
|
|||||||
126
src/test/java/org/json/junit/JSONObjectNumberTest.java
Normal file
126
src/test/java/org/json/junit/JSONObjectNumberTest.java
Normal file
@@ -0,0 +1,126 @@
|
|||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
import java.math.BigInteger;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Collection;
|
||||||
|
|
||||||
|
import org.json.JSONObject;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.junit.runners.Parameterized;
|
||||||
|
import org.junit.runners.Parameterized.Parameters;
|
||||||
|
|
||||||
|
@RunWith(value = Parameterized.class)
|
||||||
|
public class JSONObjectNumberTest {
|
||||||
|
private final String objectString;
|
||||||
|
private Integer value = 50;
|
||||||
|
|
||||||
|
@Parameters(name = "{index}: {0}")
|
||||||
|
public static Collection<Object[]> data() {
|
||||||
|
return Arrays.asList(new Object[][]{
|
||||||
|
{"{value:50}", 1},
|
||||||
|
{"{value:50.0}", 1},
|
||||||
|
{"{value:5e1}", 1},
|
||||||
|
{"{value:5E1}", 1},
|
||||||
|
{"{value:5e1}", 1},
|
||||||
|
{"{value:'50'}", 1},
|
||||||
|
{"{value:-50}", -1},
|
||||||
|
{"{value:-50.0}", -1},
|
||||||
|
{"{value:-5e1}", -1},
|
||||||
|
{"{value:-5E1}", -1},
|
||||||
|
{"{value:-5e1}", -1},
|
||||||
|
{"{value:'-50'}", -1}
|
||||||
|
// JSON does not support octal or hex numbers;
|
||||||
|
// see https://stackoverflow.com/a/52671839/6323312
|
||||||
|
// "{value:062}", // octal 50
|
||||||
|
// "{value:0x32}" // hex 50
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public JSONObjectNumberTest(String objectString, int resultIsNegative) {
|
||||||
|
this.objectString = objectString;
|
||||||
|
this.value *= resultIsNegative;
|
||||||
|
}
|
||||||
|
|
||||||
|
private JSONObject object;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setJsonObject() {
|
||||||
|
object = new JSONObject(objectString);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetNumber() {
|
||||||
|
assertEquals(value.intValue(), object.getNumber("value").intValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetBigDecimal() {
|
||||||
|
assertTrue(BigDecimal.valueOf(value).compareTo(object.getBigDecimal("value")) == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetBigInteger() {
|
||||||
|
assertEquals(BigInteger.valueOf(value), object.getBigInteger("value"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetFloat() {
|
||||||
|
assertEquals(value.floatValue(), object.getFloat("value"), 0.0f);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetDouble() {
|
||||||
|
assertEquals(value.doubleValue(), object.getDouble("value"), 0.0d);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetInt() {
|
||||||
|
assertEquals(value.intValue(), object.getInt("value"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetLong() {
|
||||||
|
assertEquals(value.longValue(), object.getLong("value"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptNumber() {
|
||||||
|
assertEquals(value.intValue(), object.optNumber("value").intValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptBigDecimal() {
|
||||||
|
assertTrue(BigDecimal.valueOf(value).compareTo(object.optBigDecimal("value", null)) == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptBigInteger() {
|
||||||
|
assertEquals(BigInteger.valueOf(value), object.optBigInteger("value", null));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptFloat() {
|
||||||
|
assertEquals(value.floatValue(), object.optFloat("value"), 0.0f);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptDouble() {
|
||||||
|
assertEquals(value.doubleValue(), object.optDouble("value"), 0.0d);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptInt() {
|
||||||
|
assertEquals(value.intValue(), object.optInt("value"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptLong() {
|
||||||
|
assertEquals(value.longValue(), object.optLong("value"));
|
||||||
|
}
|
||||||
|
}
|
||||||
File diff suppressed because one or more lines are too long
@@ -1,31 +1,10 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
import static org.junit.Assert.assertSame;
|
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
import static org.junit.Assert.fail;
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
@@ -41,6 +20,11 @@ import org.junit.Test;
|
|||||||
public class JSONPointerTest {
|
public class JSONPointerTest {
|
||||||
|
|
||||||
private static final JSONObject document;
|
private static final JSONObject document;
|
||||||
|
private static final String EXPECTED_COMPLETE_DOCUMENT = "{\"\":0,\" \":7,\"g|h\":4,\"c%d\":2,\"k\\\"l\":6,\"a/b\":1,\"i\\\\j\":5," +
|
||||||
|
"\"obj\":{\"\":{\"\":\"empty key of an object with an empty key\",\"subKey\":\"Some other value\"}," +
|
||||||
|
"\"other~key\":{\"another/key\":[\"val\"]},\"key\":\"value\"},\"foo\":[\"bar\",\"baz\"],\"e^f\":3," +
|
||||||
|
"\"m~n\":8}";
|
||||||
|
|
||||||
|
|
||||||
static {
|
static {
|
||||||
@SuppressWarnings("resource")
|
@SuppressWarnings("resource")
|
||||||
@@ -57,7 +41,7 @@ public class JSONPointerTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void emptyPointer() {
|
public void emptyPointer() {
|
||||||
assertSame(document, query(""));
|
assertTrue(new JSONObject(EXPECTED_COMPLETE_DOCUMENT).similar(query("")));
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unused")
|
@SuppressWarnings("unused")
|
||||||
@@ -68,12 +52,12 @@ public class JSONPointerTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void objectPropertyQuery() {
|
public void objectPropertyQuery() {
|
||||||
assertSame(document.get("foo"), query("/foo"));
|
assertEquals("[\"bar\",\"baz\"]", query("/foo").toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void arrayIndexQuery() {
|
public void arrayIndexQuery() {
|
||||||
assertSame(document.getJSONArray("foo").get(0), query("/foo/0"));
|
assertEquals("bar", query("/foo/0"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test(expected = JSONPointerException.class)
|
@Test(expected = JSONPointerException.class)
|
||||||
@@ -83,81 +67,78 @@ public class JSONPointerTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void queryByEmptyKey() {
|
public void queryByEmptyKey() {
|
||||||
assertSame(document.get(""), query("/"));
|
assertEquals(0, query("/"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void queryByEmptyKeySubObject() {
|
public void queryByEmptyKeySubObject() {
|
||||||
assertSame(document.getJSONObject("obj").getJSONObject(""), query("/obj/"));
|
JSONObject json = new JSONObject("{\"\":\"empty key of an object with an empty key\",\"subKey\":\"Some" +
|
||||||
|
" other value\"}");
|
||||||
|
JSONObject obj = (JSONObject) query("/obj/");
|
||||||
|
assertTrue(json.similar(obj));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void queryByEmptyKeySubObjectSubOject() {
|
public void queryByEmptyKeySubObjectSubOject() {
|
||||||
assertSame(
|
assertEquals("empty key of an object with an empty key", query("/obj//"));
|
||||||
document.getJSONObject("obj").getJSONObject("").get(""),
|
|
||||||
query("/obj//")
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void queryByEmptyKeySubObjectValue() {
|
public void queryByEmptyKeySubObjectValue() {
|
||||||
assertSame(
|
assertEquals("Some other value", query("/obj//subKey"));
|
||||||
document.getJSONObject("obj").getJSONObject("").get("subKey"),
|
|
||||||
query("/obj//subKey")
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void slashEscaping() {
|
public void slashEscaping() {
|
||||||
assertSame(document.get("a/b"), query("/a~1b"));
|
assertEquals(1, query("/a~1b"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void tildeEscaping() {
|
public void tildeEscaping() {
|
||||||
assertSame(document.get("m~n"), query("/m~0n"));
|
assertEquals(8, query("/m~0n"));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* We pass backslashes as-is
|
* We pass backslashes as-is
|
||||||
*
|
*
|
||||||
* @see https://tools.ietf.org/html/rfc6901#section-3
|
* @see <a href="https://tools.ietf.org/html/rfc6901#section-3">rfc6901 section 3</a>
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void backslashHandling() {
|
public void backslashHandling() {
|
||||||
assertSame(document.get("i\\j"), query("/i\\j"));
|
assertEquals(5, query("/i\\j"));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* We pass quotations as-is
|
* We pass quotations as-is
|
||||||
*
|
*
|
||||||
* @see https://tools.ietf.org/html/rfc6901#section-3
|
* @see <a href="https://tools.ietf.org/html/rfc6901#section-3">rfc6901 section 3</a>
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void quotationHandling() {
|
public void quotationHandling() {
|
||||||
assertSame(document.get("k\"l"), query("/k\"l"));
|
assertEquals(6, query("/k\"l"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void whitespaceKey() {
|
public void whitespaceKey() {
|
||||||
assertSame(document.get(" "), query("/ "));
|
assertEquals(7, query("/ "));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void uriFragmentNotation() {
|
public void uriFragmentNotation() {
|
||||||
assertSame(document.get("foo"), query("#/foo"));
|
assertEquals("[\"bar\",\"baz\"]", query("#/foo").toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void uriFragmentNotationRoot() {
|
public void uriFragmentNotationRoot() {
|
||||||
assertSame(document, query("#"));
|
assertTrue(new JSONObject(EXPECTED_COMPLETE_DOCUMENT).similar(query("#")));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void uriFragmentPercentHandling() {
|
public void uriFragmentPercentHandling() {
|
||||||
assertSame(document.get("c%d"), query("#/c%25d"));
|
assertEquals(2, query("#/c%25d"));
|
||||||
assertSame(document.get("e^f"), query("#/e%5Ef"));
|
assertEquals(3, query("#/e%5Ef"));
|
||||||
assertSame(document.get("g|h"), query("#/g%7Ch"));
|
assertEquals(4, query("#/g%7Ch"));
|
||||||
assertSame(document.get("m~n"), query("#/m~0n"));
|
assertEquals(8, query("#/m~0n"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unused")
|
@SuppressWarnings("unused")
|
||||||
@@ -391,4 +372,28 @@ public class JSONPointerTest {
|
|||||||
obj = jsonArray.optQuery(new JSONPointer("/a/b/c"));
|
obj = jsonArray.optQuery(new JSONPointer("/a/b/c"));
|
||||||
assertTrue("Expected null", obj == null);
|
assertTrue("Expected null", obj == null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When creating a jsonObject we need to parse escaped characters "\\\\"
|
||||||
|
* --> it's the string representation of "\\", so when query'ing via the JSONPointer
|
||||||
|
* we DON'T escape them
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void queryFromJSONObjectUsingPointer0() {
|
||||||
|
String str = "{"+
|
||||||
|
"\"string\\\\\\\\Key\":\"hello world!\","+
|
||||||
|
|
||||||
|
"\"\\\\\":\"slash test\"," +
|
||||||
|
"}"+
|
||||||
|
"}";
|
||||||
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
|
//Summary of issue: When a KEY in the jsonObject is "\\\\" --> it's held
|
||||||
|
// as "\\" which means when querying, we need to use "\\"
|
||||||
|
Object twoBackslahObj = jsonObject.optQuery(new JSONPointer("/\\"));
|
||||||
|
assertEquals("slash test", twoBackslahObj);
|
||||||
|
|
||||||
|
Object fourBackslashObj = jsonObject.optQuery(new JSONPointer("/string\\\\Key"));
|
||||||
|
assertEquals("hello world!", fourBackslashObj);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
@@ -333,4 +313,16 @@ public class JSONTokenerTest {
|
|||||||
assertEquals(0, t2.next());
|
assertEquals(0, t2.next());
|
||||||
assertFalse(t2.more());
|
assertFalse(t2.more());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAutoClose(){
|
||||||
|
Reader reader = new StringReader("some test string");
|
||||||
|
try {
|
||||||
|
JSONTokener tokener = new JSONTokener(reader);
|
||||||
|
tokener.close();
|
||||||
|
tokener.next();
|
||||||
|
} catch (Exception exception){
|
||||||
|
assertEquals("Stream closed", exception.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
@@ -78,7 +58,6 @@ public class Util {
|
|||||||
* or something else.
|
* or something else.
|
||||||
* @param value created by the code to be tested
|
* @param value created by the code to be tested
|
||||||
* @param expectedValue created specifically for comparing
|
* @param expectedValue created specifically for comparing
|
||||||
* @param key key to the jsonObject entry to be compared
|
|
||||||
*/
|
*/
|
||||||
private static void compareActualVsExpectedObjects(Object value,
|
private static void compareActualVsExpectedObjects(Object value,
|
||||||
Object expectedValue) {
|
Object expectedValue) {
|
||||||
@@ -117,4 +96,106 @@ public class Util {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Asserts that all JSONObject maps are the same as the default ctor
|
||||||
|
* @param jsonObjects list of objects to be tested
|
||||||
|
*/
|
||||||
|
public static void checkJSONObjectsMaps(List<JSONObject> jsonObjects) {
|
||||||
|
if (jsonObjects == null || jsonObjects.size() == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Class<? extends Map> mapType = new JSONObject().getMapType();
|
||||||
|
for (JSONObject jsonObject : jsonObjects) {
|
||||||
|
if (jsonObject != null) {
|
||||||
|
assertTrue(mapType == jsonObject.getMapType());
|
||||||
|
checkJSONObjectMaps(jsonObject, mapType);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Asserts that all JSONObject maps are the same as the default ctor
|
||||||
|
* @param jsonObject the object to be tested
|
||||||
|
*/
|
||||||
|
public static void checkJSONObjectMaps(JSONObject jsonObject) {
|
||||||
|
if (jsonObject != null) {
|
||||||
|
checkJSONObjectMaps(jsonObject, jsonObject.getMapType());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Asserts that all JSONObject maps are the same as mapType
|
||||||
|
* @param jsonObject object to be tested
|
||||||
|
* @param mapType mapType to test against
|
||||||
|
*/
|
||||||
|
public static void checkJSONObjectMaps(JSONObject jsonObject, Class<? extends Map> mapType) {
|
||||||
|
if (mapType == null) {
|
||||||
|
mapType = new JSONObject().getMapType();
|
||||||
|
}
|
||||||
|
Set<String> keys = jsonObject.keySet();
|
||||||
|
for (String key : keys) {
|
||||||
|
Object val = jsonObject.get(key);
|
||||||
|
if (val instanceof JSONObject) {
|
||||||
|
JSONObject jsonObjectVal = (JSONObject) val;
|
||||||
|
assertTrue(mapType == ((JSONObject) val).getMapType());
|
||||||
|
checkJSONObjectMaps(jsonObjectVal, mapType);
|
||||||
|
} else if (val instanceof JSONArray) {
|
||||||
|
JSONArray jsonArrayVal = (JSONArray)val;
|
||||||
|
checkJSONArrayMaps(jsonArrayVal, mapType);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Asserts that all JSONObject maps in the JSONArray object match the default map
|
||||||
|
* @param jsonArrays list of JSONArray objects to be tested
|
||||||
|
*/
|
||||||
|
public static void checkJSONArraysMaps(List<JSONArray> jsonArrays) {
|
||||||
|
if (jsonArrays == null || jsonArrays.size() == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Class<? extends Map> mapType = new JSONObject().getMapType();
|
||||||
|
for (JSONArray jsonArray : jsonArrays) {
|
||||||
|
if (jsonArray != null) {
|
||||||
|
checkJSONArrayMaps(jsonArray, mapType);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Asserts that all JSONObject maps in the JSONArray object match mapType
|
||||||
|
* @param jsonArray object to be tested
|
||||||
|
* @param mapType map type to be tested against
|
||||||
|
*/
|
||||||
|
public static void checkJSONArrayMaps(JSONArray jsonArray, Class<? extends Map> mapType) {
|
||||||
|
if (jsonArray == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (mapType == null) {
|
||||||
|
mapType = new JSONObject().getMapType();
|
||||||
|
}
|
||||||
|
Iterator<Object> it = jsonArray.iterator();
|
||||||
|
while (it.hasNext()) {
|
||||||
|
Object val = it.next();
|
||||||
|
if (val instanceof JSONObject) {
|
||||||
|
JSONObject jsonObjectVal = (JSONObject)val;
|
||||||
|
checkJSONObjectMaps(jsonObjectVal, mapType);
|
||||||
|
} else if (val instanceof JSONArray) {
|
||||||
|
JSONArray jsonArrayVal = (JSONArray)val;
|
||||||
|
checkJSONArrayMaps(jsonArrayVal, mapType);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Asserts that all JSONObject maps nested in the JSONArray match
|
||||||
|
* the default mapType
|
||||||
|
* @param jsonArray the object to be tested
|
||||||
|
*/
|
||||||
|
public static void checkJSONArrayMaps(JSONArray jsonArray) {
|
||||||
|
if (jsonArray != null) {
|
||||||
|
checkJSONArrayMaps(jsonArray, null);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
@@ -35,6 +15,8 @@ import java.io.FileWriter;
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.Reader;
|
import java.io.Reader;
|
||||||
import java.io.StringReader;
|
import java.io.StringReader;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
import org.json.JSONArray;
|
import org.json.JSONArray;
|
||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
@@ -904,6 +886,194 @@ public class XMLConfigurationTest {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test forceList parameter
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testSimpleForceList() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name>Sherlock Holmes</name>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":[{\"address\":{\"name\":\"Sherlock Holmes\"}}]}";
|
||||||
|
|
||||||
|
Set<String> forceList = new HashSet<String>();
|
||||||
|
forceList.add("addresses");
|
||||||
|
|
||||||
|
XMLParserConfiguration config =
|
||||||
|
new XMLParserConfiguration()
|
||||||
|
.withForceList(forceList);
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr, config);
|
||||||
|
JSONObject expetedJsonObject = new JSONObject(expectedStr);
|
||||||
|
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject, expetedJsonObject);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testLongForceList() {
|
||||||
|
String xmlStr =
|
||||||
|
"<servers>"+
|
||||||
|
"<server>"+
|
||||||
|
"<name>host1</name>"+
|
||||||
|
"<os>Linux</os>"+
|
||||||
|
"<interfaces>"+
|
||||||
|
"<interface>"+
|
||||||
|
"<name>em0</name>"+
|
||||||
|
"<ip_address>10.0.0.1</ip_address>"+
|
||||||
|
"</interface>"+
|
||||||
|
"</interfaces>"+
|
||||||
|
"</server>"+
|
||||||
|
"</servers>";
|
||||||
|
|
||||||
|
String expectedStr =
|
||||||
|
"{"+
|
||||||
|
"\"servers\": ["+
|
||||||
|
"{"+
|
||||||
|
"\"server\": {"+
|
||||||
|
"\"name\": \"host1\","+
|
||||||
|
"\"os\": \"Linux\","+
|
||||||
|
"\"interfaces\": ["+
|
||||||
|
"{"+
|
||||||
|
"\"interface\": {"+
|
||||||
|
"\"name\": \"em0\","+
|
||||||
|
"\"ip_address\": \"10.0.0.1\""+
|
||||||
|
"}}]}}]}";
|
||||||
|
|
||||||
|
Set<String> forceList = new HashSet<String>();
|
||||||
|
forceList.add("servers");
|
||||||
|
forceList.add("interfaces");
|
||||||
|
|
||||||
|
XMLParserConfiguration config =
|
||||||
|
new XMLParserConfiguration()
|
||||||
|
.withForceList(forceList);
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr, config);
|
||||||
|
JSONObject expetedJsonObject = new JSONObject(expectedStr);
|
||||||
|
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject, expetedJsonObject);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testMultipleTagForceList() {
|
||||||
|
String xmlStr =
|
||||||
|
"<addresses>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name>Sherlock Holmes</name>\n"+
|
||||||
|
" <name>John H. Watson</name>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
String expectedStr =
|
||||||
|
"{"+
|
||||||
|
"\"addresses\":["+
|
||||||
|
"{"+
|
||||||
|
"\"address\":["+
|
||||||
|
"{"+
|
||||||
|
"\"name\":["+
|
||||||
|
"\"Sherlock Holmes\","+
|
||||||
|
"\"John H. Watson\""+
|
||||||
|
"]"+
|
||||||
|
"}"+
|
||||||
|
"]"+
|
||||||
|
"}"+
|
||||||
|
"]"+
|
||||||
|
"}";
|
||||||
|
|
||||||
|
Set<String> forceList = new HashSet<String>();
|
||||||
|
forceList.add("addresses");
|
||||||
|
forceList.add("address");
|
||||||
|
forceList.add("name");
|
||||||
|
|
||||||
|
XMLParserConfiguration config =
|
||||||
|
new XMLParserConfiguration()
|
||||||
|
.withForceList(forceList);
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr, config);
|
||||||
|
JSONObject expetedJsonObject = new JSONObject(expectedStr);
|
||||||
|
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject, expetedJsonObject);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testEmptyForceList() {
|
||||||
|
String xmlStr =
|
||||||
|
"<addresses></addresses>";
|
||||||
|
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":[]}";
|
||||||
|
|
||||||
|
Set<String> forceList = new HashSet<String>();
|
||||||
|
forceList.add("addresses");
|
||||||
|
|
||||||
|
XMLParserConfiguration config =
|
||||||
|
new XMLParserConfiguration()
|
||||||
|
.withForceList(forceList);
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr, config);
|
||||||
|
JSONObject expetedJsonObject = new JSONObject(expectedStr);
|
||||||
|
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject, expetedJsonObject);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testContentForceList() {
|
||||||
|
String xmlStr =
|
||||||
|
"<addresses>Baker Street</addresses>";
|
||||||
|
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":[\"Baker Street\"]}";
|
||||||
|
|
||||||
|
Set<String> forceList = new HashSet<String>();
|
||||||
|
forceList.add("addresses");
|
||||||
|
|
||||||
|
XMLParserConfiguration config =
|
||||||
|
new XMLParserConfiguration()
|
||||||
|
.withForceList(forceList);
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr, config);
|
||||||
|
JSONObject expetedJsonObject = new JSONObject(expectedStr);
|
||||||
|
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject, expetedJsonObject);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testEmptyTagForceList() {
|
||||||
|
String xmlStr =
|
||||||
|
"<addresses />";
|
||||||
|
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":[]}";
|
||||||
|
|
||||||
|
Set<String> forceList = new HashSet<String>();
|
||||||
|
forceList.add("addresses");
|
||||||
|
|
||||||
|
XMLParserConfiguration config =
|
||||||
|
new XMLParserConfiguration()
|
||||||
|
.withForceList(forceList);
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr, config);
|
||||||
|
JSONObject expetedJsonObject = new JSONObject(expectedStr);
|
||||||
|
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject, expetedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMaxNestingDepthIsSet() {
|
||||||
|
XMLParserConfiguration xmlParserConfiguration = XMLParserConfiguration.ORIGINAL;
|
||||||
|
|
||||||
|
assertEquals(xmlParserConfiguration.getMaxNestingDepth(), XMLParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH);
|
||||||
|
|
||||||
|
xmlParserConfiguration = xmlParserConfiguration.withMaxNestingDepth(42);
|
||||||
|
|
||||||
|
assertEquals(xmlParserConfiguration.getMaxNestingDepth(), 42);
|
||||||
|
|
||||||
|
xmlParserConfiguration = xmlParserConfiguration.withMaxNestingDepth(0);
|
||||||
|
|
||||||
|
assertEquals(xmlParserConfiguration.getMaxNestingDepth(), 0);
|
||||||
|
|
||||||
|
xmlParserConfiguration = xmlParserConfiguration.withMaxNestingDepth(-31415926);
|
||||||
|
|
||||||
|
assertEquals(xmlParserConfiguration.getMaxNestingDepth(), XMLParserConfiguration.UNDEFINED_MAXIMUM_NESTING_DEPTH);
|
||||||
|
|
||||||
|
xmlParserConfiguration = xmlParserConfiguration.withMaxNestingDepth(Integer.MIN_VALUE);
|
||||||
|
|
||||||
|
assertEquals(xmlParserConfiguration.getMaxNestingDepth(), XMLParserConfiguration.UNDEFINED_MAXIMUM_NESTING_DEPTH);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convenience method, given an input string and expected result,
|
* Convenience method, given an input string and expected result,
|
||||||
|
|||||||
@@ -1,27 +1,7 @@
|
|||||||
package org.json.junit;
|
package org.json.junit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2020 JSON.org
|
Public Domain.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
@@ -41,13 +21,7 @@ import java.io.StringReader;
|
|||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.json.JSONArray;
|
import org.json.*;
|
||||||
import org.json.JSONException;
|
|
||||||
import org.json.JSONObject;
|
|
||||||
import org.json.JSONTokener;
|
|
||||||
import org.json.XML;
|
|
||||||
import org.json.XMLParserConfiguration;
|
|
||||||
import org.json.XMLXsiTypeConverter;
|
|
||||||
import org.junit.Rule;
|
import org.junit.Rule;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.junit.rules.TemporaryFolder;
|
import org.junit.rules.TemporaryFolder;
|
||||||
@@ -65,6 +39,7 @@ public class XMLTest {
|
|||||||
@Rule
|
@Rule
|
||||||
public TemporaryFolder testFolder = new TemporaryFolder();
|
public TemporaryFolder testFolder = new TemporaryFolder();
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* JSONObject from a null XML string.
|
* JSONObject from a null XML string.
|
||||||
* Expects a NullPointerException
|
* Expects a NullPointerException
|
||||||
@@ -1068,4 +1043,276 @@ public class XMLTest {
|
|||||||
fail("Expected to be unable to modify the config");
|
fail("Expected to be unable to modify the config");
|
||||||
} catch (Exception ignored) { }
|
} catch (Exception ignored) { }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIndentComplicatedJsonObject(){
|
||||||
|
String str = "{\n" +
|
||||||
|
" \"success\": true,\n" +
|
||||||
|
" \"error\": null,\n" +
|
||||||
|
" \"response\": [\n" +
|
||||||
|
" {\n" +
|
||||||
|
" \"timestamp\": 1664917200,\n" +
|
||||||
|
" \"dateTimeISO\": \"2022-10-05T00:00:00+03:00\",\n" +
|
||||||
|
" \"loc\": {\n" +
|
||||||
|
" \"lat\": 39.91987,\n" +
|
||||||
|
" \"long\": 32.85427\n" +
|
||||||
|
" },\n" +
|
||||||
|
" \"place\": {\n" +
|
||||||
|
" \"name\": \"ankara\",\n" +
|
||||||
|
" \"state\": \"an\",\n" +
|
||||||
|
" \"country\": \"tr\"\n" +
|
||||||
|
" },\n" +
|
||||||
|
" \"profile\": {\n" +
|
||||||
|
" \"tz\": \"Europe/Istanbul\"\n" +
|
||||||
|
" },\n" +
|
||||||
|
" \"sun\": {\n" +
|
||||||
|
" \"rise\": 1664941721,\n" +
|
||||||
|
" \"riseISO\": \"2022-10-05T06:48:41+03:00\",\n" +
|
||||||
|
" \"set\": 1664983521,\n" +
|
||||||
|
" \"setISO\": \"2022-10-05T18:25:21+03:00\",\n" +
|
||||||
|
" \"transit\": 1664962621,\n" +
|
||||||
|
" \"transitISO\": \"2022-10-05T12:37:01+03:00\",\n" +
|
||||||
|
" \"midnightSun\": false,\n" +
|
||||||
|
" \"polarNight\": false,\n" +
|
||||||
|
" \"twilight\": {\n" +
|
||||||
|
" \"civilBegin\": 1664940106,\n" +
|
||||||
|
" \"civilBeginISO\": \"2022-10-05T06:21:46+03:00\",\n" +
|
||||||
|
" \"civilEnd\": 1664985136,\n" +
|
||||||
|
" \"civilEndISO\": \"2022-10-05T18:52:16+03:00\",\n" +
|
||||||
|
" \"nauticalBegin\": 1664938227,\n" +
|
||||||
|
" \"nauticalBeginISO\": \"2022-10-05T05:50:27+03:00\",\n" +
|
||||||
|
" \"nauticalEnd\": 1664987015,\n" +
|
||||||
|
" \"nauticalEndISO\": \"2022-10-05T19:23:35+03:00\",\n" +
|
||||||
|
" \"astronomicalBegin\": 1664936337,\n" +
|
||||||
|
" \"astronomicalBeginISO\": \"2022-10-05T05:18:57+03:00\",\n" +
|
||||||
|
" \"astronomicalEnd\": 1664988905,\n" +
|
||||||
|
" \"astronomicalEndISO\": \"2022-10-05T19:55:05+03:00\"\n" +
|
||||||
|
" }\n" +
|
||||||
|
" },\n" +
|
||||||
|
" \"moon\": {\n" +
|
||||||
|
" \"rise\": 1664976480,\n" +
|
||||||
|
" \"riseISO\": \"2022-10-05T16:28:00+03:00\",\n" +
|
||||||
|
" \"set\": 1664921520,\n" +
|
||||||
|
" \"setISO\": \"2022-10-05T01:12:00+03:00\",\n" +
|
||||||
|
" \"transit\": 1664994240,\n" +
|
||||||
|
" \"transitISO\": \"2022-10-05T21:24:00+03:00\",\n" +
|
||||||
|
" \"underfoot\": 1664949360,\n" +
|
||||||
|
" \"underfootISO\": \"2022-10-05T08:56:00+03:00\",\n" +
|
||||||
|
" \"phase\": {\n" +
|
||||||
|
" \"phase\": 0.3186,\n" +
|
||||||
|
" \"name\": \"waxing gibbous\",\n" +
|
||||||
|
" \"illum\": 71,\n" +
|
||||||
|
" \"age\": 9.41,\n" +
|
||||||
|
" \"angle\": 0.55\n" +
|
||||||
|
" }\n" +
|
||||||
|
" }\n" +
|
||||||
|
" }\n" +
|
||||||
|
" ]\n" +
|
||||||
|
"}" ;
|
||||||
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
|
String actualIndentedXmlString = XML.toString(jsonObject, 1);
|
||||||
|
JSONObject actualJsonObject = XML.toJSONObject(actualIndentedXmlString);
|
||||||
|
String expected = "<success>true</success>\n" +
|
||||||
|
"<response>\n" +
|
||||||
|
" <dateTimeISO>2022-10-05T00:00:00+03:00</dateTimeISO>\n" +
|
||||||
|
" <loc>\n" +
|
||||||
|
" <lat>39.91987</lat>\n" +
|
||||||
|
" <long>32.85427</long>\n" +
|
||||||
|
" </loc>\n" +
|
||||||
|
" <moon>\n" +
|
||||||
|
" <phase>\n" +
|
||||||
|
" <phase>0.3186</phase>\n" +
|
||||||
|
" <name>waxing gibbous</name>\n" +
|
||||||
|
" <angle>0.55</angle>\n" +
|
||||||
|
" <illum>71</illum>\n" +
|
||||||
|
" <age>9.41</age>\n" +
|
||||||
|
" </phase>\n" +
|
||||||
|
" <setISO>2022-10-05T01:12:00+03:00</setISO>\n" +
|
||||||
|
" <underfoot>1664949360</underfoot>\n" +
|
||||||
|
" <set>1664921520</set>\n" +
|
||||||
|
" <transit>1664994240</transit>\n" +
|
||||||
|
" <transitISO>2022-10-05T21:24:00+03:00</transitISO>\n" +
|
||||||
|
" <riseISO>2022-10-05T16:28:00+03:00</riseISO>\n" +
|
||||||
|
" <rise>1664976480</rise>\n" +
|
||||||
|
" <underfootISO>2022-10-05T08:56:00+03:00</underfootISO>\n" +
|
||||||
|
" </moon>\n" +
|
||||||
|
" <profile>\n" +
|
||||||
|
" <tz>Europe/Istanbul</tz>\n" +
|
||||||
|
" </profile>\n" +
|
||||||
|
" <place>\n" +
|
||||||
|
" <country>tr</country>\n" +
|
||||||
|
" <name>ankara</name>\n" +
|
||||||
|
" <state>an</state>\n" +
|
||||||
|
" </place>\n" +
|
||||||
|
" <sun>\n" +
|
||||||
|
" <setISO>2022-10-05T18:25:21+03:00</setISO>\n" +
|
||||||
|
" <midnightSun>false</midnightSun>\n" +
|
||||||
|
" <set>1664983521</set>\n" +
|
||||||
|
" <transit>1664962621</transit>\n" +
|
||||||
|
" <polarNight>false</polarNight>\n" +
|
||||||
|
" <transitISO>2022-10-05T12:37:01+03:00</transitISO>\n" +
|
||||||
|
" <riseISO>2022-10-05T06:48:41+03:00</riseISO>\n" +
|
||||||
|
" <rise>1664941721</rise>\n" +
|
||||||
|
" <twilight>\n" +
|
||||||
|
" <civilEnd>1664985136</civilEnd>\n" +
|
||||||
|
" <astronomicalBegin>1664936337</astronomicalBegin>\n" +
|
||||||
|
" <astronomicalEnd>1664988905</astronomicalEnd>\n" +
|
||||||
|
" <astronomicalBeginISO>2022-10-05T05:18:57+03:00</astronomicalBeginISO>\n" +
|
||||||
|
" <civilBegin>1664940106</civilBegin>\n" +
|
||||||
|
" <nauticalEndISO>2022-10-05T19:23:35+03:00</nauticalEndISO>\n" +
|
||||||
|
" <astronomicalEndISO>2022-10-05T19:55:05+03:00</astronomicalEndISO>\n" +
|
||||||
|
" <nauticalBegin>1664938227</nauticalBegin>\n" +
|
||||||
|
" <nauticalEnd>1664987015</nauticalEnd>\n" +
|
||||||
|
" <nauticalBeginISO>2022-10-05T05:50:27+03:00</nauticalBeginISO>\n" +
|
||||||
|
" <civilBeginISO>2022-10-05T06:21:46+03:00</civilBeginISO>\n" +
|
||||||
|
" <civilEndISO>2022-10-05T18:52:16+03:00</civilEndISO>\n" +
|
||||||
|
" </twilight>\n" +
|
||||||
|
" </sun>\n" +
|
||||||
|
" <timestamp>1664917200</timestamp>\n" +
|
||||||
|
"</response>\n" +
|
||||||
|
"<error>null</error>\n";
|
||||||
|
JSONObject expectedJsonObject = XML.toJSONObject(expected);
|
||||||
|
assertTrue(expectedJsonObject.similar(actualJsonObject));
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testIndentSimpleJsonObject(){
|
||||||
|
String str = "{ \"employee\": { \n" +
|
||||||
|
" \"name\": \"sonoo\", \n" +
|
||||||
|
" \"salary\": 56000, \n" +
|
||||||
|
" \"married\": true \n" +
|
||||||
|
" }}";
|
||||||
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
|
String actual = XML.toString(jsonObject, "Test", 2);
|
||||||
|
JSONObject actualJsonObject = XML.toJSONObject(actual);
|
||||||
|
String expected = "<Test>\n" +
|
||||||
|
" <employee>\n" +
|
||||||
|
" <name>sonoo</name>\n" +
|
||||||
|
" <salary>56000</salary>\n" +
|
||||||
|
" <married>true</married>\n" +
|
||||||
|
" </employee>\n" +
|
||||||
|
"</Test>\n";
|
||||||
|
JSONObject expectedJsonObject = XML.toJSONObject(expected);
|
||||||
|
assertTrue(expectedJsonObject.similar(actualJsonObject));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIndentSimpleJsonArray(){
|
||||||
|
String str = "[ \n" +
|
||||||
|
" {\"name\":\"Ram\", \"email\":\"Ram@gmail.com\"}, \n" +
|
||||||
|
" {\"name\":\"Bob\", \"email\":\"bob32@gmail.com\"} \n" +
|
||||||
|
"] ";
|
||||||
|
JSONArray jsonObject = new JSONArray(str);
|
||||||
|
String actual = XML.toString(jsonObject, 2);
|
||||||
|
JSONObject actualJsonObject = XML.toJSONObject(actual);
|
||||||
|
String expected = "<array>\n" +
|
||||||
|
" <name>Ram</name>\n" +
|
||||||
|
" <email>Ram@gmail.com</email>\n" +
|
||||||
|
"</array>\n" +
|
||||||
|
"<array>\n" +
|
||||||
|
" <name>Bob</name>\n" +
|
||||||
|
" <email>bob32@gmail.com</email>\n" +
|
||||||
|
"</array>\n";
|
||||||
|
JSONObject expectedJsonObject = XML.toJSONObject(expected);
|
||||||
|
assertTrue(expectedJsonObject.similar(actualJsonObject));
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIndentComplicatedJsonObjectWithArrayAndWithConfig(){
|
||||||
|
try {
|
||||||
|
InputStream jsonStream = null;
|
||||||
|
try {
|
||||||
|
jsonStream = XMLTest.class.getClassLoader().getResourceAsStream("Issue593.json");
|
||||||
|
final JSONObject object = new JSONObject(new JSONTokener(jsonStream));
|
||||||
|
String actualString = XML.toString(object, null, XMLParserConfiguration.KEEP_STRINGS,2);
|
||||||
|
InputStream xmlStream = null;
|
||||||
|
try {
|
||||||
|
xmlStream = XMLTest.class.getClassLoader().getResourceAsStream("Issue593.xml");
|
||||||
|
int bufferSize = 1024;
|
||||||
|
char[] buffer = new char[bufferSize];
|
||||||
|
StringBuilder expected = new StringBuilder();
|
||||||
|
Reader in = new InputStreamReader(xmlStream, "UTF-8");
|
||||||
|
for (int numRead; (numRead = in.read(buffer, 0, buffer.length)) > 0; ) {
|
||||||
|
expected.append(buffer, 0, numRead);
|
||||||
|
}
|
||||||
|
assertEquals(expected.toString(), actualString.replaceAll("\\n|\\r\\n", System.getProperty("line.separator")));
|
||||||
|
} finally {
|
||||||
|
if (xmlStream != null) {
|
||||||
|
xmlStream.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
if (jsonStream != null) {
|
||||||
|
jsonStream.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
fail("file writer error: " +e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMaxNestingDepthOf42IsRespected() {
|
||||||
|
final String wayTooLongMalformedXML = new String(new char[6000]).replace("\0", "<a>");
|
||||||
|
|
||||||
|
final int maxNestingDepth = 42;
|
||||||
|
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(wayTooLongMalformedXML, XMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||||
|
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||||
|
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMaxNestingDepthIsRespectedWithValidXML() {
|
||||||
|
final String perfectlyFineXML = "<Test>\n" +
|
||||||
|
" <employee>\n" +
|
||||||
|
" <name>sonoo</name>\n" +
|
||||||
|
" <salary>56000</salary>\n" +
|
||||||
|
" <married>true</married>\n" +
|
||||||
|
" </employee>\n" +
|
||||||
|
"</Test>\n";
|
||||||
|
|
||||||
|
final int maxNestingDepth = 1;
|
||||||
|
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(perfectlyFineXML, XMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||||
|
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||||
|
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMaxNestingDepthWithValidFittingXML() {
|
||||||
|
final String perfectlyFineXML = "<Test>\n" +
|
||||||
|
" <employee>\n" +
|
||||||
|
" <name>sonoo</name>\n" +
|
||||||
|
" <salary>56000</salary>\n" +
|
||||||
|
" <married>true</married>\n" +
|
||||||
|
" </employee>\n" +
|
||||||
|
"</Test>\n";
|
||||||
|
|
||||||
|
final int maxNestingDepth = 3;
|
||||||
|
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(perfectlyFineXML, XMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||||
|
} catch (JSONException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
fail("XML document should be parsed as its maximum depth fits the maxNestingDepth " +
|
||||||
|
"parameter of the XMLParserConfiguration used");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import java.io.IOException;
|
|||||||
import java.lang.reflect.InvocationTargetException;
|
import java.lang.reflect.InvocationTargetException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Object for testing the exception handling in {@link JSONObject#populateMap}.
|
* Object for testing the exception handling in {@link org.json.JSONObject#populateMap}.
|
||||||
*
|
*
|
||||||
* @author John Aylward
|
* @author John Aylward
|
||||||
*/
|
*/
|
||||||
|
|||||||
23
src/test/java/org/json/junit/data/RecursiveBean.java
Normal file
23
src/test/java/org/json/junit/data/RecursiveBean.java
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test class for verifying if recursively defined bean can be correctly identified
|
||||||
|
* @author Zetmas
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class RecursiveBean {
|
||||||
|
private String name;
|
||||||
|
private Object reference;
|
||||||
|
private Object reference2;
|
||||||
|
public String getName() { return name; }
|
||||||
|
public Object getRef() {return reference;}
|
||||||
|
public Object getRef2() {return reference2;}
|
||||||
|
public void setRef(Object refObj) {reference = refObj;}
|
||||||
|
public void setRef2(Object refObj) {reference2 = refObj;}
|
||||||
|
|
||||||
|
public RecursiveBean(String name) {
|
||||||
|
this.name = name;
|
||||||
|
reference = null;
|
||||||
|
reference2 = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
33
src/test/java/org/json/junit/data/RecursiveBeanEquals.java
Normal file
33
src/test/java/org/json/junit/data/RecursiveBeanEquals.java
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/** test class for verifying if recursively defined bean can be correctly identified */
|
||||||
|
public class RecursiveBeanEquals {
|
||||||
|
private final String name;
|
||||||
|
private Object reference;
|
||||||
|
|
||||||
|
public RecursiveBeanEquals(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Object getRef() {
|
||||||
|
return reference;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setRef(Object refObj) {
|
||||||
|
reference = refObj;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object other) {
|
||||||
|
return other instanceof RecursiveBeanEquals && name.equals(((RecursiveBeanEquals) other).name);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return name.hashCode();
|
||||||
|
}
|
||||||
|
}
|
||||||
704
src/test/resources/Issue593.json
Normal file
704
src/test/resources/Issue593.json
Normal file
@@ -0,0 +1,704 @@
|
|||||||
|
{
|
||||||
|
"success": true,
|
||||||
|
"error": null,
|
||||||
|
"response": [
|
||||||
|
{
|
||||||
|
"loc": {
|
||||||
|
"long": 31.25,
|
||||||
|
"lat": 30.063
|
||||||
|
},
|
||||||
|
"interval": "day",
|
||||||
|
"place": {
|
||||||
|
"name": "cairo",
|
||||||
|
"state": "qh",
|
||||||
|
"country": "eg"
|
||||||
|
},
|
||||||
|
"periods": [
|
||||||
|
{
|
||||||
|
"timestamp": 1665032400,
|
||||||
|
"validTime": "2022-10-06T07:00:00+02:00",
|
||||||
|
"dateTimeISO": "2022-10-06T07:00:00+02:00",
|
||||||
|
"maxTempC": 32,
|
||||||
|
"maxTempF": 90,
|
||||||
|
"minTempC": 19,
|
||||||
|
"minTempF": 66,
|
||||||
|
"avgTempC": 25,
|
||||||
|
"avgTempF": 78,
|
||||||
|
"tempC": null,
|
||||||
|
"tempF": null,
|
||||||
|
"maxFeelslikeC": 32,
|
||||||
|
"maxFeelslikeF": 89,
|
||||||
|
"minFeelslikeC": 21,
|
||||||
|
"minFeelslikeF": 70,
|
||||||
|
"avgFeelslikeC": 26,
|
||||||
|
"avgFeelslikeF": 80,
|
||||||
|
"feelslikeC": 21,
|
||||||
|
"feelslikeF": 70,
|
||||||
|
"maxDewpointC": 17,
|
||||||
|
"maxDewpointF": 63,
|
||||||
|
"minDewpointC": 11,
|
||||||
|
"minDewpointF": 52,
|
||||||
|
"avgDewpointC": 14,
|
||||||
|
"avgDewpointF": 58,
|
||||||
|
"dewpointC": 17,
|
||||||
|
"dewpointF": 63,
|
||||||
|
"maxHumidity": 77,
|
||||||
|
"minHumidity": 29,
|
||||||
|
"humidity": 77,
|
||||||
|
"pop": 0,
|
||||||
|
"precipMM": 0,
|
||||||
|
"precipIN": 0,
|
||||||
|
"iceaccum": null,
|
||||||
|
"iceaccumMM": null,
|
||||||
|
"iceaccumIN": null,
|
||||||
|
"snowCM": 0,
|
||||||
|
"snowIN": 0,
|
||||||
|
"pressureMB": 1015,
|
||||||
|
"pressureIN": 29.97,
|
||||||
|
"windDir": "N",
|
||||||
|
"windDirDEG": 353,
|
||||||
|
"windSpeedKTS": 5,
|
||||||
|
"windSpeedKPH": 9,
|
||||||
|
"windSpeedMPH": 6,
|
||||||
|
"windGustKTS": 21,
|
||||||
|
"windGustKPH": 40,
|
||||||
|
"windGustMPH": 25,
|
||||||
|
"windDirMax": "NNW",
|
||||||
|
"windDirMaxDEG": 342,
|
||||||
|
"windSpeedMaxKTS": 9,
|
||||||
|
"windSpeedMaxKPH": 16,
|
||||||
|
"windSpeedMaxMPH": 10,
|
||||||
|
"windDirMin": "N",
|
||||||
|
"windDirMinDEG": 353,
|
||||||
|
"windSpeedMinKTS": 1,
|
||||||
|
"windSpeedMinKPH": 2,
|
||||||
|
"windSpeedMinMPH": 1,
|
||||||
|
"windDir80m": "N",
|
||||||
|
"windDir80mDEG": 11,
|
||||||
|
"windSpeed80mKTS": 12,
|
||||||
|
"windSpeed80mKPH": 22,
|
||||||
|
"windSpeed80mMPH": 13,
|
||||||
|
"windGust80mKTS": 22,
|
||||||
|
"windGust80mKPH": 41,
|
||||||
|
"windGust80mMPH": 25,
|
||||||
|
"windDirMax80m": "NNW",
|
||||||
|
"windDirMax80mDEG": 343,
|
||||||
|
"windSpeedMax80mKTS": 22,
|
||||||
|
"windSpeedMax80mKPH": 41,
|
||||||
|
"windSpeedMax80mMPH": 25,
|
||||||
|
"windDirMin80m": "E",
|
||||||
|
"windDirMin80mDEG": 95,
|
||||||
|
"windSpeedMin80mKTS": 8,
|
||||||
|
"windSpeedMin80mKPH": 15,
|
||||||
|
"windSpeedMin80mMPH": 10,
|
||||||
|
"sky": 22,
|
||||||
|
"cloudsCoded": "FW",
|
||||||
|
"weather": "Mostly Sunny",
|
||||||
|
"weatherCoded": [],
|
||||||
|
"weatherPrimary": "Mostly Sunny",
|
||||||
|
"weatherPrimaryCoded": "::FW",
|
||||||
|
"icon": "fair.png",
|
||||||
|
"visibilityKM": 24.135,
|
||||||
|
"visibilityMI": 15,
|
||||||
|
"uvi": 6,
|
||||||
|
"solradWM2": 5608,
|
||||||
|
"solradMinWM2": 0,
|
||||||
|
"solradMaxWM2": 778,
|
||||||
|
"isDay": true,
|
||||||
|
"maxCoverage": "",
|
||||||
|
"sunrise": 1665028274,
|
||||||
|
"sunset": 1665070502,
|
||||||
|
"sunriseISO": "2022-10-06T05:51:14+02:00",
|
||||||
|
"sunsetISO": "2022-10-06T17:35:02+02:00"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"timestamp": 1665118800,
|
||||||
|
"validTime": "2022-10-07T07:00:00+02:00",
|
||||||
|
"dateTimeISO": "2022-10-07T07:00:00+02:00",
|
||||||
|
"maxTempC": 30,
|
||||||
|
"maxTempF": 86,
|
||||||
|
"minTempC": 19,
|
||||||
|
"minTempF": 66,
|
||||||
|
"avgTempC": 24,
|
||||||
|
"avgTempF": 76,
|
||||||
|
"tempC": null,
|
||||||
|
"tempF": null,
|
||||||
|
"maxFeelslikeC": 29,
|
||||||
|
"maxFeelslikeF": 85,
|
||||||
|
"minFeelslikeC": 19,
|
||||||
|
"minFeelslikeF": 67,
|
||||||
|
"avgFeelslikeC": 24,
|
||||||
|
"avgFeelslikeF": 76,
|
||||||
|
"feelslikeC": 19,
|
||||||
|
"feelslikeF": 67,
|
||||||
|
"maxDewpointC": 15,
|
||||||
|
"maxDewpointF": 60,
|
||||||
|
"minDewpointC": 10,
|
||||||
|
"minDewpointF": 50,
|
||||||
|
"avgDewpointC": 12,
|
||||||
|
"avgDewpointF": 54,
|
||||||
|
"dewpointC": 15,
|
||||||
|
"dewpointF": 60,
|
||||||
|
"maxHumidity": 77,
|
||||||
|
"minHumidity": 30,
|
||||||
|
"humidity": 77,
|
||||||
|
"pop": 0,
|
||||||
|
"precipMM": 0,
|
||||||
|
"precipIN": 0,
|
||||||
|
"iceaccum": null,
|
||||||
|
"iceaccumMM": null,
|
||||||
|
"iceaccumIN": null,
|
||||||
|
"snowCM": 0,
|
||||||
|
"snowIN": 0,
|
||||||
|
"pressureMB": 1014,
|
||||||
|
"pressureIN": 29.95,
|
||||||
|
"windDir": "NW",
|
||||||
|
"windDirDEG": 325,
|
||||||
|
"windSpeedKTS": 1,
|
||||||
|
"windSpeedKPH": 2,
|
||||||
|
"windSpeedMPH": 1,
|
||||||
|
"windGustKTS": 16,
|
||||||
|
"windGustKPH": 29,
|
||||||
|
"windGustMPH": 18,
|
||||||
|
"windDirMax": "WNW",
|
||||||
|
"windDirMaxDEG": 298,
|
||||||
|
"windSpeedMaxKTS": 7,
|
||||||
|
"windSpeedMaxKPH": 13,
|
||||||
|
"windSpeedMaxMPH": 8,
|
||||||
|
"windDirMin": "NW",
|
||||||
|
"windDirMinDEG": 325,
|
||||||
|
"windSpeedMinKTS": 1,
|
||||||
|
"windSpeedMinKPH": 2,
|
||||||
|
"windSpeedMinMPH": 1,
|
||||||
|
"windDir80m": "NNW",
|
||||||
|
"windDir80mDEG": 347,
|
||||||
|
"windSpeed80mKTS": 6,
|
||||||
|
"windSpeed80mKPH": 10,
|
||||||
|
"windSpeed80mMPH": 6,
|
||||||
|
"windGust80mKTS": 20,
|
||||||
|
"windGust80mKPH": 37,
|
||||||
|
"windGust80mMPH": 23,
|
||||||
|
"windDirMax80m": "NW",
|
||||||
|
"windDirMax80mDEG": 316,
|
||||||
|
"windSpeedMax80mKTS": 20,
|
||||||
|
"windSpeedMax80mKPH": 37,
|
||||||
|
"windSpeedMax80mMPH": 23,
|
||||||
|
"windDirMin80m": "NNW",
|
||||||
|
"windDirMin80mDEG": 347,
|
||||||
|
"windSpeedMin80mKTS": 6,
|
||||||
|
"windSpeedMin80mKPH": 10,
|
||||||
|
"windSpeedMin80mMPH": 6,
|
||||||
|
"sky": 30,
|
||||||
|
"cloudsCoded": "FW",
|
||||||
|
"weather": "Mostly Sunny",
|
||||||
|
"weatherCoded": [],
|
||||||
|
"weatherPrimary": "Mostly Sunny",
|
||||||
|
"weatherPrimaryCoded": "::FW",
|
||||||
|
"icon": "fair.png",
|
||||||
|
"visibilityKM": 24.135,
|
||||||
|
"visibilityMI": 15,
|
||||||
|
"uvi": 6,
|
||||||
|
"solradWM2": 5486,
|
||||||
|
"solradMinWM2": 0,
|
||||||
|
"solradMaxWM2": 742,
|
||||||
|
"isDay": true,
|
||||||
|
"maxCoverage": "",
|
||||||
|
"sunrise": 1665114710,
|
||||||
|
"sunset": 1665156831,
|
||||||
|
"sunriseISO": "2022-10-07T05:51:50+02:00",
|
||||||
|
"sunsetISO": "2022-10-07T17:33:51+02:00"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"timestamp": 1665205200,
|
||||||
|
"validTime": "2022-10-08T07:00:00+02:00",
|
||||||
|
"dateTimeISO": "2022-10-08T07:00:00+02:00",
|
||||||
|
"maxTempC": 30,
|
||||||
|
"maxTempF": 87,
|
||||||
|
"minTempC": 19,
|
||||||
|
"minTempF": 66,
|
||||||
|
"avgTempC": 25,
|
||||||
|
"avgTempF": 76,
|
||||||
|
"tempC": null,
|
||||||
|
"tempF": null,
|
||||||
|
"maxFeelslikeC": 30,
|
||||||
|
"maxFeelslikeF": 86,
|
||||||
|
"minFeelslikeC": 19,
|
||||||
|
"minFeelslikeF": 67,
|
||||||
|
"avgFeelslikeC": 25,
|
||||||
|
"avgFeelslikeF": 76,
|
||||||
|
"feelslikeC": 19,
|
||||||
|
"feelslikeF": 67,
|
||||||
|
"maxDewpointC": 15,
|
||||||
|
"maxDewpointF": 59,
|
||||||
|
"minDewpointC": 11,
|
||||||
|
"minDewpointF": 52,
|
||||||
|
"avgDewpointC": 13,
|
||||||
|
"avgDewpointF": 56,
|
||||||
|
"dewpointC": 15,
|
||||||
|
"dewpointF": 59,
|
||||||
|
"maxHumidity": 76,
|
||||||
|
"minHumidity": 32,
|
||||||
|
"humidity": 76,
|
||||||
|
"pop": 0,
|
||||||
|
"precipMM": 0,
|
||||||
|
"precipIN": 0,
|
||||||
|
"iceaccum": null,
|
||||||
|
"iceaccumMM": null,
|
||||||
|
"iceaccumIN": null,
|
||||||
|
"snowCM": 0,
|
||||||
|
"snowIN": 0,
|
||||||
|
"pressureMB": 1014,
|
||||||
|
"pressureIN": 29.94,
|
||||||
|
"windDir": "NNE",
|
||||||
|
"windDirDEG": 21,
|
||||||
|
"windSpeedKTS": 1,
|
||||||
|
"windSpeedKPH": 2,
|
||||||
|
"windSpeedMPH": 1,
|
||||||
|
"windGustKTS": 17,
|
||||||
|
"windGustKPH": 32,
|
||||||
|
"windGustMPH": 20,
|
||||||
|
"windDirMax": "WNW",
|
||||||
|
"windDirMaxDEG": 301,
|
||||||
|
"windSpeedMaxKTS": 7,
|
||||||
|
"windSpeedMaxKPH": 13,
|
||||||
|
"windSpeedMaxMPH": 8,
|
||||||
|
"windDirMin": "NNE",
|
||||||
|
"windDirMinDEG": 21,
|
||||||
|
"windSpeedMinKTS": 1,
|
||||||
|
"windSpeedMinKPH": 2,
|
||||||
|
"windSpeedMinMPH": 1,
|
||||||
|
"windDir80m": "NW",
|
||||||
|
"windDir80mDEG": 309,
|
||||||
|
"windSpeed80mKTS": 5,
|
||||||
|
"windSpeed80mKPH": 9,
|
||||||
|
"windSpeed80mMPH": 5,
|
||||||
|
"windGust80mKTS": 17,
|
||||||
|
"windGust80mKPH": 31,
|
||||||
|
"windGust80mMPH": 19,
|
||||||
|
"windDirMax80m": "NW",
|
||||||
|
"windDirMax80mDEG": 322,
|
||||||
|
"windSpeedMax80mKTS": 17,
|
||||||
|
"windSpeedMax80mKPH": 31,
|
||||||
|
"windSpeedMax80mMPH": 19,
|
||||||
|
"windDirMin80m": "NW",
|
||||||
|
"windDirMin80mDEG": 309,
|
||||||
|
"windSpeedMin80mKTS": 5,
|
||||||
|
"windSpeedMin80mKPH": 9,
|
||||||
|
"windSpeedMin80mMPH": 5,
|
||||||
|
"sky": 47,
|
||||||
|
"cloudsCoded": "SC",
|
||||||
|
"weather": "Partly Cloudy",
|
||||||
|
"weatherCoded": [],
|
||||||
|
"weatherPrimary": "Partly Cloudy",
|
||||||
|
"weatherPrimaryCoded": "::SC",
|
||||||
|
"icon": "pcloudy.png",
|
||||||
|
"visibilityKM": 24.135,
|
||||||
|
"visibilityMI": 15,
|
||||||
|
"uvi": 7,
|
||||||
|
"solradWM2": 4785,
|
||||||
|
"solradMinWM2": 0,
|
||||||
|
"solradMaxWM2": 682,
|
||||||
|
"isDay": true,
|
||||||
|
"maxCoverage": "",
|
||||||
|
"sunrise": 1665201146,
|
||||||
|
"sunset": 1665243161,
|
||||||
|
"sunriseISO": "2022-10-08T05:52:26+02:00",
|
||||||
|
"sunsetISO": "2022-10-08T17:32:41+02:00"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"timestamp": 1665291600,
|
||||||
|
"validTime": "2022-10-09T07:00:00+02:00",
|
||||||
|
"dateTimeISO": "2022-10-09T07:00:00+02:00",
|
||||||
|
"maxTempC": 31,
|
||||||
|
"maxTempF": 87,
|
||||||
|
"minTempC": 19,
|
||||||
|
"minTempF": 67,
|
||||||
|
"avgTempC": 25,
|
||||||
|
"avgTempF": 77,
|
||||||
|
"tempC": null,
|
||||||
|
"tempF": null,
|
||||||
|
"maxFeelslikeC": 30,
|
||||||
|
"maxFeelslikeF": 86,
|
||||||
|
"minFeelslikeC": 20,
|
||||||
|
"minFeelslikeF": 67,
|
||||||
|
"avgFeelslikeC": 25,
|
||||||
|
"avgFeelslikeF": 77,
|
||||||
|
"feelslikeC": 20,
|
||||||
|
"feelslikeF": 67,
|
||||||
|
"maxDewpointC": 17,
|
||||||
|
"maxDewpointF": 63,
|
||||||
|
"minDewpointC": 11,
|
||||||
|
"minDewpointF": 52,
|
||||||
|
"avgDewpointC": 14,
|
||||||
|
"avgDewpointF": 57,
|
||||||
|
"dewpointC": 17,
|
||||||
|
"dewpointF": 63,
|
||||||
|
"maxHumidity": 86,
|
||||||
|
"minHumidity": 31,
|
||||||
|
"humidity": 86,
|
||||||
|
"pop": 0,
|
||||||
|
"precipMM": 0,
|
||||||
|
"precipIN": 0,
|
||||||
|
"iceaccum": null,
|
||||||
|
"iceaccumMM": null,
|
||||||
|
"iceaccumIN": null,
|
||||||
|
"snowCM": 0,
|
||||||
|
"snowIN": 0,
|
||||||
|
"pressureMB": 1016,
|
||||||
|
"pressureIN": 29.99,
|
||||||
|
"windDir": "N",
|
||||||
|
"windDirDEG": 356,
|
||||||
|
"windSpeedKTS": 2,
|
||||||
|
"windSpeedKPH": 4,
|
||||||
|
"windSpeedMPH": 2,
|
||||||
|
"windGustKTS": 19,
|
||||||
|
"windGustKPH": 36,
|
||||||
|
"windGustMPH": 22,
|
||||||
|
"windDirMax": "NNW",
|
||||||
|
"windDirMaxDEG": 343,
|
||||||
|
"windSpeedMaxKTS": 8,
|
||||||
|
"windSpeedMaxKPH": 14,
|
||||||
|
"windSpeedMaxMPH": 9,
|
||||||
|
"windDirMin": "N",
|
||||||
|
"windDirMinDEG": 356,
|
||||||
|
"windSpeedMinKTS": 2,
|
||||||
|
"windSpeedMinKPH": 4,
|
||||||
|
"windSpeedMinMPH": 2,
|
||||||
|
"windDir80m": "NW",
|
||||||
|
"windDir80mDEG": 316,
|
||||||
|
"windSpeed80mKTS": 5,
|
||||||
|
"windSpeed80mKPH": 9,
|
||||||
|
"windSpeed80mMPH": 6,
|
||||||
|
"windGust80mKTS": 20,
|
||||||
|
"windGust80mKPH": 36,
|
||||||
|
"windGust80mMPH": 23,
|
||||||
|
"windDirMax80m": "N",
|
||||||
|
"windDirMax80mDEG": 354,
|
||||||
|
"windSpeedMax80mKTS": 20,
|
||||||
|
"windSpeedMax80mKPH": 36,
|
||||||
|
"windSpeedMax80mMPH": 23,
|
||||||
|
"windDirMin80m": "NW",
|
||||||
|
"windDirMin80mDEG": 316,
|
||||||
|
"windSpeedMin80mKTS": 5,
|
||||||
|
"windSpeedMin80mKPH": 9,
|
||||||
|
"windSpeedMin80mMPH": 6,
|
||||||
|
"sky": 47,
|
||||||
|
"cloudsCoded": "SC",
|
||||||
|
"weather": "Partly Cloudy",
|
||||||
|
"weatherCoded": [],
|
||||||
|
"weatherPrimary": "Partly Cloudy",
|
||||||
|
"weatherPrimaryCoded": "::SC",
|
||||||
|
"icon": "pcloudy.png",
|
||||||
|
"visibilityKM": 24.135,
|
||||||
|
"visibilityMI": 15,
|
||||||
|
"uvi": 7,
|
||||||
|
"solradWM2": 4768,
|
||||||
|
"solradMinWM2": 0,
|
||||||
|
"solradMaxWM2": 726,
|
||||||
|
"isDay": true,
|
||||||
|
"maxCoverage": "",
|
||||||
|
"sunrise": 1665287583,
|
||||||
|
"sunset": 1665329491,
|
||||||
|
"sunriseISO": "2022-10-09T05:53:03+02:00",
|
||||||
|
"sunsetISO": "2022-10-09T17:31:31+02:00"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"timestamp": 1665378000,
|
||||||
|
"validTime": "2022-10-10T07:00:00+02:00",
|
||||||
|
"dateTimeISO": "2022-10-10T07:00:00+02:00",
|
||||||
|
"maxTempC": 31,
|
||||||
|
"maxTempF": 87,
|
||||||
|
"minTempC": 21,
|
||||||
|
"minTempF": 70,
|
||||||
|
"avgTempC": 26,
|
||||||
|
"avgTempF": 78,
|
||||||
|
"tempC": null,
|
||||||
|
"tempF": null,
|
||||||
|
"maxFeelslikeC": 30,
|
||||||
|
"maxFeelslikeF": 86,
|
||||||
|
"minFeelslikeC": 21,
|
||||||
|
"minFeelslikeF": 69,
|
||||||
|
"avgFeelslikeC": 25,
|
||||||
|
"avgFeelslikeF": 78,
|
||||||
|
"feelslikeC": 21,
|
||||||
|
"feelslikeF": 69,
|
||||||
|
"maxDewpointC": 16,
|
||||||
|
"maxDewpointF": 61,
|
||||||
|
"minDewpointC": 13,
|
||||||
|
"minDewpointF": 55,
|
||||||
|
"avgDewpointC": 14,
|
||||||
|
"avgDewpointF": 58,
|
||||||
|
"dewpointC": 16,
|
||||||
|
"dewpointF": 61,
|
||||||
|
"maxHumidity": 75,
|
||||||
|
"minHumidity": 35,
|
||||||
|
"humidity": 75,
|
||||||
|
"pop": 0,
|
||||||
|
"precipMM": 0,
|
||||||
|
"precipIN": 0,
|
||||||
|
"iceaccum": null,
|
||||||
|
"iceaccumMM": null,
|
||||||
|
"iceaccumIN": null,
|
||||||
|
"snowCM": 0,
|
||||||
|
"snowIN": 0,
|
||||||
|
"pressureMB": 1017,
|
||||||
|
"pressureIN": 30.03,
|
||||||
|
"windDir": "N",
|
||||||
|
"windDirDEG": 358,
|
||||||
|
"windSpeedKTS": 2,
|
||||||
|
"windSpeedKPH": 4,
|
||||||
|
"windSpeedMPH": 2,
|
||||||
|
"windGustKTS": 16,
|
||||||
|
"windGustKPH": 30,
|
||||||
|
"windGustMPH": 19,
|
||||||
|
"windDirMax": "N",
|
||||||
|
"windDirMaxDEG": 10,
|
||||||
|
"windSpeedMaxKTS": 8,
|
||||||
|
"windSpeedMaxKPH": 15,
|
||||||
|
"windSpeedMaxMPH": 9,
|
||||||
|
"windDirMin": "N",
|
||||||
|
"windDirMinDEG": 358,
|
||||||
|
"windSpeedMinKTS": 2,
|
||||||
|
"windSpeedMinKPH": 4,
|
||||||
|
"windSpeedMinMPH": 2,
|
||||||
|
"windDir80m": "N",
|
||||||
|
"windDir80mDEG": 8,
|
||||||
|
"windSpeed80mKTS": 7,
|
||||||
|
"windSpeed80mKPH": 13,
|
||||||
|
"windSpeed80mMPH": 8,
|
||||||
|
"windGust80mKTS": 19,
|
||||||
|
"windGust80mKPH": 36,
|
||||||
|
"windGust80mMPH": 22,
|
||||||
|
"windDirMax80m": "N",
|
||||||
|
"windDirMax80mDEG": 10,
|
||||||
|
"windSpeedMax80mKTS": 19,
|
||||||
|
"windSpeedMax80mKPH": 36,
|
||||||
|
"windSpeedMax80mMPH": 22,
|
||||||
|
"windDirMin80m": "E",
|
||||||
|
"windDirMin80mDEG": 91,
|
||||||
|
"windSpeedMin80mKTS": 7,
|
||||||
|
"windSpeedMin80mKPH": 13,
|
||||||
|
"windSpeedMin80mMPH": 8,
|
||||||
|
"sky": 64,
|
||||||
|
"cloudsCoded": "SC",
|
||||||
|
"weather": "Partly Cloudy",
|
||||||
|
"weatherCoded": [],
|
||||||
|
"weatherPrimary": "Partly Cloudy",
|
||||||
|
"weatherPrimaryCoded": "::SC",
|
||||||
|
"icon": "pcloudy.png",
|
||||||
|
"visibilityKM": 24.135,
|
||||||
|
"visibilityMI": 15,
|
||||||
|
"uvi": 6,
|
||||||
|
"solradWM2": 4494,
|
||||||
|
"solradMinWM2": 0,
|
||||||
|
"solradMaxWM2": 597,
|
||||||
|
"isDay": true,
|
||||||
|
"maxCoverage": "",
|
||||||
|
"sunrise": 1665374020,
|
||||||
|
"sunset": 1665415821,
|
||||||
|
"sunriseISO": "2022-10-10T05:53:40+02:00",
|
||||||
|
"sunsetISO": "2022-10-10T17:30:21+02:00"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"timestamp": 1665464400,
|
||||||
|
"validTime": "2022-10-11T07:00:00+02:00",
|
||||||
|
"dateTimeISO": "2022-10-11T07:00:00+02:00",
|
||||||
|
"maxTempC": 31,
|
||||||
|
"maxTempF": 87,
|
||||||
|
"minTempC": 21,
|
||||||
|
"minTempF": 70,
|
||||||
|
"avgTempC": 26,
|
||||||
|
"avgTempF": 78,
|
||||||
|
"tempC": null,
|
||||||
|
"tempF": null,
|
||||||
|
"maxFeelslikeC": 31,
|
||||||
|
"maxFeelslikeF": 87,
|
||||||
|
"minFeelslikeC": 22,
|
||||||
|
"minFeelslikeF": 72,
|
||||||
|
"avgFeelslikeC": 26,
|
||||||
|
"avgFeelslikeF": 79,
|
||||||
|
"feelslikeC": 22,
|
||||||
|
"feelslikeF": 72,
|
||||||
|
"maxDewpointC": 17,
|
||||||
|
"maxDewpointF": 62,
|
||||||
|
"minDewpointC": 11,
|
||||||
|
"minDewpointF": 51,
|
||||||
|
"avgDewpointC": 13,
|
||||||
|
"avgDewpointF": 55,
|
||||||
|
"dewpointC": 17,
|
||||||
|
"dewpointF": 62,
|
||||||
|
"maxHumidity": 71,
|
||||||
|
"minHumidity": 30,
|
||||||
|
"humidity": 71,
|
||||||
|
"pop": 0,
|
||||||
|
"precipMM": 0,
|
||||||
|
"precipIN": 0,
|
||||||
|
"iceaccum": null,
|
||||||
|
"iceaccumMM": null,
|
||||||
|
"iceaccumIN": null,
|
||||||
|
"snowCM": 0,
|
||||||
|
"snowIN": 0,
|
||||||
|
"pressureMB": 1015,
|
||||||
|
"pressureIN": 29.98,
|
||||||
|
"windDir": "NNE",
|
||||||
|
"windDirDEG": 13,
|
||||||
|
"windSpeedKTS": 8,
|
||||||
|
"windSpeedKPH": 15,
|
||||||
|
"windSpeedMPH": 9,
|
||||||
|
"windGustKTS": 15,
|
||||||
|
"windGustKPH": 28,
|
||||||
|
"windGustMPH": 17,
|
||||||
|
"windDirMax": "NNE",
|
||||||
|
"windDirMaxDEG": 28,
|
||||||
|
"windSpeedMaxKTS": 15,
|
||||||
|
"windSpeedMaxKPH": 28,
|
||||||
|
"windSpeedMaxMPH": 18,
|
||||||
|
"windDirMin": "NNE",
|
||||||
|
"windDirMinDEG": 14,
|
||||||
|
"windSpeedMinKTS": 7,
|
||||||
|
"windSpeedMinKPH": 14,
|
||||||
|
"windSpeedMinMPH": 8,
|
||||||
|
"windDir80m": "NNE",
|
||||||
|
"windDir80mDEG": 16,
|
||||||
|
"windSpeed80mKTS": 10,
|
||||||
|
"windSpeed80mKPH": 19,
|
||||||
|
"windSpeed80mMPH": 12,
|
||||||
|
"windGust80mKTS": 17,
|
||||||
|
"windGust80mKPH": 31,
|
||||||
|
"windGust80mMPH": 19,
|
||||||
|
"windDirMax80m": "NNE",
|
||||||
|
"windDirMax80mDEG": 28,
|
||||||
|
"windSpeedMax80mKTS": 17,
|
||||||
|
"windSpeedMax80mKPH": 31,
|
||||||
|
"windSpeedMax80mMPH": 19,
|
||||||
|
"windDirMin80m": "NNE",
|
||||||
|
"windDirMin80mDEG": 13,
|
||||||
|
"windSpeedMin80mKTS": 9,
|
||||||
|
"windSpeedMin80mKPH": 18,
|
||||||
|
"windSpeedMin80mMPH": 11,
|
||||||
|
"sky": 0,
|
||||||
|
"cloudsCoded": "CL",
|
||||||
|
"weather": "Sunny",
|
||||||
|
"weatherCoded": [],
|
||||||
|
"weatherPrimary": "Sunny",
|
||||||
|
"weatherPrimaryCoded": "::CL",
|
||||||
|
"icon": "sunny.png",
|
||||||
|
"visibilityKM": 24.135,
|
||||||
|
"visibilityMI": 15,
|
||||||
|
"uvi": null,
|
||||||
|
"solradWM2": 5450,
|
||||||
|
"solradMinWM2": 0,
|
||||||
|
"solradMaxWM2": 758,
|
||||||
|
"isDay": true,
|
||||||
|
"maxCoverage": "",
|
||||||
|
"sunrise": 1665460458,
|
||||||
|
"sunset": 1665502153,
|
||||||
|
"sunriseISO": "2022-10-11T05:54:18+02:00",
|
||||||
|
"sunsetISO": "2022-10-11T17:29:13+02:00"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"timestamp": 1665550800,
|
||||||
|
"validTime": "2022-10-12T07:00:00+02:00",
|
||||||
|
"dateTimeISO": "2022-10-12T07:00:00+02:00",
|
||||||
|
"maxTempC": 31,
|
||||||
|
"maxTempF": 88,
|
||||||
|
"minTempC": 21,
|
||||||
|
"minTempF": 69,
|
||||||
|
"avgTempC": 26,
|
||||||
|
"avgTempF": 79,
|
||||||
|
"tempC": null,
|
||||||
|
"tempF": null,
|
||||||
|
"maxFeelslikeC": 31,
|
||||||
|
"maxFeelslikeF": 88,
|
||||||
|
"minFeelslikeC": 22,
|
||||||
|
"minFeelslikeF": 72,
|
||||||
|
"avgFeelslikeC": 26,
|
||||||
|
"avgFeelslikeF": 80,
|
||||||
|
"feelslikeC": 22,
|
||||||
|
"feelslikeF": 72,
|
||||||
|
"maxDewpointC": 16,
|
||||||
|
"maxDewpointF": 60,
|
||||||
|
"minDewpointC": 11,
|
||||||
|
"minDewpointF": 51,
|
||||||
|
"avgDewpointC": 13,
|
||||||
|
"avgDewpointF": 55,
|
||||||
|
"dewpointC": 16,
|
||||||
|
"dewpointF": 60,
|
||||||
|
"maxHumidity": 68,
|
||||||
|
"minHumidity": 29,
|
||||||
|
"humidity": 68,
|
||||||
|
"pop": 0,
|
||||||
|
"precipMM": 0,
|
||||||
|
"precipIN": 0,
|
||||||
|
"iceaccum": null,
|
||||||
|
"iceaccumMM": null,
|
||||||
|
"iceaccumIN": null,
|
||||||
|
"snowCM": 0,
|
||||||
|
"snowIN": 0,
|
||||||
|
"pressureMB": 1014,
|
||||||
|
"pressureIN": 29.95,
|
||||||
|
"windDir": "NNE",
|
||||||
|
"windDirDEG": 12,
|
||||||
|
"windSpeedKTS": 8,
|
||||||
|
"windSpeedKPH": 15,
|
||||||
|
"windSpeedMPH": 9,
|
||||||
|
"windGustKTS": 15,
|
||||||
|
"windGustKPH": 28,
|
||||||
|
"windGustMPH": 17,
|
||||||
|
"windDirMax": "E",
|
||||||
|
"windDirMaxDEG": 96,
|
||||||
|
"windSpeedMaxKTS": 14,
|
||||||
|
"windSpeedMaxKPH": 26,
|
||||||
|
"windSpeedMaxMPH": 16,
|
||||||
|
"windDirMin": "NNE",
|
||||||
|
"windDirMinDEG": 12,
|
||||||
|
"windSpeedMinKTS": 7,
|
||||||
|
"windSpeedMinKPH": 13,
|
||||||
|
"windSpeedMinMPH": 8,
|
||||||
|
"windDir80m": "NNE",
|
||||||
|
"windDir80mDEG": 15,
|
||||||
|
"windSpeed80mKTS": 10,
|
||||||
|
"windSpeed80mKPH": 19,
|
||||||
|
"windSpeed80mMPH": 12,
|
||||||
|
"windGust80mKTS": 18,
|
||||||
|
"windGust80mKPH": 33,
|
||||||
|
"windGust80mMPH": 21,
|
||||||
|
"windDirMax80m": "E",
|
||||||
|
"windDirMax80mDEG": 96,
|
||||||
|
"windSpeedMax80mKTS": 18,
|
||||||
|
"windSpeedMax80mKPH": 33,
|
||||||
|
"windSpeedMax80mMPH": 21,
|
||||||
|
"windDirMin80m": "NNE",
|
||||||
|
"windDirMin80mDEG": 15,
|
||||||
|
"windSpeedMin80mKTS": 10,
|
||||||
|
"windSpeedMin80mKPH": 18,
|
||||||
|
"windSpeedMin80mMPH": 11,
|
||||||
|
"sky": 27,
|
||||||
|
"cloudsCoded": "FW",
|
||||||
|
"weather": "Mostly Sunny",
|
||||||
|
"weatherCoded": [],
|
||||||
|
"weatherPrimary": "Mostly Sunny",
|
||||||
|
"weatherPrimaryCoded": "::FW",
|
||||||
|
"icon": "fair.png",
|
||||||
|
"visibilityKM": 24.135,
|
||||||
|
"visibilityMI": 15,
|
||||||
|
"uvi": null,
|
||||||
|
"solradWM2": 4740,
|
||||||
|
"solradMinWM2": 0,
|
||||||
|
"solradMaxWM2": 743,
|
||||||
|
"isDay": true,
|
||||||
|
"maxCoverage": "",
|
||||||
|
"sunrise": 1665546895,
|
||||||
|
"sunset": 1665588484,
|
||||||
|
"sunriseISO": "2022-10-12T05:54:55+02:00",
|
||||||
|
"sunsetISO": "2022-10-12T17:28:04+02:00"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"profile": {
|
||||||
|
"tz": "Africa/Cairo",
|
||||||
|
"elevM": 23,
|
||||||
|
"elevFT": 75
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
691
src/test/resources/Issue593.xml
Normal file
691
src/test/resources/Issue593.xml
Normal file
@@ -0,0 +1,691 @@
|
|||||||
|
<success>true</success>
|
||||||
|
<response>
|
||||||
|
<loc>
|
||||||
|
<long>31.25</long>
|
||||||
|
<lat>30.063</lat>
|
||||||
|
</loc>
|
||||||
|
<profile>
|
||||||
|
<elevM>23</elevM>
|
||||||
|
<tz>Africa/Cairo</tz>
|
||||||
|
<elevFT>75</elevFT>
|
||||||
|
</profile>
|
||||||
|
<periods>
|
||||||
|
<dateTimeISO>2022-10-06T07:00:00+02:00</dateTimeISO>
|
||||||
|
<windDirMin80m>E</windDirMin80m>
|
||||||
|
<windDirMin80mDEG>95</windDirMin80mDEG>
|
||||||
|
<feelslikeC>21</feelslikeC>
|
||||||
|
<visibilityMI>15</visibilityMI>
|
||||||
|
<windSpeedMaxMPH>10</windSpeedMaxMPH>
|
||||||
|
<windDirDEG>353</windDirDEG>
|
||||||
|
<windDir>N</windDir>
|
||||||
|
<sunriseISO>2022-10-06T05:51:14+02:00</sunriseISO>
|
||||||
|
<iceaccumMM>null</iceaccumMM>
|
||||||
|
<windSpeedMaxKTS>9</windSpeedMaxKTS>
|
||||||
|
<iceaccumIN>null</iceaccumIN>
|
||||||
|
<minTempF>66</minTempF>
|
||||||
|
<snowIN>0</snowIN>
|
||||||
|
<weather>Mostly Sunny</weather>
|
||||||
|
<sunsetISO>2022-10-06T17:35:02+02:00</sunsetISO>
|
||||||
|
<maxFeelslikeC>32</maxFeelslikeC>
|
||||||
|
<humidity>77</humidity>
|
||||||
|
<windDir80m>N</windDir80m>
|
||||||
|
<maxFeelslikeF>89</maxFeelslikeF>
|
||||||
|
<precipMM>0</precipMM>
|
||||||
|
<sky>22</sky>
|
||||||
|
<windGust80mMPH>25</windGust80mMPH>
|
||||||
|
<windSpeedMax80mMPH>25</windSpeedMax80mMPH>
|
||||||
|
<weatherPrimary>Mostly Sunny</weatherPrimary>
|
||||||
|
<windGust80mKPH>41</windGust80mKPH>
|
||||||
|
<avgDewpointF>58</avgDewpointF>
|
||||||
|
<windSpeedMax80mKPH>41</windSpeedMax80mKPH>
|
||||||
|
<windGust80mKTS>22</windGust80mKTS>
|
||||||
|
<avgDewpointC>14</avgDewpointC>
|
||||||
|
<precipIN>0</precipIN>
|
||||||
|
<windSpeedMax80mKTS>22</windSpeedMax80mKTS>
|
||||||
|
<windDirMinDEG>353</windDirMinDEG>
|
||||||
|
<windSpeedMaxKPH>16</windSpeedMaxKPH>
|
||||||
|
<windSpeedMin80mKTS>8</windSpeedMin80mKTS>
|
||||||
|
<feelslikeF>70</feelslikeF>
|
||||||
|
<validTime>2022-10-06T07:00:00+02:00</validTime>
|
||||||
|
<windSpeedMin80mMPH>10</windSpeedMin80mMPH>
|
||||||
|
<solradMaxWM2>778</solradMaxWM2>
|
||||||
|
<avgTempC>25</avgTempC>
|
||||||
|
<windSpeedMin80mKPH>15</windSpeedMin80mKPH>
|
||||||
|
<weatherPrimaryCoded>::FW</weatherPrimaryCoded>
|
||||||
|
<sunrise>1665028274</sunrise>
|
||||||
|
<avgTempF>78</avgTempF>
|
||||||
|
<windDirMin>N</windDirMin>
|
||||||
|
<maxCoverage/>
|
||||||
|
<icon>fair.png</icon>
|
||||||
|
<minFeelslikeC>21</minFeelslikeC>
|
||||||
|
<dewpointC>17</dewpointC>
|
||||||
|
<cloudsCoded>FW</cloudsCoded>
|
||||||
|
<minFeelslikeF>70</minFeelslikeF>
|
||||||
|
<minHumidity>29</minHumidity>
|
||||||
|
<dewpointF>63</dewpointF>
|
||||||
|
<windSpeed80mKTS>12</windSpeed80mKTS>
|
||||||
|
<pop>0</pop>
|
||||||
|
<snowCM>0</snowCM>
|
||||||
|
<windDirMax>NNW</windDirMax>
|
||||||
|
<windSpeed80mMPH>13</windSpeed80mMPH>
|
||||||
|
<windSpeed80mKPH>22</windSpeed80mKPH>
|
||||||
|
<windDir80mDEG>11</windDir80mDEG>
|
||||||
|
<maxTempC>32</maxTempC>
|
||||||
|
<pressureMB>1015</pressureMB>
|
||||||
|
<visibilityKM>24.135</visibilityKM>
|
||||||
|
<timestamp>1665032400</timestamp>
|
||||||
|
<maxTempF>90</maxTempF>
|
||||||
|
<tempF>null</tempF>
|
||||||
|
<minDewpointC>11</minDewpointC>
|
||||||
|
<solradMinWM2>0</solradMinWM2>
|
||||||
|
<windSpeedMinKTS>1</windSpeedMinKTS>
|
||||||
|
<windDirMax80mDEG>343</windDirMax80mDEG>
|
||||||
|
<windGustKTS>21</windGustKTS>
|
||||||
|
<windSpeedMinKPH>2</windSpeedMinKPH>
|
||||||
|
<maxDewpointF>63</maxDewpointF>
|
||||||
|
<windSpeedMinMPH>1</windSpeedMinMPH>
|
||||||
|
<avgFeelslikeC>26</avgFeelslikeC>
|
||||||
|
<uvi>6</uvi>
|
||||||
|
<windDirMax80m>NNW</windDirMax80m>
|
||||||
|
<maxDewpointC>17</maxDewpointC>
|
||||||
|
<pressureIN>29.97</pressureIN>
|
||||||
|
<avgFeelslikeF>80</avgFeelslikeF>
|
||||||
|
<iceaccum>null</iceaccum>
|
||||||
|
<isDay>true</isDay>
|
||||||
|
<minTempC>19</minTempC>
|
||||||
|
<minDewpointF>52</minDewpointF>
|
||||||
|
<windSpeedKTS>5</windSpeedKTS>
|
||||||
|
<sunset>1665070502</sunset>
|
||||||
|
<solradWM2>5608</solradWM2>
|
||||||
|
<windSpeedKPH>9</windSpeedKPH>
|
||||||
|
<windGustMPH>25</windGustMPH>
|
||||||
|
<maxHumidity>77</maxHumidity>
|
||||||
|
<windSpeedMPH>6</windSpeedMPH>
|
||||||
|
<windGustKPH>40</windGustKPH>
|
||||||
|
<windDirMaxDEG>342</windDirMaxDEG>
|
||||||
|
<tempC>null</tempC>
|
||||||
|
</periods>
|
||||||
|
<periods>
|
||||||
|
<dateTimeISO>2022-10-07T07:00:00+02:00</dateTimeISO>
|
||||||
|
<windDirMin80m>NNW</windDirMin80m>
|
||||||
|
<windDirMin80mDEG>347</windDirMin80mDEG>
|
||||||
|
<feelslikeC>19</feelslikeC>
|
||||||
|
<visibilityMI>15</visibilityMI>
|
||||||
|
<windSpeedMaxMPH>8</windSpeedMaxMPH>
|
||||||
|
<windDirDEG>325</windDirDEG>
|
||||||
|
<windDir>NW</windDir>
|
||||||
|
<sunriseISO>2022-10-07T05:51:50+02:00</sunriseISO>
|
||||||
|
<iceaccumMM>null</iceaccumMM>
|
||||||
|
<windSpeedMaxKTS>7</windSpeedMaxKTS>
|
||||||
|
<iceaccumIN>null</iceaccumIN>
|
||||||
|
<minTempF>66</minTempF>
|
||||||
|
<snowIN>0</snowIN>
|
||||||
|
<weather>Mostly Sunny</weather>
|
||||||
|
<sunsetISO>2022-10-07T17:33:51+02:00</sunsetISO>
|
||||||
|
<maxFeelslikeC>29</maxFeelslikeC>
|
||||||
|
<humidity>77</humidity>
|
||||||
|
<windDir80m>NNW</windDir80m>
|
||||||
|
<maxFeelslikeF>85</maxFeelslikeF>
|
||||||
|
<precipMM>0</precipMM>
|
||||||
|
<sky>30</sky>
|
||||||
|
<windGust80mMPH>23</windGust80mMPH>
|
||||||
|
<windSpeedMax80mMPH>23</windSpeedMax80mMPH>
|
||||||
|
<weatherPrimary>Mostly Sunny</weatherPrimary>
|
||||||
|
<windGust80mKPH>37</windGust80mKPH>
|
||||||
|
<avgDewpointF>54</avgDewpointF>
|
||||||
|
<windSpeedMax80mKPH>37</windSpeedMax80mKPH>
|
||||||
|
<windGust80mKTS>20</windGust80mKTS>
|
||||||
|
<avgDewpointC>12</avgDewpointC>
|
||||||
|
<precipIN>0</precipIN>
|
||||||
|
<windSpeedMax80mKTS>20</windSpeedMax80mKTS>
|
||||||
|
<windDirMinDEG>325</windDirMinDEG>
|
||||||
|
<windSpeedMaxKPH>13</windSpeedMaxKPH>
|
||||||
|
<windSpeedMin80mKTS>6</windSpeedMin80mKTS>
|
||||||
|
<feelslikeF>67</feelslikeF>
|
||||||
|
<validTime>2022-10-07T07:00:00+02:00</validTime>
|
||||||
|
<windSpeedMin80mMPH>6</windSpeedMin80mMPH>
|
||||||
|
<solradMaxWM2>742</solradMaxWM2>
|
||||||
|
<avgTempC>24</avgTempC>
|
||||||
|
<windSpeedMin80mKPH>10</windSpeedMin80mKPH>
|
||||||
|
<weatherPrimaryCoded>::FW</weatherPrimaryCoded>
|
||||||
|
<sunrise>1665114710</sunrise>
|
||||||
|
<avgTempF>76</avgTempF>
|
||||||
|
<windDirMin>NW</windDirMin>
|
||||||
|
<maxCoverage/>
|
||||||
|
<icon>fair.png</icon>
|
||||||
|
<minFeelslikeC>19</minFeelslikeC>
|
||||||
|
<dewpointC>15</dewpointC>
|
||||||
|
<cloudsCoded>FW</cloudsCoded>
|
||||||
|
<minFeelslikeF>67</minFeelslikeF>
|
||||||
|
<minHumidity>30</minHumidity>
|
||||||
|
<dewpointF>60</dewpointF>
|
||||||
|
<windSpeed80mKTS>6</windSpeed80mKTS>
|
||||||
|
<pop>0</pop>
|
||||||
|
<snowCM>0</snowCM>
|
||||||
|
<windDirMax>WNW</windDirMax>
|
||||||
|
<windSpeed80mMPH>6</windSpeed80mMPH>
|
||||||
|
<windSpeed80mKPH>10</windSpeed80mKPH>
|
||||||
|
<windDir80mDEG>347</windDir80mDEG>
|
||||||
|
<maxTempC>30</maxTempC>
|
||||||
|
<pressureMB>1014</pressureMB>
|
||||||
|
<visibilityKM>24.135</visibilityKM>
|
||||||
|
<timestamp>1665118800</timestamp>
|
||||||
|
<maxTempF>86</maxTempF>
|
||||||
|
<tempF>null</tempF>
|
||||||
|
<minDewpointC>10</minDewpointC>
|
||||||
|
<solradMinWM2>0</solradMinWM2>
|
||||||
|
<windSpeedMinKTS>1</windSpeedMinKTS>
|
||||||
|
<windDirMax80mDEG>316</windDirMax80mDEG>
|
||||||
|
<windGustKTS>16</windGustKTS>
|
||||||
|
<windSpeedMinKPH>2</windSpeedMinKPH>
|
||||||
|
<maxDewpointF>60</maxDewpointF>
|
||||||
|
<windSpeedMinMPH>1</windSpeedMinMPH>
|
||||||
|
<avgFeelslikeC>24</avgFeelslikeC>
|
||||||
|
<uvi>6</uvi>
|
||||||
|
<windDirMax80m>NW</windDirMax80m>
|
||||||
|
<maxDewpointC>15</maxDewpointC>
|
||||||
|
<pressureIN>29.95</pressureIN>
|
||||||
|
<avgFeelslikeF>76</avgFeelslikeF>
|
||||||
|
<iceaccum>null</iceaccum>
|
||||||
|
<isDay>true</isDay>
|
||||||
|
<minTempC>19</minTempC>
|
||||||
|
<minDewpointF>50</minDewpointF>
|
||||||
|
<windSpeedKTS>1</windSpeedKTS>
|
||||||
|
<sunset>1665156831</sunset>
|
||||||
|
<solradWM2>5486</solradWM2>
|
||||||
|
<windSpeedKPH>2</windSpeedKPH>
|
||||||
|
<windGustMPH>18</windGustMPH>
|
||||||
|
<maxHumidity>77</maxHumidity>
|
||||||
|
<windSpeedMPH>1</windSpeedMPH>
|
||||||
|
<windGustKPH>29</windGustKPH>
|
||||||
|
<windDirMaxDEG>298</windDirMaxDEG>
|
||||||
|
<tempC>null</tempC>
|
||||||
|
</periods>
|
||||||
|
<periods>
|
||||||
|
<dateTimeISO>2022-10-08T07:00:00+02:00</dateTimeISO>
|
||||||
|
<windDirMin80m>NW</windDirMin80m>
|
||||||
|
<windDirMin80mDEG>309</windDirMin80mDEG>
|
||||||
|
<feelslikeC>19</feelslikeC>
|
||||||
|
<visibilityMI>15</visibilityMI>
|
||||||
|
<windSpeedMaxMPH>8</windSpeedMaxMPH>
|
||||||
|
<windDirDEG>21</windDirDEG>
|
||||||
|
<windDir>NNE</windDir>
|
||||||
|
<sunriseISO>2022-10-08T05:52:26+02:00</sunriseISO>
|
||||||
|
<iceaccumMM>null</iceaccumMM>
|
||||||
|
<windSpeedMaxKTS>7</windSpeedMaxKTS>
|
||||||
|
<iceaccumIN>null</iceaccumIN>
|
||||||
|
<minTempF>66</minTempF>
|
||||||
|
<snowIN>0</snowIN>
|
||||||
|
<weather>Partly Cloudy</weather>
|
||||||
|
<sunsetISO>2022-10-08T17:32:41+02:00</sunsetISO>
|
||||||
|
<maxFeelslikeC>30</maxFeelslikeC>
|
||||||
|
<humidity>76</humidity>
|
||||||
|
<windDir80m>NW</windDir80m>
|
||||||
|
<maxFeelslikeF>86</maxFeelslikeF>
|
||||||
|
<precipMM>0</precipMM>
|
||||||
|
<sky>47</sky>
|
||||||
|
<windGust80mMPH>19</windGust80mMPH>
|
||||||
|
<windSpeedMax80mMPH>19</windSpeedMax80mMPH>
|
||||||
|
<weatherPrimary>Partly Cloudy</weatherPrimary>
|
||||||
|
<windGust80mKPH>31</windGust80mKPH>
|
||||||
|
<avgDewpointF>56</avgDewpointF>
|
||||||
|
<windSpeedMax80mKPH>31</windSpeedMax80mKPH>
|
||||||
|
<windGust80mKTS>17</windGust80mKTS>
|
||||||
|
<avgDewpointC>13</avgDewpointC>
|
||||||
|
<precipIN>0</precipIN>
|
||||||
|
<windSpeedMax80mKTS>17</windSpeedMax80mKTS>
|
||||||
|
<windDirMinDEG>21</windDirMinDEG>
|
||||||
|
<windSpeedMaxKPH>13</windSpeedMaxKPH>
|
||||||
|
<windSpeedMin80mKTS>5</windSpeedMin80mKTS>
|
||||||
|
<feelslikeF>67</feelslikeF>
|
||||||
|
<validTime>2022-10-08T07:00:00+02:00</validTime>
|
||||||
|
<windSpeedMin80mMPH>5</windSpeedMin80mMPH>
|
||||||
|
<solradMaxWM2>682</solradMaxWM2>
|
||||||
|
<avgTempC>25</avgTempC>
|
||||||
|
<windSpeedMin80mKPH>9</windSpeedMin80mKPH>
|
||||||
|
<weatherPrimaryCoded>::SC</weatherPrimaryCoded>
|
||||||
|
<sunrise>1665201146</sunrise>
|
||||||
|
<avgTempF>76</avgTempF>
|
||||||
|
<windDirMin>NNE</windDirMin>
|
||||||
|
<maxCoverage/>
|
||||||
|
<icon>pcloudy.png</icon>
|
||||||
|
<minFeelslikeC>19</minFeelslikeC>
|
||||||
|
<dewpointC>15</dewpointC>
|
||||||
|
<cloudsCoded>SC</cloudsCoded>
|
||||||
|
<minFeelslikeF>67</minFeelslikeF>
|
||||||
|
<minHumidity>32</minHumidity>
|
||||||
|
<dewpointF>59</dewpointF>
|
||||||
|
<windSpeed80mKTS>5</windSpeed80mKTS>
|
||||||
|
<pop>0</pop>
|
||||||
|
<snowCM>0</snowCM>
|
||||||
|
<windDirMax>WNW</windDirMax>
|
||||||
|
<windSpeed80mMPH>5</windSpeed80mMPH>
|
||||||
|
<windSpeed80mKPH>9</windSpeed80mKPH>
|
||||||
|
<windDir80mDEG>309</windDir80mDEG>
|
||||||
|
<maxTempC>30</maxTempC>
|
||||||
|
<pressureMB>1014</pressureMB>
|
||||||
|
<visibilityKM>24.135</visibilityKM>
|
||||||
|
<timestamp>1665205200</timestamp>
|
||||||
|
<maxTempF>87</maxTempF>
|
||||||
|
<tempF>null</tempF>
|
||||||
|
<minDewpointC>11</minDewpointC>
|
||||||
|
<solradMinWM2>0</solradMinWM2>
|
||||||
|
<windSpeedMinKTS>1</windSpeedMinKTS>
|
||||||
|
<windDirMax80mDEG>322</windDirMax80mDEG>
|
||||||
|
<windGustKTS>17</windGustKTS>
|
||||||
|
<windSpeedMinKPH>2</windSpeedMinKPH>
|
||||||
|
<maxDewpointF>59</maxDewpointF>
|
||||||
|
<windSpeedMinMPH>1</windSpeedMinMPH>
|
||||||
|
<avgFeelslikeC>25</avgFeelslikeC>
|
||||||
|
<uvi>7</uvi>
|
||||||
|
<windDirMax80m>NW</windDirMax80m>
|
||||||
|
<maxDewpointC>15</maxDewpointC>
|
||||||
|
<pressureIN>29.94</pressureIN>
|
||||||
|
<avgFeelslikeF>76</avgFeelslikeF>
|
||||||
|
<iceaccum>null</iceaccum>
|
||||||
|
<isDay>true</isDay>
|
||||||
|
<minTempC>19</minTempC>
|
||||||
|
<minDewpointF>52</minDewpointF>
|
||||||
|
<windSpeedKTS>1</windSpeedKTS>
|
||||||
|
<sunset>1665243161</sunset>
|
||||||
|
<solradWM2>4785</solradWM2>
|
||||||
|
<windSpeedKPH>2</windSpeedKPH>
|
||||||
|
<windGustMPH>20</windGustMPH>
|
||||||
|
<maxHumidity>76</maxHumidity>
|
||||||
|
<windSpeedMPH>1</windSpeedMPH>
|
||||||
|
<windGustKPH>32</windGustKPH>
|
||||||
|
<windDirMaxDEG>301</windDirMaxDEG>
|
||||||
|
<tempC>null</tempC>
|
||||||
|
</periods>
|
||||||
|
<periods>
|
||||||
|
<dateTimeISO>2022-10-09T07:00:00+02:00</dateTimeISO>
|
||||||
|
<windDirMin80m>NW</windDirMin80m>
|
||||||
|
<windDirMin80mDEG>316</windDirMin80mDEG>
|
||||||
|
<feelslikeC>20</feelslikeC>
|
||||||
|
<visibilityMI>15</visibilityMI>
|
||||||
|
<windSpeedMaxMPH>9</windSpeedMaxMPH>
|
||||||
|
<windDirDEG>356</windDirDEG>
|
||||||
|
<windDir>N</windDir>
|
||||||
|
<sunriseISO>2022-10-09T05:53:03+02:00</sunriseISO>
|
||||||
|
<iceaccumMM>null</iceaccumMM>
|
||||||
|
<windSpeedMaxKTS>8</windSpeedMaxKTS>
|
||||||
|
<iceaccumIN>null</iceaccumIN>
|
||||||
|
<minTempF>67</minTempF>
|
||||||
|
<snowIN>0</snowIN>
|
||||||
|
<weather>Partly Cloudy</weather>
|
||||||
|
<sunsetISO>2022-10-09T17:31:31+02:00</sunsetISO>
|
||||||
|
<maxFeelslikeC>30</maxFeelslikeC>
|
||||||
|
<humidity>86</humidity>
|
||||||
|
<windDir80m>NW</windDir80m>
|
||||||
|
<maxFeelslikeF>86</maxFeelslikeF>
|
||||||
|
<precipMM>0</precipMM>
|
||||||
|
<sky>47</sky>
|
||||||
|
<windGust80mMPH>23</windGust80mMPH>
|
||||||
|
<windSpeedMax80mMPH>23</windSpeedMax80mMPH>
|
||||||
|
<weatherPrimary>Partly Cloudy</weatherPrimary>
|
||||||
|
<windGust80mKPH>36</windGust80mKPH>
|
||||||
|
<avgDewpointF>57</avgDewpointF>
|
||||||
|
<windSpeedMax80mKPH>36</windSpeedMax80mKPH>
|
||||||
|
<windGust80mKTS>20</windGust80mKTS>
|
||||||
|
<avgDewpointC>14</avgDewpointC>
|
||||||
|
<precipIN>0</precipIN>
|
||||||
|
<windSpeedMax80mKTS>20</windSpeedMax80mKTS>
|
||||||
|
<windDirMinDEG>356</windDirMinDEG>
|
||||||
|
<windSpeedMaxKPH>14</windSpeedMaxKPH>
|
||||||
|
<windSpeedMin80mKTS>5</windSpeedMin80mKTS>
|
||||||
|
<feelslikeF>67</feelslikeF>
|
||||||
|
<validTime>2022-10-09T07:00:00+02:00</validTime>
|
||||||
|
<windSpeedMin80mMPH>6</windSpeedMin80mMPH>
|
||||||
|
<solradMaxWM2>726</solradMaxWM2>
|
||||||
|
<avgTempC>25</avgTempC>
|
||||||
|
<windSpeedMin80mKPH>9</windSpeedMin80mKPH>
|
||||||
|
<weatherPrimaryCoded>::SC</weatherPrimaryCoded>
|
||||||
|
<sunrise>1665287583</sunrise>
|
||||||
|
<avgTempF>77</avgTempF>
|
||||||
|
<windDirMin>N</windDirMin>
|
||||||
|
<maxCoverage/>
|
||||||
|
<icon>pcloudy.png</icon>
|
||||||
|
<minFeelslikeC>20</minFeelslikeC>
|
||||||
|
<dewpointC>17</dewpointC>
|
||||||
|
<cloudsCoded>SC</cloudsCoded>
|
||||||
|
<minFeelslikeF>67</minFeelslikeF>
|
||||||
|
<minHumidity>31</minHumidity>
|
||||||
|
<dewpointF>63</dewpointF>
|
||||||
|
<windSpeed80mKTS>5</windSpeed80mKTS>
|
||||||
|
<pop>0</pop>
|
||||||
|
<snowCM>0</snowCM>
|
||||||
|
<windDirMax>NNW</windDirMax>
|
||||||
|
<windSpeed80mMPH>6</windSpeed80mMPH>
|
||||||
|
<windSpeed80mKPH>9</windSpeed80mKPH>
|
||||||
|
<windDir80mDEG>316</windDir80mDEG>
|
||||||
|
<maxTempC>31</maxTempC>
|
||||||
|
<pressureMB>1016</pressureMB>
|
||||||
|
<visibilityKM>24.135</visibilityKM>
|
||||||
|
<timestamp>1665291600</timestamp>
|
||||||
|
<maxTempF>87</maxTempF>
|
||||||
|
<tempF>null</tempF>
|
||||||
|
<minDewpointC>11</minDewpointC>
|
||||||
|
<solradMinWM2>0</solradMinWM2>
|
||||||
|
<windSpeedMinKTS>2</windSpeedMinKTS>
|
||||||
|
<windDirMax80mDEG>354</windDirMax80mDEG>
|
||||||
|
<windGustKTS>19</windGustKTS>
|
||||||
|
<windSpeedMinKPH>4</windSpeedMinKPH>
|
||||||
|
<maxDewpointF>63</maxDewpointF>
|
||||||
|
<windSpeedMinMPH>2</windSpeedMinMPH>
|
||||||
|
<avgFeelslikeC>25</avgFeelslikeC>
|
||||||
|
<uvi>7</uvi>
|
||||||
|
<windDirMax80m>N</windDirMax80m>
|
||||||
|
<maxDewpointC>17</maxDewpointC>
|
||||||
|
<pressureIN>29.99</pressureIN>
|
||||||
|
<avgFeelslikeF>77</avgFeelslikeF>
|
||||||
|
<iceaccum>null</iceaccum>
|
||||||
|
<isDay>true</isDay>
|
||||||
|
<minTempC>19</minTempC>
|
||||||
|
<minDewpointF>52</minDewpointF>
|
||||||
|
<windSpeedKTS>2</windSpeedKTS>
|
||||||
|
<sunset>1665329491</sunset>
|
||||||
|
<solradWM2>4768</solradWM2>
|
||||||
|
<windSpeedKPH>4</windSpeedKPH>
|
||||||
|
<windGustMPH>22</windGustMPH>
|
||||||
|
<maxHumidity>86</maxHumidity>
|
||||||
|
<windSpeedMPH>2</windSpeedMPH>
|
||||||
|
<windGustKPH>36</windGustKPH>
|
||||||
|
<windDirMaxDEG>343</windDirMaxDEG>
|
||||||
|
<tempC>null</tempC>
|
||||||
|
</periods>
|
||||||
|
<periods>
|
||||||
|
<dateTimeISO>2022-10-10T07:00:00+02:00</dateTimeISO>
|
||||||
|
<windDirMin80m>E</windDirMin80m>
|
||||||
|
<windDirMin80mDEG>91</windDirMin80mDEG>
|
||||||
|
<feelslikeC>21</feelslikeC>
|
||||||
|
<visibilityMI>15</visibilityMI>
|
||||||
|
<windSpeedMaxMPH>9</windSpeedMaxMPH>
|
||||||
|
<windDirDEG>358</windDirDEG>
|
||||||
|
<windDir>N</windDir>
|
||||||
|
<sunriseISO>2022-10-10T05:53:40+02:00</sunriseISO>
|
||||||
|
<iceaccumMM>null</iceaccumMM>
|
||||||
|
<windSpeedMaxKTS>8</windSpeedMaxKTS>
|
||||||
|
<iceaccumIN>null</iceaccumIN>
|
||||||
|
<minTempF>70</minTempF>
|
||||||
|
<snowIN>0</snowIN>
|
||||||
|
<weather>Partly Cloudy</weather>
|
||||||
|
<sunsetISO>2022-10-10T17:30:21+02:00</sunsetISO>
|
||||||
|
<maxFeelslikeC>30</maxFeelslikeC>
|
||||||
|
<humidity>75</humidity>
|
||||||
|
<windDir80m>N</windDir80m>
|
||||||
|
<maxFeelslikeF>86</maxFeelslikeF>
|
||||||
|
<precipMM>0</precipMM>
|
||||||
|
<sky>64</sky>
|
||||||
|
<windGust80mMPH>22</windGust80mMPH>
|
||||||
|
<windSpeedMax80mMPH>22</windSpeedMax80mMPH>
|
||||||
|
<weatherPrimary>Partly Cloudy</weatherPrimary>
|
||||||
|
<windGust80mKPH>36</windGust80mKPH>
|
||||||
|
<avgDewpointF>58</avgDewpointF>
|
||||||
|
<windSpeedMax80mKPH>36</windSpeedMax80mKPH>
|
||||||
|
<windGust80mKTS>19</windGust80mKTS>
|
||||||
|
<avgDewpointC>14</avgDewpointC>
|
||||||
|
<precipIN>0</precipIN>
|
||||||
|
<windSpeedMax80mKTS>19</windSpeedMax80mKTS>
|
||||||
|
<windDirMinDEG>358</windDirMinDEG>
|
||||||
|
<windSpeedMaxKPH>15</windSpeedMaxKPH>
|
||||||
|
<windSpeedMin80mKTS>7</windSpeedMin80mKTS>
|
||||||
|
<feelslikeF>69</feelslikeF>
|
||||||
|
<validTime>2022-10-10T07:00:00+02:00</validTime>
|
||||||
|
<windSpeedMin80mMPH>8</windSpeedMin80mMPH>
|
||||||
|
<solradMaxWM2>597</solradMaxWM2>
|
||||||
|
<avgTempC>26</avgTempC>
|
||||||
|
<windSpeedMin80mKPH>13</windSpeedMin80mKPH>
|
||||||
|
<weatherPrimaryCoded>::SC</weatherPrimaryCoded>
|
||||||
|
<sunrise>1665374020</sunrise>
|
||||||
|
<avgTempF>78</avgTempF>
|
||||||
|
<windDirMin>N</windDirMin>
|
||||||
|
<maxCoverage/>
|
||||||
|
<icon>pcloudy.png</icon>
|
||||||
|
<minFeelslikeC>21</minFeelslikeC>
|
||||||
|
<dewpointC>16</dewpointC>
|
||||||
|
<cloudsCoded>SC</cloudsCoded>
|
||||||
|
<minFeelslikeF>69</minFeelslikeF>
|
||||||
|
<minHumidity>35</minHumidity>
|
||||||
|
<dewpointF>61</dewpointF>
|
||||||
|
<windSpeed80mKTS>7</windSpeed80mKTS>
|
||||||
|
<pop>0</pop>
|
||||||
|
<snowCM>0</snowCM>
|
||||||
|
<windDirMax>N</windDirMax>
|
||||||
|
<windSpeed80mMPH>8</windSpeed80mMPH>
|
||||||
|
<windSpeed80mKPH>13</windSpeed80mKPH>
|
||||||
|
<windDir80mDEG>8</windDir80mDEG>
|
||||||
|
<maxTempC>31</maxTempC>
|
||||||
|
<pressureMB>1017</pressureMB>
|
||||||
|
<visibilityKM>24.135</visibilityKM>
|
||||||
|
<timestamp>1665378000</timestamp>
|
||||||
|
<maxTempF>87</maxTempF>
|
||||||
|
<tempF>null</tempF>
|
||||||
|
<minDewpointC>13</minDewpointC>
|
||||||
|
<solradMinWM2>0</solradMinWM2>
|
||||||
|
<windSpeedMinKTS>2</windSpeedMinKTS>
|
||||||
|
<windDirMax80mDEG>10</windDirMax80mDEG>
|
||||||
|
<windGustKTS>16</windGustKTS>
|
||||||
|
<windSpeedMinKPH>4</windSpeedMinKPH>
|
||||||
|
<maxDewpointF>61</maxDewpointF>
|
||||||
|
<windSpeedMinMPH>2</windSpeedMinMPH>
|
||||||
|
<avgFeelslikeC>25</avgFeelslikeC>
|
||||||
|
<uvi>6</uvi>
|
||||||
|
<windDirMax80m>N</windDirMax80m>
|
||||||
|
<maxDewpointC>16</maxDewpointC>
|
||||||
|
<pressureIN>30.03</pressureIN>
|
||||||
|
<avgFeelslikeF>78</avgFeelslikeF>
|
||||||
|
<iceaccum>null</iceaccum>
|
||||||
|
<isDay>true</isDay>
|
||||||
|
<minTempC>21</minTempC>
|
||||||
|
<minDewpointF>55</minDewpointF>
|
||||||
|
<windSpeedKTS>2</windSpeedKTS>
|
||||||
|
<sunset>1665415821</sunset>
|
||||||
|
<solradWM2>4494</solradWM2>
|
||||||
|
<windSpeedKPH>4</windSpeedKPH>
|
||||||
|
<windGustMPH>19</windGustMPH>
|
||||||
|
<maxHumidity>75</maxHumidity>
|
||||||
|
<windSpeedMPH>2</windSpeedMPH>
|
||||||
|
<windGustKPH>30</windGustKPH>
|
||||||
|
<windDirMaxDEG>10</windDirMaxDEG>
|
||||||
|
<tempC>null</tempC>
|
||||||
|
</periods>
|
||||||
|
<periods>
|
||||||
|
<dateTimeISO>2022-10-11T07:00:00+02:00</dateTimeISO>
|
||||||
|
<windDirMin80m>NNE</windDirMin80m>
|
||||||
|
<windDirMin80mDEG>13</windDirMin80mDEG>
|
||||||
|
<feelslikeC>22</feelslikeC>
|
||||||
|
<visibilityMI>15</visibilityMI>
|
||||||
|
<windSpeedMaxMPH>18</windSpeedMaxMPH>
|
||||||
|
<windDirDEG>13</windDirDEG>
|
||||||
|
<windDir>NNE</windDir>
|
||||||
|
<sunriseISO>2022-10-11T05:54:18+02:00</sunriseISO>
|
||||||
|
<iceaccumMM>null</iceaccumMM>
|
||||||
|
<windSpeedMaxKTS>15</windSpeedMaxKTS>
|
||||||
|
<iceaccumIN>null</iceaccumIN>
|
||||||
|
<minTempF>70</minTempF>
|
||||||
|
<snowIN>0</snowIN>
|
||||||
|
<weather>Sunny</weather>
|
||||||
|
<sunsetISO>2022-10-11T17:29:13+02:00</sunsetISO>
|
||||||
|
<maxFeelslikeC>31</maxFeelslikeC>
|
||||||
|
<humidity>71</humidity>
|
||||||
|
<windDir80m>NNE</windDir80m>
|
||||||
|
<maxFeelslikeF>87</maxFeelslikeF>
|
||||||
|
<precipMM>0</precipMM>
|
||||||
|
<sky>0</sky>
|
||||||
|
<windGust80mMPH>19</windGust80mMPH>
|
||||||
|
<windSpeedMax80mMPH>19</windSpeedMax80mMPH>
|
||||||
|
<weatherPrimary>Sunny</weatherPrimary>
|
||||||
|
<windGust80mKPH>31</windGust80mKPH>
|
||||||
|
<avgDewpointF>55</avgDewpointF>
|
||||||
|
<windSpeedMax80mKPH>31</windSpeedMax80mKPH>
|
||||||
|
<windGust80mKTS>17</windGust80mKTS>
|
||||||
|
<avgDewpointC>13</avgDewpointC>
|
||||||
|
<precipIN>0</precipIN>
|
||||||
|
<windSpeedMax80mKTS>17</windSpeedMax80mKTS>
|
||||||
|
<windDirMinDEG>14</windDirMinDEG>
|
||||||
|
<windSpeedMaxKPH>28</windSpeedMaxKPH>
|
||||||
|
<windSpeedMin80mKTS>9</windSpeedMin80mKTS>
|
||||||
|
<feelslikeF>72</feelslikeF>
|
||||||
|
<validTime>2022-10-11T07:00:00+02:00</validTime>
|
||||||
|
<windSpeedMin80mMPH>11</windSpeedMin80mMPH>
|
||||||
|
<solradMaxWM2>758</solradMaxWM2>
|
||||||
|
<avgTempC>26</avgTempC>
|
||||||
|
<windSpeedMin80mKPH>18</windSpeedMin80mKPH>
|
||||||
|
<weatherPrimaryCoded>::CL</weatherPrimaryCoded>
|
||||||
|
<sunrise>1665460458</sunrise>
|
||||||
|
<avgTempF>78</avgTempF>
|
||||||
|
<windDirMin>NNE</windDirMin>
|
||||||
|
<maxCoverage/>
|
||||||
|
<icon>sunny.png</icon>
|
||||||
|
<minFeelslikeC>22</minFeelslikeC>
|
||||||
|
<dewpointC>17</dewpointC>
|
||||||
|
<cloudsCoded>CL</cloudsCoded>
|
||||||
|
<minFeelslikeF>72</minFeelslikeF>
|
||||||
|
<minHumidity>30</minHumidity>
|
||||||
|
<dewpointF>62</dewpointF>
|
||||||
|
<windSpeed80mKTS>10</windSpeed80mKTS>
|
||||||
|
<pop>0</pop>
|
||||||
|
<snowCM>0</snowCM>
|
||||||
|
<windDirMax>NNE</windDirMax>
|
||||||
|
<windSpeed80mMPH>12</windSpeed80mMPH>
|
||||||
|
<windSpeed80mKPH>19</windSpeed80mKPH>
|
||||||
|
<windDir80mDEG>16</windDir80mDEG>
|
||||||
|
<maxTempC>31</maxTempC>
|
||||||
|
<pressureMB>1015</pressureMB>
|
||||||
|
<visibilityKM>24.135</visibilityKM>
|
||||||
|
<timestamp>1665464400</timestamp>
|
||||||
|
<maxTempF>87</maxTempF>
|
||||||
|
<tempF>null</tempF>
|
||||||
|
<minDewpointC>11</minDewpointC>
|
||||||
|
<solradMinWM2>0</solradMinWM2>
|
||||||
|
<windSpeedMinKTS>7</windSpeedMinKTS>
|
||||||
|
<windDirMax80mDEG>28</windDirMax80mDEG>
|
||||||
|
<windGustKTS>15</windGustKTS>
|
||||||
|
<windSpeedMinKPH>14</windSpeedMinKPH>
|
||||||
|
<maxDewpointF>62</maxDewpointF>
|
||||||
|
<windSpeedMinMPH>8</windSpeedMinMPH>
|
||||||
|
<avgFeelslikeC>26</avgFeelslikeC>
|
||||||
|
<uvi>null</uvi>
|
||||||
|
<windDirMax80m>NNE</windDirMax80m>
|
||||||
|
<maxDewpointC>17</maxDewpointC>
|
||||||
|
<pressureIN>29.98</pressureIN>
|
||||||
|
<avgFeelslikeF>79</avgFeelslikeF>
|
||||||
|
<iceaccum>null</iceaccum>
|
||||||
|
<isDay>true</isDay>
|
||||||
|
<minTempC>21</minTempC>
|
||||||
|
<minDewpointF>51</minDewpointF>
|
||||||
|
<windSpeedKTS>8</windSpeedKTS>
|
||||||
|
<sunset>1665502153</sunset>
|
||||||
|
<solradWM2>5450</solradWM2>
|
||||||
|
<windSpeedKPH>15</windSpeedKPH>
|
||||||
|
<windGustMPH>17</windGustMPH>
|
||||||
|
<maxHumidity>71</maxHumidity>
|
||||||
|
<windSpeedMPH>9</windSpeedMPH>
|
||||||
|
<windGustKPH>28</windGustKPH>
|
||||||
|
<windDirMaxDEG>28</windDirMaxDEG>
|
||||||
|
<tempC>null</tempC>
|
||||||
|
</periods>
|
||||||
|
<periods>
|
||||||
|
<dateTimeISO>2022-10-12T07:00:00+02:00</dateTimeISO>
|
||||||
|
<windDirMin80m>NNE</windDirMin80m>
|
||||||
|
<windDirMin80mDEG>15</windDirMin80mDEG>
|
||||||
|
<feelslikeC>22</feelslikeC>
|
||||||
|
<visibilityMI>15</visibilityMI>
|
||||||
|
<windSpeedMaxMPH>16</windSpeedMaxMPH>
|
||||||
|
<windDirDEG>12</windDirDEG>
|
||||||
|
<windDir>NNE</windDir>
|
||||||
|
<sunriseISO>2022-10-12T05:54:55+02:00</sunriseISO>
|
||||||
|
<iceaccumMM>null</iceaccumMM>
|
||||||
|
<windSpeedMaxKTS>14</windSpeedMaxKTS>
|
||||||
|
<iceaccumIN>null</iceaccumIN>
|
||||||
|
<minTempF>69</minTempF>
|
||||||
|
<snowIN>0</snowIN>
|
||||||
|
<weather>Mostly Sunny</weather>
|
||||||
|
<sunsetISO>2022-10-12T17:28:04+02:00</sunsetISO>
|
||||||
|
<maxFeelslikeC>31</maxFeelslikeC>
|
||||||
|
<humidity>68</humidity>
|
||||||
|
<windDir80m>NNE</windDir80m>
|
||||||
|
<maxFeelslikeF>88</maxFeelslikeF>
|
||||||
|
<precipMM>0</precipMM>
|
||||||
|
<sky>27</sky>
|
||||||
|
<windGust80mMPH>21</windGust80mMPH>
|
||||||
|
<windSpeedMax80mMPH>21</windSpeedMax80mMPH>
|
||||||
|
<weatherPrimary>Mostly Sunny</weatherPrimary>
|
||||||
|
<windGust80mKPH>33</windGust80mKPH>
|
||||||
|
<avgDewpointF>55</avgDewpointF>
|
||||||
|
<windSpeedMax80mKPH>33</windSpeedMax80mKPH>
|
||||||
|
<windGust80mKTS>18</windGust80mKTS>
|
||||||
|
<avgDewpointC>13</avgDewpointC>
|
||||||
|
<precipIN>0</precipIN>
|
||||||
|
<windSpeedMax80mKTS>18</windSpeedMax80mKTS>
|
||||||
|
<windDirMinDEG>12</windDirMinDEG>
|
||||||
|
<windSpeedMaxKPH>26</windSpeedMaxKPH>
|
||||||
|
<windSpeedMin80mKTS>10</windSpeedMin80mKTS>
|
||||||
|
<feelslikeF>72</feelslikeF>
|
||||||
|
<validTime>2022-10-12T07:00:00+02:00</validTime>
|
||||||
|
<windSpeedMin80mMPH>11</windSpeedMin80mMPH>
|
||||||
|
<solradMaxWM2>743</solradMaxWM2>
|
||||||
|
<avgTempC>26</avgTempC>
|
||||||
|
<windSpeedMin80mKPH>18</windSpeedMin80mKPH>
|
||||||
|
<weatherPrimaryCoded>::FW</weatherPrimaryCoded>
|
||||||
|
<sunrise>1665546895</sunrise>
|
||||||
|
<avgTempF>79</avgTempF>
|
||||||
|
<windDirMin>NNE</windDirMin>
|
||||||
|
<maxCoverage/>
|
||||||
|
<icon>fair.png</icon>
|
||||||
|
<minFeelslikeC>22</minFeelslikeC>
|
||||||
|
<dewpointC>16</dewpointC>
|
||||||
|
<cloudsCoded>FW</cloudsCoded>
|
||||||
|
<minFeelslikeF>72</minFeelslikeF>
|
||||||
|
<minHumidity>29</minHumidity>
|
||||||
|
<dewpointF>60</dewpointF>
|
||||||
|
<windSpeed80mKTS>10</windSpeed80mKTS>
|
||||||
|
<pop>0</pop>
|
||||||
|
<snowCM>0</snowCM>
|
||||||
|
<windDirMax>E</windDirMax>
|
||||||
|
<windSpeed80mMPH>12</windSpeed80mMPH>
|
||||||
|
<windSpeed80mKPH>19</windSpeed80mKPH>
|
||||||
|
<windDir80mDEG>15</windDir80mDEG>
|
||||||
|
<maxTempC>31</maxTempC>
|
||||||
|
<pressureMB>1014</pressureMB>
|
||||||
|
<visibilityKM>24.135</visibilityKM>
|
||||||
|
<timestamp>1665550800</timestamp>
|
||||||
|
<maxTempF>88</maxTempF>
|
||||||
|
<tempF>null</tempF>
|
||||||
|
<minDewpointC>11</minDewpointC>
|
||||||
|
<solradMinWM2>0</solradMinWM2>
|
||||||
|
<windSpeedMinKTS>7</windSpeedMinKTS>
|
||||||
|
<windDirMax80mDEG>96</windDirMax80mDEG>
|
||||||
|
<windGustKTS>15</windGustKTS>
|
||||||
|
<windSpeedMinKPH>13</windSpeedMinKPH>
|
||||||
|
<maxDewpointF>60</maxDewpointF>
|
||||||
|
<windSpeedMinMPH>8</windSpeedMinMPH>
|
||||||
|
<avgFeelslikeC>26</avgFeelslikeC>
|
||||||
|
<uvi>null</uvi>
|
||||||
|
<windDirMax80m>E</windDirMax80m>
|
||||||
|
<maxDewpointC>16</maxDewpointC>
|
||||||
|
<pressureIN>29.95</pressureIN>
|
||||||
|
<avgFeelslikeF>80</avgFeelslikeF>
|
||||||
|
<iceaccum>null</iceaccum>
|
||||||
|
<isDay>true</isDay>
|
||||||
|
<minTempC>21</minTempC>
|
||||||
|
<minDewpointF>51</minDewpointF>
|
||||||
|
<windSpeedKTS>8</windSpeedKTS>
|
||||||
|
<sunset>1665588484</sunset>
|
||||||
|
<solradWM2>4740</solradWM2>
|
||||||
|
<windSpeedKPH>15</windSpeedKPH>
|
||||||
|
<windGustMPH>17</windGustMPH>
|
||||||
|
<maxHumidity>68</maxHumidity>
|
||||||
|
<windSpeedMPH>9</windSpeedMPH>
|
||||||
|
<windGustKPH>28</windGustKPH>
|
||||||
|
<windDirMaxDEG>96</windDirMaxDEG>
|
||||||
|
<tempC>null</tempC>
|
||||||
|
</periods>
|
||||||
|
<interval>day</interval>
|
||||||
|
<place>
|
||||||
|
<country>eg</country>
|
||||||
|
<name>cairo</name>
|
||||||
|
<state>qh</state>
|
||||||
|
</place>
|
||||||
|
</response>
|
||||||
|
<error>null</error>
|
||||||
822
src/test/resources/Issue654WellFormedArray.json
Normal file
822
src/test/resources/Issue654WellFormedArray.json
Normal file
@@ -0,0 +1,822 @@
|
|||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",
|
||||||
|
["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",["a",[]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
|
]]]]]]]]]]]]
|
||||||
822
src/test/resources/Issue654WellFormedObject.json
Normal file
822
src/test/resources/Issue654WellFormedObject.json
Normal file
@@ -0,0 +1,822 @@
|
|||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":
|
||||||
|
{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
|
}}}}}}}}}}}}
|
||||||
Reference in New Issue
Block a user