mirror of
https://github.com/stleary/JSON-java.git
synced 2026-01-24 00:03:17 -05:00
Compare commits
169 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
8e5b516f2b | ||
|
|
6b6e8e85d8 | ||
|
|
6ecbeaa0d2 | ||
|
|
3ac647a16f | ||
|
|
6f06801296 | ||
|
|
dd7056cb6d | ||
|
|
e62d763294 | ||
|
|
4990c3a180 | ||
|
|
4b49bc94ce | ||
|
|
9afa90d9a9 | ||
|
|
065f9a94bc | ||
|
|
223e328161 | ||
|
|
044b035847 | ||
|
|
5b62cf13ff | ||
|
|
e9c27ab376 | ||
|
|
328e7d8944 | ||
|
|
2a6af29235 | ||
|
|
af6d3c63bd | ||
|
|
c3ea0249c9 | ||
|
|
b044b7db4d | ||
|
|
115f8b660b | ||
|
|
f63d21fd13 | ||
|
|
c8ae720caf | ||
|
|
4d451468fd | ||
|
|
a03a01531a | ||
|
|
5b845f28cf | ||
|
|
00e0e6c0a2 | ||
|
|
d5b278539e | ||
|
|
12bbe8cd9a | ||
|
|
09dddb826e | ||
|
|
19e9bb6c07 | ||
|
|
fea0aca2ab | ||
|
|
1a811f1ada | ||
|
|
34cfe6df14 | ||
|
|
71c6dd1e34 | ||
|
|
30c1bd16ba | ||
|
|
bc347d2c19 | ||
|
|
a63fa03062 | ||
|
|
16225efbdd | ||
|
|
b8a3342eb1 | ||
|
|
37f5bf28e9 | ||
|
|
7a17ae0b3e | ||
|
|
7cad4c3b26 | ||
|
|
05074386d3 | ||
|
|
a490ebdb78 | ||
|
|
3c1535d724 | ||
|
|
a6284df9c7 | ||
|
|
bfb300835f | ||
|
|
ca9df04539 | ||
|
|
06e9ad280f | ||
|
|
2362c930d1 | ||
|
|
2a6b5bacc5 | ||
|
|
a509a28ed4 | ||
|
|
1c1ef5b211 | ||
|
|
74b9a60f98 | ||
|
|
b63b976acb | ||
|
|
97e180444d | ||
|
|
d402a99fd8 | ||
|
|
7073bc8c47 | ||
|
|
61cdfefc36 | ||
|
|
fbad2d0017 | ||
|
|
15719886f7 | ||
|
|
ca45b02ffc | ||
|
|
b6efbabc32 | ||
|
|
9eb8c27724 | ||
|
|
195963357c | ||
|
|
28efdb4860 | ||
|
|
c88653ca2e | ||
|
|
b3068d9fe4 | ||
|
|
dba4afd0cf | ||
|
|
26160e1619 | ||
|
|
b7e2eee4d6 | ||
|
|
28e09dc493 | ||
|
|
4a4b2db8c1 | ||
|
|
f16682bf44 | ||
|
|
18952b5ac0 | ||
|
|
d2a66a4287 | ||
|
|
722003d479 | ||
|
|
ed8745cd63 | ||
|
|
057e0c75ca | ||
|
|
cdf3cf7f81 | ||
|
|
2565abdaaa | ||
|
|
de855c50aa | ||
|
|
4cb1ae802a | ||
|
|
2e0a8137bd | ||
|
|
f177c97258 | ||
|
|
7d8353401a | ||
|
|
7fed023080 | ||
|
|
d9b8507e6a | ||
|
|
d345bc528e | ||
|
|
6f238a3698 | ||
|
|
4dbc5ef803 | ||
|
|
5c80c9157d | ||
|
|
a129ebe8e4 | ||
|
|
641b68dd55 | ||
|
|
643b25140f | ||
|
|
3997a90d58 | ||
|
|
1736a60ffe | ||
|
|
e8b1b66888 | ||
|
|
5024f2d210 | ||
|
|
16baa323cf | ||
|
|
52845366bd | ||
|
|
e7e6ed9205 | ||
|
|
1add1247fa | ||
|
|
5b2e5e7579 | ||
|
|
c9ae1f17d7 | ||
|
|
246350bbcd | ||
|
|
2fbe4d96cf | ||
|
|
3645f91b55 | ||
|
|
9c092753b0 | ||
|
|
d0f5607998 | ||
|
|
ad6bdd715d | ||
|
|
ef7a5e40be | ||
|
|
237bf0adb6 | ||
|
|
f76fbe7005 | ||
|
|
4f5bf16676 | ||
|
|
fbd2be7431 | ||
|
|
757b6edb03 | ||
|
|
f2b642a1ca | ||
|
|
04d6e83fc2 | ||
|
|
849b392c01 | ||
|
|
a7f8ff24df | ||
|
|
1ab5260a7a | ||
|
|
c28a2bdf39 | ||
|
|
382f62e781 | ||
|
|
0c7bd725a6 | ||
|
|
fcdb8671b2 | ||
|
|
c46774cf13 | ||
|
|
bd4b180f4e | ||
|
|
a8d4e4734f | ||
|
|
4865f51dd5 | ||
|
|
c870094f69 | ||
|
|
cbd8b18c4a | ||
|
|
f12fa9ba5f | ||
|
|
ae1e9e2b6a | ||
|
|
80e2ea2a80 | ||
|
|
2917104b53 | ||
|
|
9e0fc5e680 | ||
|
|
724fb888f7 | ||
|
|
eb806f4c14 | ||
|
|
5ff8b4cb08 | ||
|
|
5ef4f58ef1 | ||
|
|
413bb53b48 | ||
|
|
237376eca6 | ||
|
|
e0616a129e | ||
|
|
93ffca36c3 | ||
|
|
e477d7002b | ||
|
|
fb1db9341e | ||
|
|
adb0478f66 | ||
|
|
f58a0f4684 | ||
|
|
c11e09959c | ||
|
|
68f92eb395 | ||
|
|
34652a8706 | ||
|
|
a2d3b59394 | ||
|
|
c24be0e4ea | ||
|
|
88f65c5bea | ||
|
|
ebe69df8e4 | ||
|
|
2f2cd4dfc5 | ||
|
|
349a209df3 | ||
|
|
7851e9b2e8 | ||
|
|
7232a95c0b | ||
|
|
f96f505188 | ||
|
|
91107e3e82 | ||
|
|
4e8e24d49d | ||
|
|
f881b61c81 | ||
|
|
37582a44ad | ||
|
|
154cfda9aa | ||
|
|
45a7decba4 | ||
|
|
0c157cae75 |
6
.gitignore
vendored
Normal file
6
.gitignore
vendored
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
# ignore eclipse project files
|
||||||
|
.project
|
||||||
|
.classpath
|
||||||
|
# ignore Intellij Idea project files
|
||||||
|
.idea
|
||||||
|
*.iml
|
||||||
17
CDL.java
17
CDL.java
@@ -22,7 +22,7 @@ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
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
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This provides static methods to convert comma delimited text into a
|
* This provides static methods to convert comma delimited text into a
|
||||||
@@ -55,7 +55,7 @@ public class CDL {
|
|||||||
private static String getValue(JSONTokener x) throws JSONException {
|
private static String getValue(JSONTokener x) throws JSONException {
|
||||||
char c;
|
char c;
|
||||||
char q;
|
char q;
|
||||||
StringBuffer sb;
|
StringBuilder sb;
|
||||||
do {
|
do {
|
||||||
c = x.next();
|
c = x.next();
|
||||||
} while (c == ' ' || c == '\t');
|
} while (c == ' ' || c == '\t');
|
||||||
@@ -65,14 +65,17 @@ public class CDL {
|
|||||||
case '"':
|
case '"':
|
||||||
case '\'':
|
case '\'':
|
||||||
q = c;
|
q = c;
|
||||||
sb = new StringBuffer();
|
sb = new StringBuilder();
|
||||||
for (;;) {
|
for (;;) {
|
||||||
c = x.next();
|
c = x.next();
|
||||||
if (c == q) {
|
if (c == q) {
|
||||||
//Handle escaped double-quote
|
//Handle escaped double-quote
|
||||||
if(x.next() != '\"')
|
char nextC = x.next();
|
||||||
{
|
if(nextC != '\"') {
|
||||||
x.back();
|
// if our quote was the end of the file, don't step
|
||||||
|
if(nextC > 0) {
|
||||||
|
x.back();
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -272,7 +275,7 @@ public class CDL {
|
|||||||
if (names == null || names.length() == 0) {
|
if (names == null || names.length() == 0) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
StringBuffer sb = new StringBuffer();
|
StringBuilder sb = new StringBuilder();
|
||||||
for (int i = 0; i < ja.length(); i += 1) {
|
for (int i = 0; i < ja.length(); i += 1) {
|
||||||
JSONObject jo = ja.optJSONObject(i);
|
JSONObject jo = ja.optJSONObject(i);
|
||||||
if (jo != null) {
|
if (jo != null) {
|
||||||
|
|||||||
@@ -22,9 +22,7 @@ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
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
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Iterator;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a web browser cookie list string to a JSONObject and back.
|
* Convert a web browser cookie list string to a JSONObject and back.
|
||||||
@@ -39,7 +37,7 @@ public class CookieList {
|
|||||||
* The pairs are separated by ';'. The names and the values
|
* The pairs are separated by ';'. The names and the values
|
||||||
* will be unescaped, possibly converting '+' and '%' sequences.
|
* will be unescaped, possibly converting '+' and '%' sequences.
|
||||||
*
|
*
|
||||||
* To add a cookie to a cooklist,
|
* To add a cookie to a cookie list,
|
||||||
* cookielistJSONObject.put(cookieJSONObject.getString("name"),
|
* cookielistJSONObject.put(cookieJSONObject.getString("name"),
|
||||||
* cookieJSONObject.getString("value"));
|
* cookieJSONObject.getString("value"));
|
||||||
* @param string A cookie list string
|
* @param string A cookie list string
|
||||||
@@ -69,18 +67,17 @@ public class CookieList {
|
|||||||
*/
|
*/
|
||||||
public static String toString(JSONObject jo) throws JSONException {
|
public static String toString(JSONObject jo) throws JSONException {
|
||||||
boolean b = false;
|
boolean b = false;
|
||||||
Iterator<String> keys = jo.keys();
|
final StringBuilder sb = new StringBuilder();
|
||||||
String string;
|
// Don't use the new entrySet API to maintain Android support
|
||||||
StringBuilder sb = new StringBuilder();
|
for (final String key : jo.keySet()) {
|
||||||
while (keys.hasNext()) {
|
final Object value = jo.opt(key);
|
||||||
string = keys.next();
|
if (!JSONObject.NULL.equals(value)) {
|
||||||
if (!jo.isNull(string)) {
|
|
||||||
if (b) {
|
if (b) {
|
||||||
sb.append(';');
|
sb.append(';');
|
||||||
}
|
}
|
||||||
sb.append(Cookie.escape(string));
|
sb.append(Cookie.escape(key));
|
||||||
sb.append("=");
|
sb.append("=");
|
||||||
sb.append(Cookie.escape(jo.getString(string)));
|
sb.append(Cookie.escape(value.toString()));
|
||||||
b = true;
|
b = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
21
HTTP.java
21
HTTP.java
@@ -24,7 +24,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Iterator;
|
import java.util.Locale;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert an HTTP header to a JSONObject and back.
|
* Convert an HTTP header to a JSONObject and back.
|
||||||
@@ -74,7 +74,7 @@ public class HTTP {
|
|||||||
String token;
|
String token;
|
||||||
|
|
||||||
token = x.nextToken();
|
token = x.nextToken();
|
||||||
if (token.toUpperCase().startsWith("HTTP")) {
|
if (token.toUpperCase(Locale.ROOT).startsWith("HTTP")) {
|
||||||
|
|
||||||
// Response
|
// Response
|
||||||
|
|
||||||
@@ -125,8 +125,6 @@ public class HTTP {
|
|||||||
* information.
|
* information.
|
||||||
*/
|
*/
|
||||||
public static String toString(JSONObject jo) throws JSONException {
|
public static String toString(JSONObject jo) throws JSONException {
|
||||||
Iterator<String> keys = jo.keys();
|
|
||||||
String string;
|
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
if (jo.has("Status-Code") && jo.has("Reason-Phrase")) {
|
if (jo.has("Status-Code") && jo.has("Reason-Phrase")) {
|
||||||
sb.append(jo.getString("HTTP-Version"));
|
sb.append(jo.getString("HTTP-Version"));
|
||||||
@@ -146,14 +144,15 @@ public class HTTP {
|
|||||||
throw new JSONException("Not enough material for an HTTP header.");
|
throw new JSONException("Not enough material for an HTTP header.");
|
||||||
}
|
}
|
||||||
sb.append(CRLF);
|
sb.append(CRLF);
|
||||||
while (keys.hasNext()) {
|
// Don't use the new entrySet API to maintain Android support
|
||||||
string = keys.next();
|
for (final String key : jo.keySet()) {
|
||||||
if (!"HTTP-Version".equals(string) && !"Status-Code".equals(string) &&
|
String value = jo.optString(key);
|
||||||
!"Reason-Phrase".equals(string) && !"Method".equals(string) &&
|
if (!"HTTP-Version".equals(key) && !"Status-Code".equals(key) &&
|
||||||
!"Request-URI".equals(string) && !jo.isNull(string)) {
|
!"Reason-Phrase".equals(key) && !"Method".equals(key) &&
|
||||||
sb.append(string);
|
!"Request-URI".equals(key) && !JSONObject.NULL.equals(value)) {
|
||||||
|
sb.append(key);
|
||||||
sb.append(": ");
|
sb.append(": ");
|
||||||
sb.append(jo.getString(string));
|
sb.append(jo.optString(key));
|
||||||
sb.append(CRLF);
|
sb.append(CRLF);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
672
JSONArray.java
672
JSONArray.java
File diff suppressed because it is too large
Load Diff
@@ -1,5 +1,29 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
|
/*
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The JSONException is thrown by the JSON.org classes when things are amiss.
|
* The JSONException is thrown by the JSON.org classes when things are amiss.
|
||||||
*
|
*
|
||||||
|
|||||||
34
JSONML.java
34
JSONML.java
@@ -24,9 +24,6 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Iterator;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This provides static methods to convert an XML text into a JSONArray or
|
* This provides static methods to convert an XML text into a JSONArray or
|
||||||
* JSONObject, and to covert a JSONArray or JSONObject into an XML text using
|
* JSONObject, and to covert a JSONArray or JSONObject into an XML text using
|
||||||
@@ -42,7 +39,7 @@ public class JSONML {
|
|||||||
* @param arrayForm true if array form, false if object form.
|
* @param arrayForm true if array form, false if object form.
|
||||||
* @param ja The JSONArray that is containing the current tag or null
|
* @param ja The JSONArray that is containing the current tag or null
|
||||||
* if we are at the outermost level.
|
* if we are at the outermost level.
|
||||||
* @param keepStrings Don't type-convert text nodes and attibute values
|
* @param keepStrings Don't type-convert text nodes and attribute values
|
||||||
* @return A JSONArray if the value is the outermost tag, otherwise null.
|
* @return A JSONArray if the value is the outermost tag, otherwise null.
|
||||||
* @throws JSONException
|
* @throws JSONException
|
||||||
*/
|
*/
|
||||||
@@ -175,7 +172,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 ? token :JSONObject.stringToValue((String)token));
|
newjo.accumulate(attribute, keepStrings ? ((String)token) :XML.stringToValue((String)token));
|
||||||
token = null;
|
token = null;
|
||||||
} else {
|
} else {
|
||||||
newjo.accumulate(attribute, "");
|
newjo.accumulate(attribute, "");
|
||||||
@@ -226,7 +223,7 @@ public class JSONML {
|
|||||||
} else {
|
} else {
|
||||||
if (ja != null) {
|
if (ja != null) {
|
||||||
ja.put(token instanceof String
|
ja.put(token instanceof String
|
||||||
? keepStrings ? token :JSONObject.stringToValue((String)token)
|
? keepStrings ? XML.unescape((String)token) :XML.stringToValue((String)token)
|
||||||
: token);
|
: token);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -397,13 +394,10 @@ public class JSONML {
|
|||||||
public static String toString(JSONArray ja) throws JSONException {
|
public static String toString(JSONArray ja) throws JSONException {
|
||||||
int i;
|
int i;
|
||||||
JSONObject jo;
|
JSONObject jo;
|
||||||
String key;
|
|
||||||
Iterator<String> keys;
|
|
||||||
int length;
|
int length;
|
||||||
Object object;
|
Object object;
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
String tagName;
|
String tagName;
|
||||||
String value;
|
|
||||||
|
|
||||||
// Emit <tagName
|
// Emit <tagName
|
||||||
|
|
||||||
@@ -420,17 +414,16 @@ public class JSONML {
|
|||||||
|
|
||||||
// Emit the attributes
|
// Emit the attributes
|
||||||
|
|
||||||
keys = jo.keys();
|
// Don't use the new entrySet API to maintain Android support
|
||||||
while (keys.hasNext()) {
|
for (final String key : jo.keySet()) {
|
||||||
key = keys.next();
|
final Object value = jo.opt(key);
|
||||||
XML.noSpace(key);
|
XML.noSpace(key);
|
||||||
value = jo.optString(key);
|
|
||||||
if (value != null) {
|
if (value != null) {
|
||||||
sb.append(' ');
|
sb.append(' ');
|
||||||
sb.append(XML.escape(key));
|
sb.append(XML.escape(key));
|
||||||
sb.append('=');
|
sb.append('=');
|
||||||
sb.append('"');
|
sb.append('"');
|
||||||
sb.append(XML.escape(value));
|
sb.append(XML.escape(value.toString()));
|
||||||
sb.append('"');
|
sb.append('"');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -482,12 +475,10 @@ public class JSONML {
|
|||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
int i;
|
int i;
|
||||||
JSONArray ja;
|
JSONArray ja;
|
||||||
String key;
|
|
||||||
Iterator<String> keys;
|
|
||||||
int length;
|
int length;
|
||||||
Object object;
|
Object object;
|
||||||
String tagName;
|
String tagName;
|
||||||
String value;
|
Object value;
|
||||||
|
|
||||||
//Emit <tagName
|
//Emit <tagName
|
||||||
|
|
||||||
@@ -502,18 +493,17 @@ public class JSONML {
|
|||||||
|
|
||||||
//Emit the attributes
|
//Emit the attributes
|
||||||
|
|
||||||
keys = jo.keys();
|
// Don't use the new entrySet API to maintain Android support
|
||||||
while (keys.hasNext()) {
|
for (final String key : jo.keySet()) {
|
||||||
key = keys.next();
|
|
||||||
if (!"tagName".equals(key) && !"childNodes".equals(key)) {
|
if (!"tagName".equals(key) && !"childNodes".equals(key)) {
|
||||||
XML.noSpace(key);
|
XML.noSpace(key);
|
||||||
value = jo.optString(key);
|
value = jo.opt(key);
|
||||||
if (value != null) {
|
if (value != null) {
|
||||||
sb.append(' ');
|
sb.append(' ');
|
||||||
sb.append(XML.escape(key));
|
sb.append(XML.escape(key));
|
||||||
sb.append('=');
|
sb.append('=');
|
||||||
sb.append('"');
|
sb.append('"');
|
||||||
sb.append(XML.escape(value));
|
sb.append(XML.escape(value.toString()));
|
||||||
sb.append('"');
|
sb.append('"');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
1349
JSONObject.java
1349
JSONObject.java
File diff suppressed because it is too large
Load Diff
@@ -5,7 +5,9 @@ import static java.lang.String.format;
|
|||||||
import java.io.UnsupportedEncodingException;
|
import java.io.UnsupportedEncodingException;
|
||||||
import java.net.URLDecoder;
|
import java.net.URLDecoder;
|
||||||
import java.net.URLEncoder;
|
import java.net.URLEncoder;
|
||||||
import java.util.*;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Copyright (c) 2002 JSON.org
|
||||||
@@ -68,11 +70,11 @@ public class JSONPointer {
|
|||||||
* {@link #append(String)} method calls.
|
* {@link #append(String)} method calls.
|
||||||
*/
|
*/
|
||||||
public JSONPointer build() {
|
public JSONPointer build() {
|
||||||
return new JSONPointer(refTokens);
|
return new JSONPointer(this.refTokens);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds an arbitary token to the list of reference tokens. It can be any non-null value.
|
* Adds an arbitrary token to the list of reference tokens. It can be any non-null value.
|
||||||
*
|
*
|
||||||
* Unlike in the case of JSON string or URI fragment representation of JSON pointers, the
|
* Unlike in the case of JSON string or URI fragment representation of JSON pointers, the
|
||||||
* argument of this method MUST NOT be escaped. If you want to query the property called
|
* argument of this method MUST NOT be escaped. If you want to query the property called
|
||||||
@@ -87,7 +89,7 @@ public class JSONPointer {
|
|||||||
if (token == null) {
|
if (token == null) {
|
||||||
throw new NullPointerException("token cannot be null");
|
throw new NullPointerException("token cannot be null");
|
||||||
}
|
}
|
||||||
refTokens.add(token);
|
this.refTokens.add(token);
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -99,7 +101,7 @@ public class JSONPointer {
|
|||||||
* @return {@code this}
|
* @return {@code this}
|
||||||
*/
|
*/
|
||||||
public Builder append(int arrayIndex) {
|
public Builder append(int arrayIndex) {
|
||||||
refTokens.add(String.valueOf(arrayIndex));
|
this.refTokens.add(String.valueOf(arrayIndex));
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -134,37 +136,57 @@ public class JSONPointer {
|
|||||||
* @param pointer the JSON String or URI Fragment representation of the JSON pointer.
|
* @param pointer the JSON String or URI Fragment representation of the JSON pointer.
|
||||||
* @throws IllegalArgumentException if {@code pointer} is not a valid JSON pointer
|
* @throws IllegalArgumentException if {@code pointer} is not a valid JSON pointer
|
||||||
*/
|
*/
|
||||||
public JSONPointer(String pointer) {
|
public JSONPointer(final String pointer) {
|
||||||
if (pointer == null) {
|
if (pointer == null) {
|
||||||
throw new NullPointerException("pointer cannot be null");
|
throw new NullPointerException("pointer cannot be null");
|
||||||
}
|
}
|
||||||
if (pointer.isEmpty()) {
|
if (pointer.isEmpty() || pointer.equals("#")) {
|
||||||
refTokens = Collections.emptyList();
|
this.refTokens = Collections.emptyList();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
String refs;
|
||||||
if (pointer.startsWith("#/")) {
|
if (pointer.startsWith("#/")) {
|
||||||
pointer = pointer.substring(2);
|
refs = pointer.substring(2);
|
||||||
try {
|
try {
|
||||||
pointer = URLDecoder.decode(pointer, ENCODING);
|
refs = URLDecoder.decode(refs, ENCODING);
|
||||||
} catch (UnsupportedEncodingException e) {
|
} catch (UnsupportedEncodingException e) {
|
||||||
throw new RuntimeException(e);
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
} else if (pointer.startsWith("/")) {
|
} else if (pointer.startsWith("/")) {
|
||||||
pointer = pointer.substring(1);
|
refs = pointer.substring(1);
|
||||||
} else {
|
} else {
|
||||||
throw new IllegalArgumentException("a JSON pointer should start with '/' or '#/'");
|
throw new IllegalArgumentException("a JSON pointer should start with '/' or '#/'");
|
||||||
}
|
}
|
||||||
refTokens = new ArrayList<String>();
|
this.refTokens = new ArrayList<String>();
|
||||||
for (String token : pointer.split("/")) {
|
int slashIdx = -1;
|
||||||
refTokens.add(unescape(token));
|
int prevSlashIdx = 0;
|
||||||
}
|
do {
|
||||||
|
prevSlashIdx = slashIdx + 1;
|
||||||
|
slashIdx = refs.indexOf('/', prevSlashIdx);
|
||||||
|
if(prevSlashIdx == slashIdx || prevSlashIdx == refs.length()) {
|
||||||
|
// found 2 slashes in a row ( obj//next )
|
||||||
|
// or single slash at the end of a string ( obj/test/ )
|
||||||
|
this.refTokens.add("");
|
||||||
|
} else if (slashIdx >= 0) {
|
||||||
|
final String token = refs.substring(prevSlashIdx, slashIdx);
|
||||||
|
this.refTokens.add(unescape(token));
|
||||||
|
} else {
|
||||||
|
// last item after separator, or no separator at all.
|
||||||
|
final String token = refs.substring(prevSlashIdx);
|
||||||
|
this.refTokens.add(unescape(token));
|
||||||
|
}
|
||||||
|
} while (slashIdx >= 0);
|
||||||
|
// using split does not take into account consecutive separators or "ending nulls"
|
||||||
|
//for (String token : refs.split("/")) {
|
||||||
|
// this.refTokens.add(unescape(token));
|
||||||
|
//}
|
||||||
}
|
}
|
||||||
|
|
||||||
public JSONPointer(List<String> refTokens) {
|
public JSONPointer(List<String> refTokens) {
|
||||||
this.refTokens = new ArrayList<String>(refTokens);
|
this.refTokens = new ArrayList<String>(refTokens);
|
||||||
}
|
}
|
||||||
|
|
||||||
private String unescape(String token) {
|
private static String unescape(String token) {
|
||||||
return token.replace("~1", "/").replace("~0", "~")
|
return token.replace("~1", "/").replace("~0", "~")
|
||||||
.replace("\\\"", "\"")
|
.replace("\\\"", "\"")
|
||||||
.replace("\\\\", "\\");
|
.replace("\\\\", "\\");
|
||||||
@@ -180,12 +202,12 @@ public class JSONPointer {
|
|||||||
* @return the result of the evaluation
|
* @return the result of the evaluation
|
||||||
* @throws JSONPointerException if an error occurs during evaluation
|
* @throws JSONPointerException if an error occurs during evaluation
|
||||||
*/
|
*/
|
||||||
public Object queryFrom(Object document) {
|
public Object queryFrom(Object document) throws JSONPointerException {
|
||||||
if (refTokens.isEmpty()) {
|
if (this.refTokens.isEmpty()) {
|
||||||
return document;
|
return document;
|
||||||
}
|
}
|
||||||
Object current = document;
|
Object current = document;
|
||||||
for (String token : refTokens) {
|
for (String token : this.refTokens) {
|
||||||
if (current instanceof JSONObject) {
|
if (current instanceof JSONObject) {
|
||||||
current = ((JSONObject) current).opt(unescape(token));
|
current = ((JSONObject) current).opt(unescape(token));
|
||||||
} else if (current instanceof JSONArray) {
|
} else if (current instanceof JSONArray) {
|
||||||
@@ -204,17 +226,21 @@ public class JSONPointer {
|
|||||||
* @param current the JSONArray to be evaluated
|
* @param current the JSONArray to be evaluated
|
||||||
* @param indexToken the array index in string form
|
* @param indexToken the array index in string form
|
||||||
* @return the matched object. If no matching item is found a
|
* @return the matched object. If no matching item is found a
|
||||||
* JSONPointerException is thrown
|
* @throws JSONPointerException is thrown if the index is out of bounds
|
||||||
*/
|
*/
|
||||||
private Object readByIndexToken(Object current, String indexToken) {
|
private static Object readByIndexToken(Object current, String indexToken) throws JSONPointerException {
|
||||||
try {
|
try {
|
||||||
int index = Integer.parseInt(indexToken);
|
int index = Integer.parseInt(indexToken);
|
||||||
JSONArray currentArr = (JSONArray) current;
|
JSONArray currentArr = (JSONArray) current;
|
||||||
if (index >= currentArr.length()) {
|
if (index >= currentArr.length()) {
|
||||||
throw new JSONPointerException(format("index %d is out of bounds - the array has %d elements", index,
|
throw new JSONPointerException(format("index %s is out of bounds - the array has %d elements", indexToken,
|
||||||
currentArr.length()));
|
Integer.valueOf(currentArr.length())));
|
||||||
}
|
}
|
||||||
return currentArr.get(index);
|
try {
|
||||||
|
return currentArr.get(index);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
throw new JSONPointerException("Error reading value at index position " + index, e);
|
||||||
|
}
|
||||||
} catch (NumberFormatException e) {
|
} catch (NumberFormatException e) {
|
||||||
throw new JSONPointerException(format("%s is not an array index", indexToken), e);
|
throw new JSONPointerException(format("%s is not an array index", indexToken), e);
|
||||||
}
|
}
|
||||||
@@ -227,7 +253,7 @@ public class JSONPointer {
|
|||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
StringBuilder rval = new StringBuilder("");
|
StringBuilder rval = new StringBuilder("");
|
||||||
for (String token: refTokens) {
|
for (String token: this.refTokens) {
|
||||||
rval.append('/').append(escape(token));
|
rval.append('/').append(escape(token));
|
||||||
}
|
}
|
||||||
return rval.toString();
|
return rval.toString();
|
||||||
@@ -241,7 +267,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
|
||||||
*/
|
*/
|
||||||
private String escape(String token) {
|
private static String escape(String token) {
|
||||||
return token.replace("~", "~0")
|
return token.replace("~", "~0")
|
||||||
.replace("/", "~1")
|
.replace("/", "~1")
|
||||||
.replace("\\", "\\\\")
|
.replace("\\", "\\\\")
|
||||||
@@ -255,7 +281,7 @@ public class JSONPointer {
|
|||||||
public String toURIFragment() {
|
public String toURIFragment() {
|
||||||
try {
|
try {
|
||||||
StringBuilder rval = new StringBuilder("#");
|
StringBuilder rval = new StringBuilder("#");
|
||||||
for (String token : refTokens) {
|
for (String token : this.refTokens) {
|
||||||
rval.append('/').append(URLEncoder.encode(token, ENCODING));
|
rval.append('/').append(URLEncoder.encode(token, ENCODING));
|
||||||
}
|
}
|
||||||
return rval.toString();
|
return rval.toString();
|
||||||
|
|||||||
43
JSONPropertyIgnore.java
Normal file
43
JSONPropertyIgnore.java
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
package org.json;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2018 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static java.lang.annotation.ElementType.METHOD;
|
||||||
|
import static java.lang.annotation.RetentionPolicy.RUNTIME;
|
||||||
|
|
||||||
|
import java.lang.annotation.Documented;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
|
@Documented
|
||||||
|
@Retention(RUNTIME)
|
||||||
|
@Target({METHOD})
|
||||||
|
/**
|
||||||
|
* Use this annotation on a getter method to override the Bean name
|
||||||
|
* parser for Bean -> JSONObject mapping. If this annotation is
|
||||||
|
* present at any level in the class hierarchy, then the method will
|
||||||
|
* not be serialized from the bean into the JSONObject.
|
||||||
|
*/
|
||||||
|
public @interface JSONPropertyIgnore { }
|
||||||
47
JSONPropertyName.java
Normal file
47
JSONPropertyName.java
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
package org.json;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2018 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static java.lang.annotation.ElementType.METHOD;
|
||||||
|
import static java.lang.annotation.RetentionPolicy.RUNTIME;
|
||||||
|
|
||||||
|
import java.lang.annotation.Documented;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
|
@Documented
|
||||||
|
@Retention(RUNTIME)
|
||||||
|
@Target({METHOD})
|
||||||
|
/**
|
||||||
|
* 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>
|
||||||
|
* will have the Bean parser fall back to the default field name processing.
|
||||||
|
*/
|
||||||
|
public @interface JSONPropertyName {
|
||||||
|
/**
|
||||||
|
* @return The name of the property as to be used in the JSON Object.
|
||||||
|
*/
|
||||||
|
String value();
|
||||||
|
}
|
||||||
@@ -1,4 +1,29 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
|
/*
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The <code>JSONString</code> interface allows a <code>toJSONString()</code>
|
* The <code>JSONString</code> interface allows a <code>toJSONString()</code>
|
||||||
* method so that a class can change the behavior of
|
* method so that a class can change the behavior of
|
||||||
|
|||||||
@@ -72,6 +72,7 @@ public class JSONStringer extends JSONWriter {
|
|||||||
* <code>endArray</code>).
|
* <code>endArray</code>).
|
||||||
* @return The JSON text.
|
* @return The JSON text.
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return this.mode == 'd' ? this.writer.toString() : null;
|
return this.mode == 'd' ? this.writer.toString() : null;
|
||||||
}
|
}
|
||||||
|
|||||||
178
JSONTokener.java
178
JSONTokener.java
@@ -29,7 +29,7 @@ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
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
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A JSONTokener takes a source string and extracts characters and tokens from
|
* A JSONTokener takes a source string and extracts characters and tokens from
|
||||||
@@ -39,36 +39,45 @@ SOFTWARE.
|
|||||||
* @version 2014-05-03
|
* @version 2014-05-03
|
||||||
*/
|
*/
|
||||||
public class JSONTokener {
|
public class JSONTokener {
|
||||||
|
/** current read character position on the current line. */
|
||||||
private long character;
|
private long character;
|
||||||
|
/** flag to indicate if the end of the input has been found. */
|
||||||
private boolean eof;
|
private boolean eof;
|
||||||
private long index;
|
/** current read index of the input. */
|
||||||
private long line;
|
private long index;
|
||||||
private char previous;
|
/** current line of the input. */
|
||||||
private Reader reader;
|
private long line;
|
||||||
|
/** previous character read from the input. */
|
||||||
|
private char previous;
|
||||||
|
/** Reader for the input. */
|
||||||
|
private final Reader reader;
|
||||||
|
/** flag to indicate that a previous character was requested. */
|
||||||
private boolean usePrevious;
|
private boolean usePrevious;
|
||||||
|
/** the number of characters read in the previous line. */
|
||||||
|
private long characterPreviousLine;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a JSONTokener from a Reader.
|
* Construct a JSONTokener from a Reader. The caller must close the Reader.
|
||||||
*
|
*
|
||||||
* @param reader A reader.
|
* @param reader A reader.
|
||||||
*/
|
*/
|
||||||
public JSONTokener(Reader reader) {
|
public JSONTokener(Reader reader) {
|
||||||
this.reader = reader.markSupported()
|
this.reader = reader.markSupported()
|
||||||
? reader
|
? reader
|
||||||
: new BufferedReader(reader);
|
: new BufferedReader(reader);
|
||||||
this.eof = false;
|
this.eof = false;
|
||||||
this.usePrevious = false;
|
this.usePrevious = false;
|
||||||
this.previous = 0;
|
this.previous = 0;
|
||||||
this.index = 0;
|
this.index = 0;
|
||||||
this.character = 1;
|
this.character = 1;
|
||||||
|
this.characterPreviousLine = 0;
|
||||||
this.line = 1;
|
this.line = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a JSONTokener from an InputStream.
|
* Construct a JSONTokener from an InputStream. The caller must close the input stream.
|
||||||
* @param inputStream The source.
|
* @param inputStream The source.
|
||||||
*/
|
*/
|
||||||
public JSONTokener(InputStream inputStream) {
|
public JSONTokener(InputStream inputStream) {
|
||||||
@@ -97,12 +106,23 @@ public class JSONTokener {
|
|||||||
if (this.usePrevious || this.index <= 0) {
|
if (this.usePrevious || this.index <= 0) {
|
||||||
throw new JSONException("Stepping back two steps is not supported");
|
throw new JSONException("Stepping back two steps is not supported");
|
||||||
}
|
}
|
||||||
this.index -= 1;
|
this.decrementIndexes();
|
||||||
this.character -= 1;
|
|
||||||
this.usePrevious = true;
|
this.usePrevious = true;
|
||||||
this.eof = false;
|
this.eof = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Decrements the indexes for the {@link #back()} method based on the previous character read.
|
||||||
|
*/
|
||||||
|
private void decrementIndexes() {
|
||||||
|
this.index--;
|
||||||
|
if(this.previous=='\r' || this.previous == '\n') {
|
||||||
|
this.line--;
|
||||||
|
this.character=this.characterPreviousLine ;
|
||||||
|
} else if(this.character > 0){
|
||||||
|
this.character--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the hex value of a character (base16).
|
* Get the hex value of a character (base16).
|
||||||
@@ -124,6 +144,8 @@ public class JSONTokener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* Checks if the end of the input has been reached.
|
||||||
|
*
|
||||||
* @return true if at the end of the file and we didn't step back
|
* @return true if at the end of the file and we didn't step back
|
||||||
*/
|
*/
|
||||||
public boolean end() {
|
public boolean end() {
|
||||||
@@ -139,11 +161,24 @@ public class JSONTokener {
|
|||||||
* or backward while checking for more data.
|
* or backward while checking for more data.
|
||||||
*/
|
*/
|
||||||
public boolean more() throws JSONException {
|
public boolean more() throws JSONException {
|
||||||
this.next();
|
if(this.usePrevious) {
|
||||||
if (this.end()) {
|
return true;
|
||||||
return false;
|
}
|
||||||
|
try {
|
||||||
|
this.reader.mark(1);
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new JSONException("Unable to preserve stream position", e);
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
// -1 is EOF, but next() can not consume the null character '\0'
|
||||||
|
if(this.reader.read() <= 0) {
|
||||||
|
this.eof = true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
this.reader.reset();
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new JSONException("Unable to read the next character from the stream", e);
|
||||||
}
|
}
|
||||||
this.back();
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -165,26 +200,39 @@ public class JSONTokener {
|
|||||||
} catch (IOException exception) {
|
} catch (IOException exception) {
|
||||||
throw new JSONException(exception);
|
throw new JSONException(exception);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (c <= 0) { // End of stream
|
|
||||||
this.eof = true;
|
|
||||||
c = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
this.index += 1;
|
if (c <= 0) { // End of stream
|
||||||
if (this.previous == '\r') {
|
this.eof = true;
|
||||||
this.line += 1;
|
return 0;
|
||||||
this.character = c == '\n' ? 0 : 1;
|
|
||||||
} else if (c == '\n') {
|
|
||||||
this.line += 1;
|
|
||||||
this.character = 0;
|
|
||||||
} else {
|
|
||||||
this.character += 1;
|
|
||||||
}
|
}
|
||||||
|
this.incrementIndexes(c);
|
||||||
this.previous = (char) c;
|
this.previous = (char) c;
|
||||||
return this.previous;
|
return this.previous;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Increments the internal indexes according to the previous character
|
||||||
|
* read and the character passed as the current character.
|
||||||
|
* @param c the current character read.
|
||||||
|
*/
|
||||||
|
private void incrementIndexes(int c) {
|
||||||
|
if(c > 0) {
|
||||||
|
this.index++;
|
||||||
|
if(c=='\r') {
|
||||||
|
this.line++;
|
||||||
|
this.characterPreviousLine = this.character;
|
||||||
|
this.character=0;
|
||||||
|
}else if (c=='\n') {
|
||||||
|
if(this.previous != '\r') {
|
||||||
|
this.line++;
|
||||||
|
this.characterPreviousLine = this.character;
|
||||||
|
}
|
||||||
|
this.character=0;
|
||||||
|
} else {
|
||||||
|
this.character++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Consume the next character, and check that it matches a specified
|
* Consume the next character, and check that it matches a specified
|
||||||
@@ -196,8 +244,11 @@ public class JSONTokener {
|
|||||||
public char next(char c) throws JSONException {
|
public char next(char c) throws JSONException {
|
||||||
char n = this.next();
|
char n = this.next();
|
||||||
if (n != c) {
|
if (n != c) {
|
||||||
throw this.syntaxError("Expected '" + c + "' and instead saw '" +
|
if(n > 0) {
|
||||||
n + "'");
|
throw this.syntaxError("Expected '" + c + "' and instead saw '" +
|
||||||
|
n + "'");
|
||||||
|
}
|
||||||
|
throw this.syntaxError("Expected '" + c + "' and instead saw ''");
|
||||||
}
|
}
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
@@ -212,23 +263,23 @@ public class JSONTokener {
|
|||||||
* Substring bounds error if there are not
|
* Substring bounds error if there are not
|
||||||
* n characters remaining in the source string.
|
* n characters remaining in the source string.
|
||||||
*/
|
*/
|
||||||
public String next(int n) throws JSONException {
|
public String next(int n) throws JSONException {
|
||||||
if (n == 0) {
|
if (n == 0) {
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
char[] chars = new char[n];
|
char[] chars = new char[n];
|
||||||
int pos = 0;
|
int pos = 0;
|
||||||
|
|
||||||
while (pos < n) {
|
while (pos < n) {
|
||||||
chars[pos] = this.next();
|
chars[pos] = this.next();
|
||||||
if (this.end()) {
|
if (this.end()) {
|
||||||
throw this.syntaxError("Substring bounds error");
|
throw this.syntaxError("Substring bounds error");
|
||||||
}
|
}
|
||||||
pos += 1;
|
pos += 1;
|
||||||
}
|
}
|
||||||
return new String(chars);
|
return new String(chars);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -372,15 +423,15 @@ public class JSONTokener {
|
|||||||
String string;
|
String string;
|
||||||
|
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case '"':
|
case '"':
|
||||||
case '\'':
|
case '\'':
|
||||||
return this.nextString(c);
|
return this.nextString(c);
|
||||||
case '{':
|
case '{':
|
||||||
this.back();
|
this.back();
|
||||||
return new JSONObject(this);
|
return new JSONObject(this);
|
||||||
case '[':
|
case '[':
|
||||||
this.back();
|
this.back();
|
||||||
return new JSONArray(this);
|
return new JSONArray(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -397,7 +448,9 @@ public class JSONTokener {
|
|||||||
sb.append(c);
|
sb.append(c);
|
||||||
c = this.next();
|
c = this.next();
|
||||||
}
|
}
|
||||||
this.back();
|
if (!this.eof) {
|
||||||
|
this.back();
|
||||||
|
}
|
||||||
|
|
||||||
string = sb.toString().trim();
|
string = sb.toString().trim();
|
||||||
if ("".equals(string)) {
|
if ("".equals(string)) {
|
||||||
@@ -426,13 +479,17 @@ public class JSONTokener {
|
|||||||
do {
|
do {
|
||||||
c = this.next();
|
c = this.next();
|
||||||
if (c == 0) {
|
if (c == 0) {
|
||||||
|
// in some readers, reset() may throw an exception if
|
||||||
|
// the remaining portion of the input is greater than
|
||||||
|
// the mark size (1,000,000 above).
|
||||||
this.reader.reset();
|
this.reader.reset();
|
||||||
this.index = startIndex;
|
this.index = startIndex;
|
||||||
this.character = startCharacter;
|
this.character = startCharacter;
|
||||||
this.line = startLine;
|
this.line = startLine;
|
||||||
return c;
|
return 0;
|
||||||
}
|
}
|
||||||
} while (c != to);
|
} while (c != to);
|
||||||
|
this.reader.mark(1);
|
||||||
} catch (IOException exception) {
|
} catch (IOException exception) {
|
||||||
throw new JSONException(exception);
|
throw new JSONException(exception);
|
||||||
}
|
}
|
||||||
@@ -440,7 +497,6 @@ public class JSONTokener {
|
|||||||
return c;
|
return c;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make a JSONException to signal a syntax error.
|
* Make a JSONException to signal a syntax error.
|
||||||
*
|
*
|
||||||
@@ -470,6 +526,6 @@ public class JSONTokener {
|
|||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return " at " + this.index + " [character " + this.character + " line " +
|
return " at " + this.index + " [character " + this.character + " line " +
|
||||||
this.line + "]";
|
this.line + "]";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
124
JSONWriter.java
124
JSONWriter.java
@@ -1,7 +1,8 @@
|
|||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.Writer;
|
import java.util.Collection;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2006 JSON.org
|
Copyright (c) 2006 JSON.org
|
||||||
@@ -50,11 +51,11 @@ SOFTWARE.
|
|||||||
* <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. JSONWriter adds them for
|
* There are no methods for adding commas or colons. JSONWriter 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
|
||||||
* @version 2015-12-09
|
* @version 2016-08-08
|
||||||
*/
|
*/
|
||||||
public class JSONWriter {
|
public class JSONWriter {
|
||||||
private static final int maxdepth = 200;
|
private static final int maxdepth = 200;
|
||||||
@@ -88,12 +89,12 @@ public class JSONWriter {
|
|||||||
/**
|
/**
|
||||||
* The writer that will receive the output.
|
* The writer that will receive the output.
|
||||||
*/
|
*/
|
||||||
protected Writer writer;
|
protected Appendable writer;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make a fresh JSONWriter. It can be used to build one JSON text.
|
* Make a fresh JSONWriter. It can be used to build one JSON text.
|
||||||
*/
|
*/
|
||||||
public JSONWriter(Writer w) {
|
public JSONWriter(Appendable w) {
|
||||||
this.comma = false;
|
this.comma = false;
|
||||||
this.mode = 'i';
|
this.mode = 'i';
|
||||||
this.stack = new JSONObject[maxdepth];
|
this.stack = new JSONObject[maxdepth];
|
||||||
@@ -114,10 +115,13 @@ public class JSONWriter {
|
|||||||
if (this.mode == 'o' || this.mode == 'a') {
|
if (this.mode == 'o' || this.mode == 'a') {
|
||||||
try {
|
try {
|
||||||
if (this.comma && this.mode == 'a') {
|
if (this.comma && this.mode == 'a') {
|
||||||
this.writer.write(',');
|
this.writer.append(',');
|
||||||
}
|
}
|
||||||
this.writer.write(string);
|
this.writer.append(string);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
|
// Android as of API 25 does not support this exception constructor
|
||||||
|
// however we won't worry about it. If an exception is happening here
|
||||||
|
// it will just throw a "Method not found" exception instead.
|
||||||
throw new JSONException(e);
|
throw new JSONException(e);
|
||||||
}
|
}
|
||||||
if (this.mode == 'o') {
|
if (this.mode == 'o') {
|
||||||
@@ -150,21 +154,24 @@ public class JSONWriter {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* End something.
|
* End something.
|
||||||
* @param mode Mode
|
* @param m Mode
|
||||||
* @param c Closing character
|
* @param c Closing character
|
||||||
* @return this
|
* @return this
|
||||||
* @throws JSONException If unbalanced.
|
* @throws JSONException If unbalanced.
|
||||||
*/
|
*/
|
||||||
private JSONWriter end(char mode, char c) throws JSONException {
|
private JSONWriter end(char m, char c) throws JSONException {
|
||||||
if (this.mode != mode) {
|
if (this.mode != m) {
|
||||||
throw new JSONException(mode == 'a'
|
throw new JSONException(m == 'a'
|
||||||
? "Misplaced endArray."
|
? "Misplaced endArray."
|
||||||
: "Misplaced endObject.");
|
: "Misplaced endObject.");
|
||||||
}
|
}
|
||||||
this.pop(mode);
|
this.pop(m);
|
||||||
try {
|
try {
|
||||||
this.writer.write(c);
|
this.writer.append(c);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
|
// Android as of API 25 does not support this exception constructor
|
||||||
|
// however we won't worry about it. If an exception is happening here
|
||||||
|
// it will just throw a "Method not found" exception instead.
|
||||||
throw new JSONException(e);
|
throw new JSONException(e);
|
||||||
}
|
}
|
||||||
this.comma = true;
|
this.comma = true;
|
||||||
@@ -205,16 +212,24 @@ public class JSONWriter {
|
|||||||
}
|
}
|
||||||
if (this.mode == 'k') {
|
if (this.mode == 'k') {
|
||||||
try {
|
try {
|
||||||
this.stack[this.top - 1].putOnce(string, Boolean.TRUE);
|
JSONObject topObject = this.stack[this.top - 1];
|
||||||
|
// don't use the built in putOnce method to maintain Android support
|
||||||
|
if(topObject.has(string)) {
|
||||||
|
throw new JSONException("Duplicate key \"" + string + "\"");
|
||||||
|
}
|
||||||
|
topObject.put(string, true);
|
||||||
if (this.comma) {
|
if (this.comma) {
|
||||||
this.writer.write(',');
|
this.writer.append(',');
|
||||||
}
|
}
|
||||||
this.writer.write(JSONObject.quote(string));
|
this.writer.append(JSONObject.quote(string));
|
||||||
this.writer.write(':');
|
this.writer.append(':');
|
||||||
this.comma = false;
|
this.comma = false;
|
||||||
this.mode = 'o';
|
this.mode = 'o';
|
||||||
return this;
|
return this;
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
|
// Android as of API 25 does not support this exception constructor
|
||||||
|
// however we won't worry about it. If an exception is happening here
|
||||||
|
// it will just throw a "Method not found" exception instead.
|
||||||
throw new JSONException(e);
|
throw new JSONException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -281,6 +296,77 @@ public class JSONWriter {
|
|||||||
this.top += 1;
|
this.top += 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make a JSON text of an Object value. If the object has an
|
||||||
|
* value.toJSONString() method, then that method will be used to produce the
|
||||||
|
* JSON text. The method is required to produce a strictly conforming text.
|
||||||
|
* If the object does not contain a toJSONString method (which is the most
|
||||||
|
* common case), then a text will be produced by other means. If the value
|
||||||
|
* is an array or Collection, then a JSONArray will be made from it and its
|
||||||
|
* toJSONString method will be called. If the value is a MAP, then a
|
||||||
|
* JSONObject will be made from it and its toJSONString method will be
|
||||||
|
* called. Otherwise, the value's toString method will be called, and the
|
||||||
|
* result will be quoted.
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* Warning: This method assumes that the data structure is acyclical.
|
||||||
|
*
|
||||||
|
* @param value
|
||||||
|
* The value to be serialized.
|
||||||
|
* @return a printable, displayable, transmittable representation of the
|
||||||
|
* object, beginning with <code>{</code> <small>(left
|
||||||
|
* brace)</small> and ending with <code>}</code> <small>(right
|
||||||
|
* brace)</small>.
|
||||||
|
* @throws JSONException
|
||||||
|
* If the value is or contains an invalid number.
|
||||||
|
*/
|
||||||
|
public static String valueToString(Object value) throws JSONException {
|
||||||
|
if (value == null || value.equals(null)) {
|
||||||
|
return "null";
|
||||||
|
}
|
||||||
|
if (value instanceof JSONString) {
|
||||||
|
String object;
|
||||||
|
try {
|
||||||
|
object = ((JSONString) value).toJSONString();
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new JSONException(e);
|
||||||
|
}
|
||||||
|
if (object != null) {
|
||||||
|
return object;
|
||||||
|
}
|
||||||
|
throw new JSONException("Bad value from toJSONString: " + object);
|
||||||
|
}
|
||||||
|
if (value instanceof Number) {
|
||||||
|
// not all Numbers may match actual JSON Numbers. i.e. Fractions or Complex
|
||||||
|
final String numberAsString = JSONObject.numberToString((Number) value);
|
||||||
|
if(JSONObject.NUMBER_PATTERN.matcher(numberAsString).matches()) {
|
||||||
|
// Close enough to a JSON number that we will return it unquoted
|
||||||
|
return numberAsString;
|
||||||
|
}
|
||||||
|
// The Number value is not a valid JSON number.
|
||||||
|
// Instead we will quote it as a string
|
||||||
|
return JSONObject.quote(numberAsString);
|
||||||
|
}
|
||||||
|
if (value instanceof Boolean || value instanceof JSONObject
|
||||||
|
|| value instanceof JSONArray) {
|
||||||
|
return value.toString();
|
||||||
|
}
|
||||||
|
if (value instanceof Map) {
|
||||||
|
Map<?, ?> map = (Map<?, ?>) value;
|
||||||
|
return new JSONObject(map).toString();
|
||||||
|
}
|
||||||
|
if (value instanceof Collection) {
|
||||||
|
Collection<?> coll = (Collection<?>) value;
|
||||||
|
return new JSONArray(coll).toString();
|
||||||
|
}
|
||||||
|
if (value.getClass().isArray()) {
|
||||||
|
return new JSONArray(value).toString();
|
||||||
|
}
|
||||||
|
if(value instanceof Enum<?>){
|
||||||
|
return JSONObject.quote(((Enum<?>)value).name());
|
||||||
|
}
|
||||||
|
return JSONObject.quote(value.toString());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Append either the value <code>true</code> or the value
|
* Append either the value <code>true</code> or the value
|
||||||
@@ -300,7 +386,7 @@ public class JSONWriter {
|
|||||||
* @throws JSONException If the number is not finite.
|
* @throws JSONException If the number is not finite.
|
||||||
*/
|
*/
|
||||||
public JSONWriter value(double d) throws JSONException {
|
public JSONWriter value(double d) throws JSONException {
|
||||||
return this.value(new Double(d));
|
return this.value(Double.valueOf(d));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -322,6 +408,6 @@ public class JSONWriter {
|
|||||||
* @throws JSONException If the value is out of sequence.
|
* @throws JSONException If the value is out of sequence.
|
||||||
*/
|
*/
|
||||||
public JSONWriter value(Object object) throws JSONException {
|
public JSONWriter value(Object object) throws JSONException {
|
||||||
return this.append(JSONObject.valueToString(object));
|
return this.append(valueToString(object));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
2
LICENSE
2
LICENSE
@@ -1,3 +1,5 @@
|
|||||||
|
============================================================================
|
||||||
|
|
||||||
Copyright (c) 2002 JSON.org
|
Copyright (c) 2002 JSON.org
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
|||||||
@@ -25,7 +25,6 @@ SOFTWARE.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Enumeration;
|
import java.util.Enumeration;
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -41,6 +40,8 @@ public class Property {
|
|||||||
* @throws JSONException
|
* @throws JSONException
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(java.util.Properties properties) throws JSONException {
|
public static JSONObject toJSONObject(java.util.Properties properties) throws JSONException {
|
||||||
|
// can't use the new constructor for Android support
|
||||||
|
// JSONObject jo = new JSONObject(properties == null ? 0 : properties.size());
|
||||||
JSONObject jo = new JSONObject();
|
JSONObject jo = new JSONObject();
|
||||||
if (properties != null && !properties.isEmpty()) {
|
if (properties != null && !properties.isEmpty()) {
|
||||||
Enumeration<?> enumProperties = properties.propertyNames();
|
Enumeration<?> enumProperties = properties.propertyNames();
|
||||||
@@ -61,10 +62,12 @@ public class Property {
|
|||||||
public static Properties toProperties(JSONObject jo) throws JSONException {
|
public static Properties toProperties(JSONObject jo) throws JSONException {
|
||||||
Properties properties = new Properties();
|
Properties properties = new Properties();
|
||||||
if (jo != null) {
|
if (jo != null) {
|
||||||
Iterator<String> keys = jo.keys();
|
// Don't use the new entrySet API to maintain Android support
|
||||||
while (keys.hasNext()) {
|
for (final String key : jo.keySet()) {
|
||||||
String name = keys.next();
|
Object value = jo.opt(key);
|
||||||
properties.put(name, jo.getString(name));
|
if (!JSONObject.NULL.equals(value)) {
|
||||||
|
properties.put(key, value.toString());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return properties;
|
return properties;
|
||||||
|
|||||||
107
README
107
README
@@ -1,107 +0,0 @@
|
|||||||
JSON in Java [package org.json]
|
|
||||||
|
|
||||||
JSON is a light-weight, language independent, data interchange format.
|
|
||||||
See http://www.JSON.org/
|
|
||||||
|
|
||||||
The files in this package implement JSON encoders/decoders in Java.
|
|
||||||
It also includes the capability to convert between JSON and XML, HTTP
|
|
||||||
headers, Cookies, and CDL.
|
|
||||||
|
|
||||||
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
|
|
||||||
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.
|
|
||||||
|
|
||||||
The package compiles on Java 1.6-1.8.
|
|
||||||
|
|
||||||
|
|
||||||
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 JSONObject 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.
|
|
||||||
|
|
||||||
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.
|
|
||||||
|
|
||||||
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.
|
|
||||||
|
|
||||||
Unit tests are maintained in a separate project. Contributing developers can test
|
|
||||||
JSON-java pull requests with the code in this project:
|
|
||||||
https://github.com/stleary/JSON-Java-unit-test
|
|
||||||
|
|
||||||
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 7159: The JavaScript Object Notation (JSON) Data Interchange Format
|
|
||||||
(https://tools.ietf.org/html/rfc7159#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 may be
|
|
||||||
removed in some future release.
|
|
||||||
|
|
||||||
In compliance with RFC7159 page 10 section 9, the parser is more lax with what is valid
|
|
||||||
JSON than the Generator. For Example, the tab character (U+0009) is allowed when reading
|
|
||||||
JSON Text strings, but when output by the Generator, 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.
|
|
||||||
|
|
||||||
Release history:
|
|
||||||
|
|
||||||
20160807 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.
|
|
||||||
|
|
||||||
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%7Cgav%7C1%7Cg%3A%22org.json%22%20AND%20a%3A%22json%22
|
|
||||||
150
README.md
Normal file
150
README.md
Normal file
@@ -0,0 +1,150 @@
|
|||||||
|
JSON in Java [package org.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/20190722/json-20190722.jar)**
|
||||||
|
|
||||||
|
JSON is a light-weight, language independent, data interchange format.
|
||||||
|
See http://www.JSON.org/
|
||||||
|
|
||||||
|
The files in this package implement JSON encoders/decoders in Java.
|
||||||
|
It also includes the capability to convert between JSON and XML, HTTP
|
||||||
|
headers, Cookies, and CDL.
|
||||||
|
|
||||||
|
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
|
||||||
|
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.
|
||||||
|
|
||||||
|
The package compiles on Java 1.6-1.8.
|
||||||
|
|
||||||
|
|
||||||
|
**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.
|
||||||
|
|
||||||
|
Unit tests are maintained in a separate project. Contributing developers can test
|
||||||
|
JSON-java pull requests with the code in this project:
|
||||||
|
https://github.com/stleary/JSON-Java-unit-test
|
||||||
|
|
||||||
|
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 may be
|
||||||
|
removed in some future release.
|
||||||
|
|
||||||
|
In compliance with RFC8259 page 10 section 9, the parser is more lax with what is valid
|
||||||
|
JSON than the Generator. For Example, the tab character (U+0009) is allowed when reading
|
||||||
|
JSON Text strings, but when output by the Generator, 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 notible 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
|
||||||
|
|
||||||
|
Release history:
|
||||||
|
|
||||||
|
~~~
|
||||||
|
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.
|
||||||
|
~~~
|
||||||
|
|
||||||
|
|
||||||
|
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
|
||||||
430
XML.java
430
XML.java
@@ -24,14 +24,16 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import java.io.Reader;
|
||||||
|
import java.io.StringReader;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This provides static methods to convert an XML text into a JSONObject, and to
|
* This provides static methods to convert an XML text into a JSONObject, and to
|
||||||
* covert a JSONObject into an XML text.
|
* covert a JSONObject into an XML text.
|
||||||
*
|
*
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
* @version 2016-01-30
|
* @version 2016-08-10
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings("boxing")
|
@SuppressWarnings("boxing")
|
||||||
public class XML {
|
public class XML {
|
||||||
@@ -63,25 +65,70 @@ public class XML {
|
|||||||
/** The Character '/'. */
|
/** The Character '/'. */
|
||||||
public static final Character SLASH = '/';
|
public static final Character SLASH = '/';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Null attribute name
|
||||||
|
*/
|
||||||
|
public static final String NULL_ATTR = "xsi:nil";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an iterator for navigating Code Points in a string instead of
|
||||||
|
* characters. Once Java7 support is dropped, this can be replaced with
|
||||||
|
* <code>
|
||||||
|
* string.codePoints()
|
||||||
|
* </code>
|
||||||
|
* which is available in Java8 and above.
|
||||||
|
*
|
||||||
|
* @see <a href=
|
||||||
|
* "http://stackoverflow.com/a/21791059/6030888">http://stackoverflow.com/a/21791059/6030888</a>
|
||||||
|
*/
|
||||||
|
private static Iterable<Integer> codePointIterator(final String string) {
|
||||||
|
return new Iterable<Integer>() {
|
||||||
|
@Override
|
||||||
|
public Iterator<Integer> iterator() {
|
||||||
|
return new Iterator<Integer>() {
|
||||||
|
private int nextIndex = 0;
|
||||||
|
private int length = string.length();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasNext() {
|
||||||
|
return this.nextIndex < this.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Integer next() {
|
||||||
|
int result = string.codePointAt(this.nextIndex);
|
||||||
|
this.nextIndex += Character.charCount(result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void remove() {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Replace special characters with XML escapes:
|
* Replace special characters with XML escapes:
|
||||||
*
|
*
|
||||||
* <pre>
|
* <pre>
|
||||||
* & <small>(ampersand)</small> is replaced by &amp;
|
* & <small>(ampersand)</small> is replaced by &amp;
|
||||||
* < <small>(less than)</small> is replaced by &lt;
|
* < <small>(less than)</small> is replaced by &lt;
|
||||||
* > <small>(greater than)</small> is replaced by &gt;
|
* > <small>(greater than)</small> is replaced by &gt;
|
||||||
* " <small>(double quote)</small> is replaced by &quot;
|
* " <small>(double quote)</small> is replaced by &quot;
|
||||||
|
* ' <small>(single quote / apostrophe)</small> is replaced by &apos;
|
||||||
* </pre>
|
* </pre>
|
||||||
*
|
*
|
||||||
* @param string
|
* @param string
|
||||||
* The string to be escaped.
|
* The string to be escaped.
|
||||||
* @return The escaped string.
|
* @return The escaped string.
|
||||||
*/
|
*/
|
||||||
public static String escape(String string) {
|
public static String escape(String string) {
|
||||||
StringBuilder sb = new StringBuilder(string.length());
|
StringBuilder sb = new StringBuilder(string.length());
|
||||||
for (int i = 0, length = string.length(); i < length; i++) {
|
for (final int cp : codePointIterator(string)) {
|
||||||
char c = string.charAt(i);
|
switch (cp) {
|
||||||
switch (c) {
|
|
||||||
case '&':
|
case '&':
|
||||||
sb.append("&");
|
sb.append("&");
|
||||||
break;
|
break;
|
||||||
@@ -98,6 +145,69 @@ public class XML {
|
|||||||
sb.append("'");
|
sb.append("'");
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
if (mustEscape(cp)) {
|
||||||
|
sb.append("&#x");
|
||||||
|
sb.append(Integer.toHexString(cp));
|
||||||
|
sb.append(';');
|
||||||
|
} else {
|
||||||
|
sb.appendCodePoint(cp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param cp code point to test
|
||||||
|
* @return true if the code point is not valid for an XML
|
||||||
|
*/
|
||||||
|
private static boolean mustEscape(int cp) {
|
||||||
|
/* Valid range from https://www.w3.org/TR/REC-xml/#charsets
|
||||||
|
*
|
||||||
|
* #x9 | #xA | #xD | [#x20-#xD7FF] | [#xE000-#xFFFD] | [#x10000-#x10FFFF]
|
||||||
|
*
|
||||||
|
* any Unicode character, excluding the surrogate blocks, FFFE, and FFFF.
|
||||||
|
*/
|
||||||
|
// isISOControl is true when (cp >= 0 && cp <= 0x1F) || (cp >= 0x7F && cp <= 0x9F)
|
||||||
|
// all ISO control characters are out of range except tabs and new lines
|
||||||
|
return (Character.isISOControl(cp)
|
||||||
|
&& cp != 0x9
|
||||||
|
&& cp != 0xA
|
||||||
|
&& cp != 0xD
|
||||||
|
) || !(
|
||||||
|
// valid the range of acceptable characters that aren't control
|
||||||
|
(cp >= 0x20 && cp <= 0xD7FF)
|
||||||
|
|| (cp >= 0xE000 && cp <= 0xFFFD)
|
||||||
|
|| (cp >= 0x10000 && cp <= 0x10FFFF)
|
||||||
|
)
|
||||||
|
;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes XML escapes from the string.
|
||||||
|
*
|
||||||
|
* @param string
|
||||||
|
* string to remove escapes from
|
||||||
|
* @return string with converted entities
|
||||||
|
*/
|
||||||
|
public static String unescape(String string) {
|
||||||
|
StringBuilder sb = new StringBuilder(string.length());
|
||||||
|
for (int i = 0, length = string.length(); i < length; i++) {
|
||||||
|
char c = string.charAt(i);
|
||||||
|
if (c == '&') {
|
||||||
|
final int semic = string.indexOf(';', i);
|
||||||
|
if (semic > i) {
|
||||||
|
final String entity = string.substring(i + 1, semic);
|
||||||
|
sb.append(XMLTokener.unescapeEntity(entity));
|
||||||
|
// skip past the entity we just parsed.
|
||||||
|
i += entity.length() + 1;
|
||||||
|
} else {
|
||||||
|
// this shouldn't happen in most cases since the parser
|
||||||
|
// errors on unclosed entries.
|
||||||
|
sb.append(c);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// not part of an entity
|
||||||
sb.append(c);
|
sb.append(c);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -107,7 +217,7 @@ public class XML {
|
|||||||
/**
|
/**
|
||||||
* Throw an exception if the string contains whitespace. Whitespace is not
|
* Throw an exception if the string contains whitespace. Whitespace is not
|
||||||
* allowed in tagNames and attributes.
|
* allowed in tagNames and attributes.
|
||||||
*
|
*
|
||||||
* @param string
|
* @param string
|
||||||
* A string.
|
* A string.
|
||||||
* @throws JSONException Thrown if the string contains whitespace or is empty.
|
* @throws JSONException Thrown if the string contains whitespace or is empty.
|
||||||
@@ -127,7 +237,7 @@ public class XML {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Scan the content following the named tag, attaching it to the context.
|
* Scan the content following the named tag, attaching it to the context.
|
||||||
*
|
*
|
||||||
* @param x
|
* @param x
|
||||||
* The XMLTokener containing the source string.
|
* The XMLTokener containing the source string.
|
||||||
* @param context
|
* @param context
|
||||||
@@ -137,11 +247,11 @@ public class XML {
|
|||||||
* @return true if the close tag is processed.
|
* @return true if the close tag is processed.
|
||||||
* @throws JSONException
|
* @throws JSONException
|
||||||
*/
|
*/
|
||||||
private static boolean parse(XMLTokener x, JSONObject context, String name, boolean keepStrings)
|
private static boolean parse(XMLTokener x, JSONObject context, String name, XMLParserConfiguration config)
|
||||||
throws JSONException {
|
throws JSONException {
|
||||||
char c;
|
char c;
|
||||||
int i;
|
int i;
|
||||||
JSONObject jsonobject = null;
|
JSONObject jsonObject = null;
|
||||||
String string;
|
String string;
|
||||||
String tagName;
|
String tagName;
|
||||||
Object token;
|
Object token;
|
||||||
@@ -174,7 +284,7 @@ public class XML {
|
|||||||
if (x.next() == '[') {
|
if (x.next() == '[') {
|
||||||
string = x.nextCDATA();
|
string = x.nextCDATA();
|
||||||
if (string.length() > 0) {
|
if (string.length() > 0) {
|
||||||
context.accumulate("content", string);
|
context.accumulate(config.cDataTagName, string);
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -222,12 +332,12 @@ public class XML {
|
|||||||
} else {
|
} else {
|
||||||
tagName = (String) token;
|
tagName = (String) token;
|
||||||
token = null;
|
token = null;
|
||||||
jsonobject = new JSONObject();
|
jsonObject = new JSONObject();
|
||||||
|
boolean nilAttributeFound = false;
|
||||||
for (;;) {
|
for (;;) {
|
||||||
if (token == null) {
|
if (token == null) {
|
||||||
token = x.nextToken();
|
token = x.nextToken();
|
||||||
}
|
}
|
||||||
|
|
||||||
// attribute = value
|
// attribute = value
|
||||||
if (token instanceof String) {
|
if (token instanceof String) {
|
||||||
string = (String) token;
|
string = (String) token;
|
||||||
@@ -237,11 +347,20 @@ public class XML {
|
|||||||
if (!(token instanceof String)) {
|
if (!(token instanceof String)) {
|
||||||
throw x.syntaxError("Missing value");
|
throw x.syntaxError("Missing value");
|
||||||
}
|
}
|
||||||
jsonobject.accumulate(string,
|
|
||||||
keepStrings ? token : JSONObject.stringToValue((String) token));
|
if (config.convertNilAttributeToNull
|
||||||
|
&& NULL_ATTR.equals(string)
|
||||||
|
&& Boolean.parseBoolean((String) token)) {
|
||||||
|
nilAttributeFound = true;
|
||||||
|
} else if (!nilAttributeFound) {
|
||||||
|
jsonObject.accumulate(string,
|
||||||
|
config.keepStrings
|
||||||
|
? ((String) token)
|
||||||
|
: stringToValue((String) token));
|
||||||
|
}
|
||||||
token = null;
|
token = null;
|
||||||
} else {
|
} else {
|
||||||
jsonobject.accumulate(string, "");
|
jsonObject.accumulate(string, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -250,8 +369,10 @@ public class XML {
|
|||||||
if (x.nextToken() != GT) {
|
if (x.nextToken() != GT) {
|
||||||
throw x.syntaxError("Misshaped tag");
|
throw x.syntaxError("Misshaped tag");
|
||||||
}
|
}
|
||||||
if (jsonobject.length() > 0) {
|
if (nilAttributeFound) {
|
||||||
context.accumulate(tagName, jsonobject);
|
context.accumulate(tagName, JSONObject.NULL);
|
||||||
|
} else if (jsonObject.length() > 0) {
|
||||||
|
context.accumulate(tagName, jsonObject);
|
||||||
} else {
|
} else {
|
||||||
context.accumulate(tagName, "");
|
context.accumulate(tagName, "");
|
||||||
}
|
}
|
||||||
@@ -269,21 +390,20 @@ public class XML {
|
|||||||
} else if (token instanceof String) {
|
} else if (token instanceof String) {
|
||||||
string = (String) token;
|
string = (String) token;
|
||||||
if (string.length() > 0) {
|
if (string.length() > 0) {
|
||||||
jsonobject.accumulate("content",
|
jsonObject.accumulate(config.cDataTagName,
|
||||||
keepStrings ? token : JSONObject.stringToValue(string));
|
config.keepStrings ? string : stringToValue(string));
|
||||||
}
|
}
|
||||||
|
|
||||||
} else if (token == LT) {
|
} else if (token == LT) {
|
||||||
// Nested element
|
// Nested element
|
||||||
if (parse(x, jsonobject, tagName,keepStrings)) {
|
if (parse(x, jsonObject, tagName, config)) {
|
||||||
if (jsonobject.length() == 0) {
|
if (jsonObject.length() == 0) {
|
||||||
context.accumulate(tagName, "");
|
context.accumulate(tagName, "");
|
||||||
} else if (jsonobject.length() == 1
|
} else if (jsonObject.length() == 1
|
||||||
&& jsonobject.opt("content") != null) {
|
&& jsonObject.opt(config.cDataTagName) != null) {
|
||||||
context.accumulate(tagName,
|
context.accumulate(tagName, jsonObject.opt(config.cDataTagName));
|
||||||
jsonobject.opt("content"));
|
|
||||||
} else {
|
} else {
|
||||||
context.accumulate(tagName, jsonobject);
|
context.accumulate(tagName, jsonObject);
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -295,18 +415,58 @@ public class XML {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method has been deprecated in favor of the
|
* This method is the same as {@link JSONObject#stringToValue(String)}.
|
||||||
* {@link JSONObject.stringToValue(String)} method. Use it instead.
|
*
|
||||||
*
|
|
||||||
* @deprecated Use {@link JSONObject#stringToValue(String)} instead.
|
|
||||||
* @param string String to convert
|
* @param string String to convert
|
||||||
* @return JSON value of this string or the string
|
* @return JSON value of this string or the string
|
||||||
*/
|
*/
|
||||||
@Deprecated
|
// To maintain compatibility with the Android API, this method is a direct copy of
|
||||||
|
// the one in JSONObject. Changes made here should be reflected there.
|
||||||
public static Object stringToValue(String string) {
|
public static Object stringToValue(String string) {
|
||||||
return JSONObject.stringToValue(string);
|
if (string.equals("")) {
|
||||||
|
return string;
|
||||||
|
}
|
||||||
|
if (string.equalsIgnoreCase("true")) {
|
||||||
|
return Boolean.TRUE;
|
||||||
|
}
|
||||||
|
if (string.equalsIgnoreCase("false")) {
|
||||||
|
return Boolean.FALSE;
|
||||||
|
}
|
||||||
|
if (string.equalsIgnoreCase("null")) {
|
||||||
|
return JSONObject.NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If it might be a number, try converting it. If a number cannot be
|
||||||
|
* produced, then the value will just be a string.
|
||||||
|
*/
|
||||||
|
|
||||||
|
char initial = string.charAt(0);
|
||||||
|
if ((initial >= '0' && initial <= '9') || initial == '-') {
|
||||||
|
try {
|
||||||
|
// if we want full Big Number support this block can be replaced with:
|
||||||
|
// return stringToNumber(string);
|
||||||
|
if (string.indexOf('.') > -1 || string.indexOf('e') > -1
|
||||||
|
|| string.indexOf('E') > -1 || "-0".equals(string)) {
|
||||||
|
Double d = Double.valueOf(string);
|
||||||
|
if (!d.isInfinite() && !d.isNaN()) {
|
||||||
|
return d;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Long myLong = Long.valueOf(string);
|
||||||
|
if (string.equals(myLong.toString())) {
|
||||||
|
if (myLong.longValue() == myLong.intValue()) {
|
||||||
|
return Integer.valueOf(myLong.intValue());
|
||||||
|
}
|
||||||
|
return myLong;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception ignore) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return string;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -319,16 +479,92 @@ public class XML {
|
|||||||
* elements are represented as JSONArrays. Content text may be placed in a
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
* "content" member. Comments, prologs, DTDs, and <code><[ [ ]]></code>
|
* "content" member. Comments, prologs, DTDs, and <code><[ [ ]]></code>
|
||||||
* are ignored.
|
* are ignored.
|
||||||
*
|
*
|
||||||
* @param string
|
* @param string
|
||||||
* The source string.
|
* The source string.
|
||||||
* @return A JSONObject containing the structured data from the XML string.
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
* @throws JSONException Thrown if there is an errors while parsing the string
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(String string) throws JSONException {
|
public static JSONObject toJSONObject(String string) throws JSONException {
|
||||||
return toJSONObject(string, false);
|
return toJSONObject(string, XMLParserConfiguration.ORIGINAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML into a
|
||||||
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
|
* JSON is a data format and XML is a document format. XML uses elements,
|
||||||
|
* attributes, and content text, while JSON uses unordered collections of
|
||||||
|
* name/value pairs and arrays of values. JSON does not does not like to
|
||||||
|
* distinguish between elements and attributes. Sequences of similar
|
||||||
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
|
* "content" member. Comments, prologs, DTDs, and <code><[ [ ]]></code>
|
||||||
|
* are ignored.
|
||||||
|
*
|
||||||
|
* @param reader The XML source reader.
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(Reader reader) throws JSONException {
|
||||||
|
return toJSONObject(reader, XMLParserConfiguration.ORIGINAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML into a
|
||||||
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
|
* JSON is a data format and XML is a document format. XML uses elements,
|
||||||
|
* attributes, and content text, while JSON uses unordered collections of
|
||||||
|
* name/value pairs and arrays of values. JSON does not does not like to
|
||||||
|
* distinguish between elements and attributes. Sequences of similar
|
||||||
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
|
* "content" member. Comments, prologs, DTDs, and <code><[ [ ]]></code>
|
||||||
|
* are ignored.
|
||||||
|
*
|
||||||
|
* All values are converted as strings, for 1, 01, 29.0 will not be coerced to
|
||||||
|
* numbers but will instead be the exact value as seen in the XML document.
|
||||||
|
*
|
||||||
|
* @param reader The XML source reader.
|
||||||
|
* @param keepStrings If true, then 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 if there is an errors while parsing the string
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(Reader reader, boolean keepStrings) throws JSONException {
|
||||||
|
if(keepStrings) {
|
||||||
|
return toJSONObject(reader, XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
}
|
||||||
|
return toJSONObject(reader, XMLParserConfiguration.ORIGINAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML into a
|
||||||
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
|
* JSON is a data format and XML is a document format. XML uses elements,
|
||||||
|
* attributes, and content text, while JSON uses unordered collections of
|
||||||
|
* name/value pairs and arrays of values. JSON does not does not like to
|
||||||
|
* distinguish between elements and attributes. Sequences of similar
|
||||||
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
|
* "content" member. Comments, prologs, DTDs, and <code><[ [ ]]></code>
|
||||||
|
* are ignored.
|
||||||
|
*
|
||||||
|
* All values are converted as strings, for 1, 01, 29.0 will not be coerced to
|
||||||
|
* numbers but will instead be the exact value as seen in the XML document.
|
||||||
|
*
|
||||||
|
* @param reader The XML source reader.
|
||||||
|
* @param config Configuration options for the parser
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(Reader reader, XMLParserConfiguration config) throws JSONException {
|
||||||
|
JSONObject jo = new JSONObject();
|
||||||
|
XMLTokener x = new XMLTokener(reader);
|
||||||
|
while (x.more()) {
|
||||||
|
x.skipPast("<");
|
||||||
|
if(x.more()) {
|
||||||
|
parse(x, jo, null, config);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return jo;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
@@ -340,10 +576,10 @@ public class XML {
|
|||||||
* elements are represented as JSONArrays. Content text may be placed in a
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
* "content" member. Comments, prologs, DTDs, and <code><[ [ ]]></code>
|
* "content" member. Comments, prologs, DTDs, and <code><[ [ ]]></code>
|
||||||
* are ignored.
|
* are ignored.
|
||||||
*
|
*
|
||||||
* All values are converted as strings, for 1, 01, 29.0 will not be coerced to
|
* All values are converted as strings, for 1, 01, 29.0 will not be coerced to
|
||||||
* numbers but will instead be the exact value as seen in the XML document.
|
* numbers but will instead be the exact value as seen in the XML document.
|
||||||
*
|
*
|
||||||
* @param string
|
* @param string
|
||||||
* The source string.
|
* The source string.
|
||||||
* @param keepStrings If true, then values will not be coerced into boolean
|
* @param keepStrings If true, then values will not be coerced into boolean
|
||||||
@@ -352,28 +588,48 @@ public class XML {
|
|||||||
* @throws JSONException Thrown if there is an errors while parsing the string
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(String string, boolean keepStrings) throws JSONException {
|
public static JSONObject toJSONObject(String string, boolean keepStrings) throws JSONException {
|
||||||
JSONObject jo = new JSONObject();
|
return toJSONObject(new StringReader(string), keepStrings);
|
||||||
XMLTokener x = new XMLTokener(string);
|
|
||||||
while (x.more() && x.skipPast("<")) {
|
|
||||||
parse(x, jo, null, keepStrings);
|
|
||||||
}
|
|
||||||
return jo;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
|
* JSON is a data format and XML is a document format. XML uses elements,
|
||||||
|
* attributes, and content text, while JSON uses unordered collections of
|
||||||
|
* name/value pairs and arrays of values. JSON does not does not like to
|
||||||
|
* distinguish between elements and attributes. Sequences of similar
|
||||||
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
|
* "content" member. Comments, prologs, DTDs, and <code><[ [ ]]></code>
|
||||||
|
* are ignored.
|
||||||
|
*
|
||||||
|
* All values are converted as strings, for 1, 01, 29.0 will not be coerced to
|
||||||
|
* numbers but will instead be the exact value as seen in the XML document.
|
||||||
|
*
|
||||||
|
* @param string
|
||||||
|
* The source string.
|
||||||
|
* @param config Configuration options for the parser.
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(String string, XMLParserConfiguration config) throws JSONException {
|
||||||
|
return toJSONObject(new StringReader(string), config);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a JSONObject into a well-formed, element-normal XML string.
|
* Convert a JSONObject into a well-formed, element-normal XML string.
|
||||||
*
|
*
|
||||||
* @param object
|
* @param object
|
||||||
* A JSONObject.
|
* A JSONObject.
|
||||||
* @return A string.
|
* @return A string.
|
||||||
* @throws JSONException Thrown if there is an error parsing the string
|
* @throws JSONException Thrown if there is an error parsing the string
|
||||||
*/
|
*/
|
||||||
public static String toString(Object object) throws JSONException {
|
public static String toString(Object object) throws JSONException {
|
||||||
return toString(object, null);
|
return toString(object, null, XMLParserConfiguration.ORIGINAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a JSONObject into a well-formed, element-normal XML string.
|
* Convert a JSONObject into a well-formed, element-normal XML string.
|
||||||
*
|
*
|
||||||
* @param object
|
* @param object
|
||||||
* A JSONObject.
|
* A JSONObject.
|
||||||
* @param tagName
|
* @param tagName
|
||||||
@@ -381,15 +637,28 @@ public class XML {
|
|||||||
* @return A string.
|
* @return A string.
|
||||||
* @throws JSONException Thrown if there is an error parsing the string
|
* @throws JSONException Thrown if there is an error parsing the string
|
||||||
*/
|
*/
|
||||||
public static String toString(Object object, String tagName)
|
public static String toString(final Object object, final String tagName) {
|
||||||
|
return toString(object, tagName, XMLParserConfiguration.ORIGINAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a JSONObject into a well-formed, 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.
|
||||||
|
* @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)
|
||||||
throws JSONException {
|
throws JSONException {
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
JSONArray ja;
|
JSONArray ja;
|
||||||
JSONObject jo;
|
JSONObject jo;
|
||||||
String key;
|
|
||||||
Iterator<String> keys;
|
|
||||||
String string;
|
String string;
|
||||||
Object value;
|
|
||||||
|
|
||||||
if (object instanceof JSONObject) {
|
if (object instanceof JSONObject) {
|
||||||
|
|
||||||
@@ -401,29 +670,28 @@ public class XML {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Loop thru the keys.
|
// Loop thru the keys.
|
||||||
|
// don't use the new entrySet accessor to maintain Android Support
|
||||||
jo = (JSONObject) object;
|
jo = (JSONObject) object;
|
||||||
keys = jo.keys();
|
for (final String key : jo.keySet()) {
|
||||||
while (keys.hasNext()) {
|
Object value = jo.opt(key);
|
||||||
key = keys.next();
|
|
||||||
value = jo.opt(key);
|
|
||||||
if (value == null) {
|
if (value == null) {
|
||||||
value = "";
|
value = "";
|
||||||
} else if (value.getClass().isArray()) {
|
} else if (value.getClass().isArray()) {
|
||||||
value = new JSONArray(value);
|
value = new JSONArray(value);
|
||||||
}
|
}
|
||||||
string = value instanceof String ? (String) value : null;
|
|
||||||
|
|
||||||
// Emit content in body
|
// Emit content in body
|
||||||
if ("content".equals(key)) {
|
if (key.equals(config.cDataTagName)) {
|
||||||
if (value instanceof JSONArray) {
|
if (value instanceof JSONArray) {
|
||||||
ja = (JSONArray) value;
|
ja = (JSONArray) value;
|
||||||
int i = 0;
|
int jaLength = ja.length();
|
||||||
for (Object val : ja) {
|
// don't use the new iterator API to maintain support for Android
|
||||||
|
for (int i = 0; i < jaLength; i++) {
|
||||||
if (i > 0) {
|
if (i > 0) {
|
||||||
sb.append('\n');
|
sb.append('\n');
|
||||||
}
|
}
|
||||||
|
Object val = ja.opt(i);
|
||||||
sb.append(escape(val.toString()));
|
sb.append(escape(val.toString()));
|
||||||
i++;
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
sb.append(escape(value.toString()));
|
sb.append(escape(value.toString()));
|
||||||
@@ -433,17 +701,20 @@ public class XML {
|
|||||||
|
|
||||||
} else if (value instanceof JSONArray) {
|
} else if (value instanceof JSONArray) {
|
||||||
ja = (JSONArray) value;
|
ja = (JSONArray) value;
|
||||||
for (Object val : ja) {
|
int jaLength = ja.length();
|
||||||
|
// don't use the new iterator API to maintain support for Android
|
||||||
|
for (int i = 0; i < jaLength; i++) {
|
||||||
|
Object val = ja.opt(i);
|
||||||
if (val instanceof JSONArray) {
|
if (val instanceof JSONArray) {
|
||||||
sb.append('<');
|
sb.append('<');
|
||||||
sb.append(key);
|
sb.append(key);
|
||||||
sb.append('>');
|
sb.append('>');
|
||||||
sb.append(toString(val));
|
sb.append(toString(val, null, config));
|
||||||
sb.append("</");
|
sb.append("</");
|
||||||
sb.append(key);
|
sb.append(key);
|
||||||
sb.append('>');
|
sb.append('>');
|
||||||
} else {
|
} else {
|
||||||
sb.append(toString(val, key));
|
sb.append(toString(val, key, config));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if ("".equals(value)) {
|
} else if ("".equals(value)) {
|
||||||
@@ -454,12 +725,12 @@ public class XML {
|
|||||||
// Emit a new tag <k>
|
// Emit a new tag <k>
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
sb.append(toString(value, key));
|
sb.append(toString(value, key, config));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (tagName != null) {
|
if (tagName != null) {
|
||||||
|
|
||||||
// Emit the </tagname> close tag
|
// Emit the </tagName> close tag
|
||||||
sb.append("</");
|
sb.append("</");
|
||||||
sb.append(tagName);
|
sb.append(tagName);
|
||||||
sb.append('>');
|
sb.append('>');
|
||||||
@@ -468,21 +739,22 @@ public class XML {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (object != null) {
|
if (object != null && (object instanceof JSONArray || object.getClass().isArray())) {
|
||||||
if (object.getClass().isArray()) {
|
if(object.getClass().isArray()) {
|
||||||
object = new JSONArray(object);
|
ja = new JSONArray(object);
|
||||||
}
|
} else {
|
||||||
|
|
||||||
if (object instanceof JSONArray) {
|
|
||||||
ja = (JSONArray) object;
|
ja = (JSONArray) object;
|
||||||
for (Object val : ja) {
|
|
||||||
// XML does not have good support for arrays. If an array
|
|
||||||
// appears in a place where XML is lacking, synthesize an
|
|
||||||
// <array> element.
|
|
||||||
sb.append(toString(val, tagName == null ? "array" : tagName));
|
|
||||||
}
|
|
||||||
return sb.toString();
|
|
||||||
}
|
}
|
||||||
|
int jaLength = ja.length();
|
||||||
|
// don't use the new iterator API to maintain support for Android
|
||||||
|
for (int i = 0; i < jaLength; i++) {
|
||||||
|
Object val = ja.opt(i);
|
||||||
|
// XML does not have good support for arrays. If an array
|
||||||
|
// appears in a place where XML is lacking, synthesize an
|
||||||
|
// <array> element.
|
||||||
|
sb.append(toString(val, tagName == null ? "array" : tagName, config));
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
string = (object == null) ? "null" : escape(object.toString());
|
string = (object == null) ? "null" : escape(object.toString());
|
||||||
|
|||||||
107
XMLParserConfiguration.java
Normal file
107
XMLParserConfiguration.java
Normal file
@@ -0,0 +1,107 @@
|
|||||||
|
package org.json;
|
||||||
|
/*
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configuration object for the XML parser.
|
||||||
|
* @author AylwardJ
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class XMLParserConfiguration {
|
||||||
|
/** Original Configuration of the XML Parser. */
|
||||||
|
public static final XMLParserConfiguration ORIGINAL = new XMLParserConfiguration();
|
||||||
|
/** Original configuration of the XML Parser except that values are kept as strings. */
|
||||||
|
public static final XMLParserConfiguration KEEP_STRINGS = new XMLParserConfiguration(true);
|
||||||
|
/**
|
||||||
|
* When parsing the XML into JSON, specifies if values should be kept as strings (true), or if
|
||||||
|
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||||
|
*/
|
||||||
|
public final boolean keepStrings;
|
||||||
|
/**
|
||||||
|
* 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
|
||||||
|
* processing.
|
||||||
|
*/
|
||||||
|
public final String cDataTagName;
|
||||||
|
/**
|
||||||
|
* When parsing the XML into JSON, specifies if values with attribute xsi:nil="true"
|
||||||
|
* should be kept as attribute(false), or they should be converted to null(true)
|
||||||
|
*/
|
||||||
|
public final boolean convertNilAttributeToNull;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Default parser configuration. Does not keep strings, and the CDATA Tag Name is "content".
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration () {
|
||||||
|
this(false, "content", false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration (final boolean keepStrings) {
|
||||||
|
this(keepStrings, "content", false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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
|
||||||
|
* disable CDATA processing
|
||||||
|
* @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.
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration (final String cDataTagName) {
|
||||||
|
this(false, cDataTagName, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configure the parser to use custom settings.
|
||||||
|
* @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 cDataTagName<code>null</code> to disable CDATA processing. Any other value
|
||||||
|
* to use that value as the JSONObject key name to process as CDATA.
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration (final boolean keepStrings, final String cDataTagName) {
|
||||||
|
this.keepStrings = keepStrings;
|
||||||
|
this.cDataTagName = cDataTagName;
|
||||||
|
this.convertNilAttributeToNull = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configure the parser to use custom settings.
|
||||||
|
* @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 cDataTagName <code>null</code> to disable CDATA processing. Any other value
|
||||||
|
* to use that value as the JSONObject key name to process as CDATA.
|
||||||
|
* @param convertNilAttributeToNull <code>true</code> to parse values with attribute xsi:nil="true" as null.
|
||||||
|
* <code>false</code> to parse values with attribute xsi:nil="true" as {"xsi:nil":true}.
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration (final boolean keepStrings, final String cDataTagName, final boolean convertNilAttributeToNull) {
|
||||||
|
this.keepStrings = keepStrings;
|
||||||
|
this.cDataTagName = cDataTagName;
|
||||||
|
this.convertNilAttributeToNull = convertNilAttributeToNull;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -24,6 +24,8 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import java.io.Reader;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The XMLTokener extends the JSONTokener to provide additional methods
|
* The XMLTokener extends the JSONTokener to provide additional methods
|
||||||
* for the parsing of XML texts.
|
* for the parsing of XML texts.
|
||||||
@@ -47,6 +49,14 @@ public class XMLTokener extends JSONTokener {
|
|||||||
entity.put("quot", XML.QUOT);
|
entity.put("quot", XML.QUOT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct an XMLTokener from a Reader.
|
||||||
|
* @param r A source reader.
|
||||||
|
*/
|
||||||
|
public XMLTokener(Reader r) {
|
||||||
|
super(r);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct an XMLTokener from a string.
|
* Construct an XMLTokener from a string.
|
||||||
* @param s A source string.
|
* @param s A source string.
|
||||||
@@ -64,11 +74,8 @@ public class XMLTokener extends JSONTokener {
|
|||||||
char c;
|
char c;
|
||||||
int i;
|
int i;
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
for (;;) {
|
while (more()) {
|
||||||
c = next();
|
c = next();
|
||||||
if (end()) {
|
|
||||||
throw syntaxError("Unclosed CDATA");
|
|
||||||
}
|
|
||||||
sb.append(c);
|
sb.append(c);
|
||||||
i = sb.length() - 3;
|
i = sb.length() - 3;
|
||||||
if (i >= 0 && sb.charAt(i) == ']' &&
|
if (i >= 0 && sb.charAt(i) == ']' &&
|
||||||
@@ -77,6 +84,7 @@ public class XMLTokener extends JSONTokener {
|
|||||||
return sb.toString();
|
return sb.toString();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
throw syntaxError("Unclosed CDATA");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -103,7 +111,10 @@ public class XMLTokener extends JSONTokener {
|
|||||||
}
|
}
|
||||||
sb = new StringBuilder();
|
sb = new StringBuilder();
|
||||||
for (;;) {
|
for (;;) {
|
||||||
if (c == '<' || c == 0) {
|
if (c == 0) {
|
||||||
|
return sb.toString().trim();
|
||||||
|
}
|
||||||
|
if (c == '<') {
|
||||||
back();
|
back();
|
||||||
return sb.toString().trim();
|
return sb.toString().trim();
|
||||||
}
|
}
|
||||||
@@ -124,7 +135,7 @@ public class XMLTokener extends JSONTokener {
|
|||||||
* @return A Character or an entity String if the entity is not recognized.
|
* @return A Character or an entity String if the entity is not recognized.
|
||||||
* @throws JSONException If missing ';' in XML entity.
|
* @throws JSONException If missing ';' in XML entity.
|
||||||
*/
|
*/
|
||||||
public Object nextEntity(char ampersand) throws JSONException {
|
public Object nextEntity(@SuppressWarnings("unused") char ampersand) throws JSONException {
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
for (;;) {
|
for (;;) {
|
||||||
char c = next();
|
char c = next();
|
||||||
@@ -137,8 +148,37 @@ public class XMLTokener extends JSONTokener {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
String string = sb.toString();
|
String string = sb.toString();
|
||||||
Object object = entity.get(string);
|
return unescapeEntity(string);
|
||||||
return object != null ? object : ampersand + string + ";";
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unescape an XML entity encoding;
|
||||||
|
* @param e entity (only the actual entity value, not the preceding & or ending ;
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
static String unescapeEntity(String e) {
|
||||||
|
// validate
|
||||||
|
if (e == null || e.isEmpty()) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
// if our entity is an encoded unicode point, parse it.
|
||||||
|
if (e.charAt(0) == '#') {
|
||||||
|
int cp;
|
||||||
|
if (e.charAt(1) == 'x') {
|
||||||
|
// hex encoded unicode
|
||||||
|
cp = Integer.parseInt(e.substring(2), 16);
|
||||||
|
} else {
|
||||||
|
// decimal encoded unicode
|
||||||
|
cp = Integer.parseInt(e.substring(1));
|
||||||
|
}
|
||||||
|
return new String(new int[] {cp},0,1);
|
||||||
|
}
|
||||||
|
Character knownEntity = entity.get(e);
|
||||||
|
if(knownEntity==null) {
|
||||||
|
// we don't know the entity so keep it encoded
|
||||||
|
return '&' + e + ';';
|
||||||
|
}
|
||||||
|
return knownEntity.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -192,6 +232,7 @@ public class XMLTokener extends JSONTokener {
|
|||||||
}
|
}
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case 0:
|
case 0:
|
||||||
|
throw syntaxError("Unterminated string");
|
||||||
case '<':
|
case '<':
|
||||||
case '>':
|
case '>':
|
||||||
case '/':
|
case '/':
|
||||||
@@ -296,9 +337,11 @@ public class XMLTokener extends JSONTokener {
|
|||||||
* Skip characters until past the requested string.
|
* Skip characters until past the requested string.
|
||||||
* If it is not found, we are left at the end of the source with a result of false.
|
* If it is not found, we are left at the end of the source with a result of false.
|
||||||
* @param to A string to skip past.
|
* @param to A string to skip past.
|
||||||
* @throws JSONException
|
|
||||||
*/
|
*/
|
||||||
public boolean skipPast(String to) throws JSONException {
|
// The Android implementation of JSONTokener has a public method of public void skipPast(String to)
|
||||||
|
// even though ours does not have that method, to have API compatibility, our method in the subclass
|
||||||
|
// should match.
|
||||||
|
public void skipPast(String to) {
|
||||||
boolean b;
|
boolean b;
|
||||||
char c;
|
char c;
|
||||||
int i;
|
int i;
|
||||||
@@ -315,7 +358,7 @@ public class XMLTokener extends JSONTokener {
|
|||||||
for (i = 0; i < length; i += 1) {
|
for (i = 0; i < length; i += 1) {
|
||||||
c = next();
|
c = next();
|
||||||
if (c == 0) {
|
if (c == 0) {
|
||||||
return false;
|
return;
|
||||||
}
|
}
|
||||||
circle[i] = c;
|
circle[i] = c;
|
||||||
}
|
}
|
||||||
@@ -342,14 +385,14 @@ public class XMLTokener extends JSONTokener {
|
|||||||
/* If we exit the loop with b intact, then victory is ours. */
|
/* If we exit the loop with b intact, then victory is ours. */
|
||||||
|
|
||||||
if (b) {
|
if (b) {
|
||||||
return true;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the next character. If there isn't one, then defeat is ours. */
|
/* Get the next character. If there isn't one, then defeat is ours. */
|
||||||
|
|
||||||
c = next();
|
c = next();
|
||||||
if (c == 0) {
|
if (c == 0) {
|
||||||
return false;
|
return;
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
* Shove the character in the circle buffer and advance the
|
* Shove the character in the circle buffer and advance the
|
||||||
|
|||||||
Reference in New Issue
Block a user