mirror of
https://github.com/stleary/JSON-java.git
synced 2026-01-24 00:03:17 -05:00
Compare commits
60 Commits
89f6e7f6a6
...
pre-releas
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
c048b36516 | ||
|
|
f6e5bfa2db | ||
|
|
084b24cbe7 | ||
|
|
22ccf1b8e5 | ||
|
|
133c0cc75f | ||
|
|
0578285036 | ||
|
|
48fb5261fe | ||
|
|
8353b9c3f0 | ||
|
|
fe22b242b5 | ||
|
|
7eca507d13 | ||
|
|
e1eabc9c27 | ||
|
|
0d436d92e2 | ||
|
|
45bcba518f | ||
|
|
47fb49b6a8 | ||
|
|
0df034c9fd | ||
|
|
f0a05e6911 | ||
|
|
24093491a8 | ||
|
|
1275f6809d | ||
|
|
9234eab00a | ||
|
|
72f4c3e646 | ||
|
|
df2d6f8363 | ||
|
|
a6e412bded | ||
|
|
2391d248cc | ||
|
|
401495ae86 | ||
|
|
448e204186 | ||
|
|
eb56704e68 | ||
|
|
651511f500 | ||
|
|
a14cb12c85 | ||
|
|
f566a1d9ee | ||
|
|
5920eca2d7 | ||
|
|
3b097d051a | ||
|
|
4e630e58a4 | ||
|
|
b732188e4e | ||
|
|
5369442671 | ||
|
|
bb1138762a | ||
|
|
6a732ec99d | ||
|
|
c798c76ddd | ||
|
|
23d5e52a53 | ||
|
|
98df35449a | ||
|
|
1be6ee31a7 | ||
|
|
a2d3d3c9b5 | ||
|
|
bf9219386a | ||
|
|
85495facbd | ||
|
|
7aba3ac941 | ||
|
|
d51250f6b0 | ||
|
|
9cb8e153bf | ||
|
|
80c1479ad8 | ||
|
|
444335d12a | ||
|
|
a2c0562e04 | ||
|
|
153972afdf | ||
|
|
4a8ff28fd8 | ||
|
|
fa457a4113 | ||
|
|
b7f708b222 | ||
|
|
12411b7981 | ||
|
|
61801c623e | ||
|
|
1915aab7c4 | ||
|
|
8439039da7 | ||
|
|
a6bdd081eb | ||
|
|
a30d71fdca | ||
|
|
cdd67b0aef |
572
Examples.md
572
Examples.md
@@ -1,7 +1,7 @@
|
||||
<h1>Examples</h1>
|
||||
<p>Imports used in the examples: </p>
|
||||
|
||||
```
|
||||
```java
|
||||
import java.util.HashMap;
|
||||
import java.util.Map;
|
||||
import java.util.Properties;
|
||||
@@ -14,208 +14,208 @@ import java.util.Properties;
|
||||
|
||||
<h3>Using JSONArray</h3>
|
||||
|
||||
```
|
||||
private static void JSONExampleArray1() {
|
||||
//We create a JSONObject from a String containing an array using JSONArray
|
||||
//Firstly, we declare an Array in a String
|
||||
```java
|
||||
private static void JSONExampleArray1() {
|
||||
//We create a JSONObject from a String containing an array using JSONArray
|
||||
//Firstly, we declare an Array in a String
|
||||
|
||||
String arrayStr =
|
||||
"["+"true,"+"false,"+ "\"true\","+ "\"false\","+"\"hello\","+"23.45e-4,"+
|
||||
"\"23.45\","+"42,"+"\"43\","+"["+"\"world\""+"],"+
|
||||
"{"+
|
||||
"\"key1\":\"value1\","+
|
||||
"\"key2\":\"value2\","+
|
||||
"\"key3\":\"value3\","+
|
||||
"\"key4\":\"value4\""+
|
||||
"},"+
|
||||
"0,"+"\"-1\""+
|
||||
"]";
|
||||
String arrayStr =
|
||||
"["+"true,"+"false,"+ "\"true\","+ "\"false\","+"\"hello\","+"23.45e-4,"+
|
||||
"\"23.45\","+"42,"+"\"43\","+"["+"\"world\""+"],"+
|
||||
"{"+
|
||||
"\"key1\":\"value1\","+
|
||||
"\"key2\":\"value2\","+
|
||||
"\"key3\":\"value3\","+
|
||||
"\"key4\":\"value4\""+
|
||||
"},"+
|
||||
"0,"+"\"-1\""+
|
||||
"]";
|
||||
|
||||
//Then, we initializate the JSONArray thanks to its constructor
|
||||
//Then, we initializate the JSONArray thanks to its constructor
|
||||
|
||||
JSONArray array = new JSONArray(arrayStr);
|
||||
System.out.println("Values array: "+ array);
|
||||
JSONArray array = new JSONArray(arrayStr);
|
||||
System.out.println("Values array: "+ array);
|
||||
|
||||
//We convert that array into a JSONObject, but first, we need the labels, so we need another JSONArray with the labels.
|
||||
//Here we will use an auxiliary function to get one for the example.
|
||||
//We convert that array into a JSONObject, but first, we need the labels, so we need another JSONArray with the labels.
|
||||
//Here we will use an auxiliary function to get one for the example.
|
||||
|
||||
JSONArray list = listNumberArray(array.length());
|
||||
System.out.println("Label Array: "+ list.toString());
|
||||
//Now, we construct the JSONObject using both the value array and the label array.
|
||||
JSONObject object = array.toJSONObject(list);
|
||||
System.out.println("Final JSONOBject: " + object);
|
||||
}
|
||||
|
||||
//This method creates an JSONArray of labels in which those are generated by their positions
|
||||
|
||||
private static JSONArray listNumberArray(int max){
|
||||
JSONArray res = new JSONArray();
|
||||
for (int i=0; i<max;i++) {
|
||||
//The value of the labels must be an String in order to make it work
|
||||
res.put(String.valueOf(i));
|
||||
}
|
||||
return res;
|
||||
JSONArray list = listNumberArray(array.length());
|
||||
System.out.println("Label Array: "+ list.toString());
|
||||
//Now, we construct the JSONObject using both the value array and the label array.
|
||||
JSONObject object = array.toJSONObject(list);
|
||||
System.out.println("Final JSONOBject: " + object);
|
||||
}
|
||||
|
||||
//This method creates an JSONArray of labels in which those are generated by their positions
|
||||
|
||||
private static JSONArray listNumberArray(int max){
|
||||
JSONArray res = new JSONArray();
|
||||
for (int i=0; i<max;i++) {
|
||||
//The value of the labels must be an String in order to make it work
|
||||
res.put(String.valueOf(i));
|
||||
}
|
||||
return res;
|
||||
}
|
||||
```
|
||||
```
|
||||
private static void JSONExampleArray2() {
|
||||
```java
|
||||
private static void JSONExampleArray2() {
|
||||
|
||||
//We can also create an Array without a String by creating an empty array and adding elements to it
|
||||
//We can also create an Array without a String by creating an empty array and adding elements to it
|
||||
|
||||
JSONArray array = new JSONArray();
|
||||
JSONArray array = new JSONArray();
|
||||
|
||||
//Adding elements with .put()
|
||||
//Adding elements with .put()
|
||||
|
||||
array.put("value");
|
||||
array.put(5);
|
||||
array.put(-23.45e67);
|
||||
array.put(true);
|
||||
|
||||
//We convert it to JSONObject providing a label arrray like last time
|
||||
array.put("value");
|
||||
array.put(5);
|
||||
array.put(-23.45e67);
|
||||
array.put(true);
|
||||
|
||||
JSONArray list = listNumberArray(array.length());
|
||||
JSONObject object = array.toJSONObject(list);
|
||||
System.out.println("Final JSONOBject: " + object);
|
||||
}
|
||||
//We convert it to JSONObject providing a label arrray like last time
|
||||
|
||||
JSONArray list = listNumberArray(array.length());
|
||||
JSONObject object = array.toJSONObject(list);
|
||||
System.out.println("Final JSONOBject: " + object);
|
||||
}
|
||||
```
|
||||
|
||||
<h3>Using JSONStringer</h3>
|
||||
|
||||
```
|
||||
private static void JSONExampleStringer() {
|
||||
```java
|
||||
private static void JSONExampleStringer() {
|
||||
|
||||
//We initializate the JSONStringer
|
||||
//We initializate the JSONStringer
|
||||
|
||||
JSONStringer jsonStringer = new JSONStringer();
|
||||
JSONStringer jsonStringer = new JSONStringer();
|
||||
|
||||
//Now we start the process of adding elements with .object()
|
||||
//Now we start the process of adding elements with .object()
|
||||
|
||||
jsonStringer.object();
|
||||
jsonStringer.object();
|
||||
|
||||
//We can now add elements as keys and values with .values () and .key()
|
||||
//We can now add elements as keys and values with .values () and .key()
|
||||
|
||||
jsonStringer.key("trueValue").value(true);
|
||||
jsonStringer.key("falseValue").value(false);
|
||||
jsonStringer.key("nullValue").value(null);
|
||||
jsonStringer.key("stringValue").value("hello world!");
|
||||
jsonStringer.key("complexStringValue").value("h\be\tllo w\u1234orld!");
|
||||
jsonStringer.key("intValue").value(42);
|
||||
jsonStringer.key("doubleValue").value(-23.45e67);
|
||||
jsonStringer.key("trueValue").value(true);
|
||||
jsonStringer.key("falseValue").value(false);
|
||||
jsonStringer.key("nullValue").value(null);
|
||||
jsonStringer.key("stringValue").value("hello world!");
|
||||
jsonStringer.key("complexStringValue").value("h\be\tllo w\u1234orld!");
|
||||
jsonStringer.key("intValue").value(42);
|
||||
jsonStringer.key("doubleValue").value(-23.45e67);
|
||||
|
||||
//We end this prcedure with .ednObject
|
||||
//We end this prcedure with .ednObject
|
||||
|
||||
jsonStringer.endObject();
|
||||
jsonStringer.endObject();
|
||||
|
||||
//Once we have a JSONStringer, we convert it to JSONObject generating a String and using JSONObject's contructor.
|
||||
//Once we have a JSONStringer, we convert it to JSONObject generating a String and using JSONObject's contructor.
|
||||
|
||||
String str = jsonStringer.toString();
|
||||
JSONObject jsonObject = new JSONObject(str);
|
||||
|
||||
System.out.println("Final JSONOBject: " + jsonObject);
|
||||
}
|
||||
String str = jsonStringer.toString();
|
||||
JSONObject jsonObject = new JSONObject(str);
|
||||
|
||||
System.out.println("Final JSONOBject: " + jsonObject);
|
||||
}
|
||||
```
|
||||
<h3>Using JSONObject</h3>
|
||||
|
||||
```java
|
||||
private static void JSONExampleObject1() {
|
||||
|
||||
//We can create a JSONObject from a String with the class builder
|
||||
|
||||
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||
JSONObject example = new JSONObject(string);
|
||||
System.out.println("Final JSONObject: " + example);
|
||||
|
||||
}
|
||||
```
|
||||
private static void JSONExampleObject1() {
|
||||
```java
|
||||
private static void JSONExampleObject2() {
|
||||
|
||||
//We can create a JSONObject from a String with the class builder
|
||||
//We can also create a JSONObject directly without messing around with any of the other functions.
|
||||
|
||||
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||
JSONObject example = new JSONObject(string);
|
||||
System.out.println("Final JSONObject: " + example);
|
||||
|
||||
}
|
||||
JSONObject example = new JSONObject();
|
||||
|
||||
|
||||
//Now we add the keys and values in a similar way as the Stringer method
|
||||
example.put("key", "value");
|
||||
|
||||
//As you can see, the first entry is the key and the second would be its associeted value.
|
||||
|
||||
example.put("key2", 5);
|
||||
example.put("key3", -23.45e67);
|
||||
example.put("trueValue", true);
|
||||
|
||||
//We can't add null values thougth
|
||||
|
||||
//example.put("nullValue", null); //This is not possible
|
||||
|
||||
System.out.println("Final JSONOBject: " + example);
|
||||
}
|
||||
```
|
||||
```
|
||||
private static void JSONExampleObject2() {
|
||||
```java
|
||||
private static void JSONExampleObject3() {
|
||||
|
||||
//We can also create a JSONObject directly without messing around with any of the other functions.
|
||||
//We can also create a JSONObject with a Java Map
|
||||
//YoU will need a Map whose keys are Strings. The values can be whatever you want
|
||||
|
||||
JSONObject example = new JSONObject();
|
||||
Map<String,Double> map = new HashMap<String, Double>();
|
||||
|
||||
map.put("key1", 1.0);
|
||||
map.put("key2", -23.45e67);
|
||||
|
||||
//Now we add the keys and values in a similar way as the Stringer method
|
||||
example.put("key", "value");
|
||||
//We create the JSONObject with the map with its class builder
|
||||
|
||||
//As you can see, the first entry is the key and the second would be its associeted value.
|
||||
|
||||
example.put("key2", 5);
|
||||
example.put("key3", -23.45e67);
|
||||
example.put("trueValue", true);
|
||||
|
||||
//We can't add null values thougth
|
||||
|
||||
//example.put("nullValue", null); //This is not possible
|
||||
|
||||
System.out.println("Final JSONOBject: " + example);
|
||||
}
|
||||
```
|
||||
```
|
||||
private static void JSONExampleObject3() {
|
||||
|
||||
//We can also create a JSONObject with a Java Map
|
||||
//YoU will need a Map whose keys are Strings. The values can be whatever you want
|
||||
|
||||
Map<String,Double> map = new HashMap<String, Double>();
|
||||
|
||||
map.put("key1", 1.0);
|
||||
map.put("key2", -23.45e67);
|
||||
|
||||
//We create the JSONObject with the map with its class builder
|
||||
|
||||
JSONObject example = new JSONObject(map);
|
||||
System.out.println("Final JSONOBject: " + example);
|
||||
}
|
||||
JSONObject example = new JSONObject(map);
|
||||
System.out.println("Final JSONOBject: " + example);
|
||||
}
|
||||
```
|
||||
<h3>Using JSONWriter</h3>
|
||||
|
||||
```
|
||||
private static void JSONExamplWriter() {
|
||||
```java
|
||||
private static void JSONExamplWriter() {
|
||||
|
||||
//This method works in a very similar way to Object and Stringer in the construction of the JSON.
|
||||
//The difference is that it needs a Java object called "Appendable" like StringBuilder
|
||||
//This method works in a very similar way to Object and Stringer in the construction of the JSON.
|
||||
//The difference is that it needs a Java object called "Appendable" like StringBuilder
|
||||
|
||||
StringBuilder write = new StringBuilder();
|
||||
JSONWriter jsonWriter = new JSONWriter(write);
|
||||
StringBuilder write = new StringBuilder();
|
||||
JSONWriter jsonWriter = new JSONWriter(write);
|
||||
|
||||
//We behave now the same way as Stringer
|
||||
//We behave now the same way as Stringer
|
||||
|
||||
jsonWriter.object();
|
||||
|
||||
jsonWriter.key("trueValue").value(true);
|
||||
jsonWriter.key("falseValue").value(false);
|
||||
jsonWriter.key("nullValue").value(null);
|
||||
jsonWriter.key("stringValue").value("hello world!");
|
||||
jsonWriter.key("complexStringValue").value("h\be\tllo w\u1234orld!");
|
||||
jsonWriter.key("intValue").value(42);
|
||||
jsonWriter.key("doubleValue").value(-23.45e67);
|
||||
|
||||
jsonWriter.endObject();
|
||||
|
||||
//The resoult should be in the "write" object
|
||||
|
||||
System.out.println("JSON: " + write.toString());
|
||||
|
||||
//The difference is that we don't get a JSONObject in this one.
|
||||
|
||||
|
||||
}
|
||||
jsonWriter.object();
|
||||
|
||||
jsonWriter.key("trueValue").value(true);
|
||||
jsonWriter.key("falseValue").value(false);
|
||||
jsonWriter.key("nullValue").value(null);
|
||||
jsonWriter.key("stringValue").value("hello world!");
|
||||
jsonWriter.key("complexStringValue").value("h\be\tllo w\u1234orld!");
|
||||
jsonWriter.key("intValue").value(42);
|
||||
jsonWriter.key("doubleValue").value(-23.45e67);
|
||||
|
||||
jsonWriter.endObject();
|
||||
|
||||
//The resoult should be in the "write" object
|
||||
|
||||
System.out.println("JSON: " + write.toString());
|
||||
|
||||
//The difference is that we don't get a JSONObject in this one.
|
||||
|
||||
|
||||
}
|
||||
```
|
||||
```
|
||||
private static void JSONExampleTokener() {
|
||||
```java
|
||||
private static void JSONExampleTokener() {
|
||||
|
||||
//A partir de una String podemos crear un JSONTokener, que lo podemos usar alternativamente para JSONArray,JSONObject
|
||||
//A partir de una String podemos crear un JSONTokener, que lo podemos usar alternativamente para JSONArray,JSONObject
|
||||
|
||||
String string = "this is not a valid JSON string";
|
||||
JSONTokener token = new JSONTokener(string);
|
||||
|
||||
//Now you can use the token in JSONObject and Array the same way as a String
|
||||
String string = "this is not a valid JSON string";
|
||||
JSONTokener token = new JSONTokener(string);
|
||||
|
||||
JSONObject object = new JSONObject(token);
|
||||
JSONArray array = new JSONArray(token);
|
||||
|
||||
}
|
||||
//Now you can use the token in JSONObject and Array the same way as a String
|
||||
|
||||
JSONObject object = new JSONObject(token);
|
||||
JSONArray array = new JSONArray(token);
|
||||
|
||||
}
|
||||
```
|
||||
<h2>Part 2: Conversion methods</h2>
|
||||
<p>We don't need to have a JSON document to work. This project also admits conversions from other type of files.</p>
|
||||
@@ -223,144 +223,144 @@ import java.util.Properties;
|
||||
|
||||
<h3>Extra: Conversion to JSONArray</h3>
|
||||
|
||||
```
|
||||
private static void JSONObjectToArray() {
|
||||
//We start with a JSONObject
|
||||
```java
|
||||
private static void JSONObjectToArray() {
|
||||
//We start with a JSONObject
|
||||
|
||||
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||
|
||||
JSONObject example = new JSONObject(string);
|
||||
|
||||
//We need a list of key strings like the reverse operation
|
||||
|
||||
JSONArray keyStrings = listNumberArray(example.length());
|
||||
|
||||
//Then we convert to the Array using both elelements
|
||||
|
||||
JSONArray array = example.toJSONArray(keyStrings);
|
||||
|
||||
System.out.println("Final JSONArray: " + array);
|
||||
}
|
||||
JSONObject example = new JSONObject(string);
|
||||
|
||||
//We need a list of key strings like the reverse operation
|
||||
|
||||
JSONArray keyStrings = listNumberArray(example.length());
|
||||
|
||||
//Then we convert to the Array using both elelements
|
||||
|
||||
JSONArray array = example.toJSONArray(keyStrings);
|
||||
|
||||
System.out.println("Final JSONArray: " + array);
|
||||
}
|
||||
```
|
||||
<h3>XML Conversions</h3>
|
||||
|
||||
```
|
||||
private static void XMLToExampleConversion() {
|
||||
```java
|
||||
private static void XMLToExampleConversion() {
|
||||
|
||||
//We start with a JSONObject
|
||||
|
||||
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||
JSONObject example = new JSONObject(string);
|
||||
//We start with a JSONObject
|
||||
|
||||
//We obtain a String with XML format with toString()
|
||||
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||
JSONObject example = new JSONObject(string);
|
||||
|
||||
String output = XML.toString(example);
|
||||
System.out.println("Final XML: " + output);
|
||||
}
|
||||
//We obtain a String with XML format with toString()
|
||||
|
||||
String output = XML.toString(example);
|
||||
System.out.println("Final XML: " + output);
|
||||
}
|
||||
```
|
||||
```
|
||||
private static void XMLFromExampleConversion() {
|
||||
```java
|
||||
private static void XMLFromExampleConversion() {
|
||||
|
||||
//We start with a string with the XML format
|
||||
//We start with a string with the XML format
|
||||
|
||||
String string = "<0>value</0><1>5</1><2>-2.345E+68</2><3>true</3>";
|
||||
String string = "<0>value</0><1>5</1><2>-2.345E+68</2><3>true</3>";
|
||||
|
||||
//We obtain a JSONObject with toJSONOBject()
|
||||
//We obtain a JSONObject with toJSONOBject()
|
||||
|
||||
JSONObject output = XML.toJSONObject(string);
|
||||
|
||||
System.out.println("Final JSONObject: " + output);
|
||||
}
|
||||
JSONObject output = XML.toJSONObject(string);
|
||||
|
||||
System.out.println("Final JSONObject: " + output);
|
||||
}
|
||||
```
|
||||
<h3>Cookie Conversions</h3>
|
||||
|
||||
```java
|
||||
private static void CookieToExampleConversion() {
|
||||
|
||||
//We start with a JSONObject
|
||||
//The JSONOBject needs to entries that gives the cookie a name and gives the field "name" a name too.
|
||||
//The Cokkie format doesn't support booleans
|
||||
|
||||
String string = "{\"name\":\"Cookie-Name\",\"value\":\"name\",\"1\":5,\"2\":-2.345E68,\"3\":'true'}";
|
||||
JSONObject example = new JSONObject(string);
|
||||
|
||||
//We obtain a String with Cookie format with toString()
|
||||
|
||||
String output = Cookie.toString(example);
|
||||
System.out.println("Final Cookie: " + output);
|
||||
}
|
||||
```
|
||||
private static void CookieToExampleConversion() {
|
||||
```java
|
||||
private static void CookieFromExampleConversion() {
|
||||
|
||||
//We start with a JSONObject
|
||||
//The JSONOBject needs to entries that gives the cookie a name and gives the field "name" a name too.
|
||||
//The Cokkie format doesn't support booleans
|
||||
//We start with a string with the Cookie format
|
||||
|
||||
String string = "{\"name\":\"Cookie-Name\",\"value\":\"name\",\"1\":5,\"2\":-2.345E68,\"3\":'true'}";
|
||||
JSONObject example = new JSONObject(string);
|
||||
|
||||
//We obtain a String with Cookie format with toString()
|
||||
String string = "Cookie-Name=name;1=5;2=-2.345E%2b68;3=true";
|
||||
|
||||
String output = Cookie.toString(example);
|
||||
System.out.println("Final Cookie: " + output);
|
||||
}
|
||||
```
|
||||
```
|
||||
private static void CookieFromExampleConversion() {
|
||||
//We obtain a JSONObject with toJSONOBject()
|
||||
|
||||
//We start with a string with the Cookie format
|
||||
|
||||
String string = "Cookie-Name=name;1=5;2=-2.345E%2b68;3=true";
|
||||
|
||||
//We obtain a JSONObject with toJSONOBject()
|
||||
|
||||
JSONObject output = Cookie.toJSONObject(string);
|
||||
System.out.println("Final JSONObject: " + output);
|
||||
}
|
||||
JSONObject output = Cookie.toJSONObject(string);
|
||||
System.out.println("Final JSONObject: " + output);
|
||||
}
|
||||
```
|
||||
|
||||
<h3>HTTP Conversions</h3>
|
||||
|
||||
```java
|
||||
private static void HTTPToExampleConversion() {
|
||||
|
||||
//We start with a JSONObject
|
||||
//The JSONObject must have the minimun header for a HTTP request or header
|
||||
|
||||
String string = "{\"Method\":\"POST\",\"Request-URI\":'/',\"HTTP-Version\":'HTTP/1.1',\"Value1\":true,\"Value2\":2,\"Value3\":-2.345E68}";
|
||||
|
||||
JSONObject example = new JSONObject(string);
|
||||
|
||||
//We obtain a String with HTTP format with toString()
|
||||
|
||||
String output = HTTP.toString(example);
|
||||
System.out.println("Final HTTP: " + output);
|
||||
}
|
||||
```
|
||||
private static void HTTPToExampleConversion() {
|
||||
```java
|
||||
private static void HTTPFromExampleConversion() {
|
||||
|
||||
//We start with a JSONObject
|
||||
//The JSONObject must have the minimun header for a HTTP request or header
|
||||
//We start with a string with the HTTP format
|
||||
|
||||
String string = "{\"Method\":\"POST\",\"Request-URI\":'/',\"HTTP-Version\":'HTTP/1.1',\"Value1\":true,\"Value2\":2,\"Value3\":-2.345E68}";
|
||||
String string = "Final HTTP: POST '/' HTTP/1.1 Value3: -2.345E+68 Value1: true Value2: 2";
|
||||
|
||||
JSONObject example = new JSONObject(string);
|
||||
//We obtain a JSONObject with toJSONOBject()
|
||||
|
||||
//We obtain a String with HTTP format with toString()
|
||||
|
||||
String output = HTTP.toString(example);
|
||||
System.out.println("Final HTTP: " + output);
|
||||
}
|
||||
```
|
||||
```
|
||||
private static void HTTPFromExampleConversion() {
|
||||
|
||||
//We start with a string with the HTTP format
|
||||
|
||||
String string = "Final HTTP: POST '/' HTTP/1.1 Value3: -2.345E+68 Value1: true Value2: 2";
|
||||
|
||||
//We obtain a JSONObject with toJSONOBject()
|
||||
|
||||
JSONObject output = HTTP.toJSONObject(string);
|
||||
System.out.println("Final JSONObject: " + output);
|
||||
}
|
||||
JSONObject output = HTTP.toJSONObject(string);
|
||||
System.out.println("Final JSONObject: " + output);
|
||||
}
|
||||
```
|
||||
<h3>CDL Conversions</h3>
|
||||
|
||||
```
|
||||
```java
|
||||
private static void CDLToExampleConversion() {
|
||||
|
||||
//We start with some JSONObjects with the same values in the keys but different values in the "values"
|
||||
//We start with some JSONObjects with the same values in the keys but different values in the "values"
|
||||
|
||||
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||
JSONObject example = new JSONObject(string);
|
||||
|
||||
String string2 = "{\"0\":\"value2\",\"1\":6,\"2\":-8.345E68,\"3\":false}";
|
||||
JSONObject example2 = new JSONObject(string2);
|
||||
|
||||
//We need now a JSONArray with those JSONObjects
|
||||
String string = "{\"0\":\"value\",\"1\":5,\"2\":-2.345E68,\"3\":true}";
|
||||
JSONObject example = new JSONObject(string);
|
||||
|
||||
JSONArray array = new JSONArray();
|
||||
array.put(example);
|
||||
array.put(example2);
|
||||
String string2 = "{\"0\":\"value2\",\"1\":6,\"2\":-8.345E68,\"3\":false}";
|
||||
JSONObject example2 = new JSONObject(string2);
|
||||
|
||||
//We obtain a String with XML format with toString()
|
||||
//We need now a JSONArray with those JSONObjects
|
||||
|
||||
String output = CDL.toString(array);
|
||||
System.out.println("Final CDL: \r\n" + output);
|
||||
}
|
||||
```
|
||||
JSONArray array = new JSONArray();
|
||||
array.put(example);
|
||||
array.put(example2);
|
||||
|
||||
//We obtain a String with XML format with toString()
|
||||
|
||||
String output = CDL.toString(array);
|
||||
System.out.println("Final CDL: \r\n" + output);
|
||||
}
|
||||
```
|
||||
```java
|
||||
private static void CDLFromExampleConversion() {
|
||||
|
||||
//We start wtih a String with the CDL format
|
||||
@@ -368,7 +368,7 @@ private static void CDLFromExampleConversion() {
|
||||
String string = "0,1,2,3\n"
|
||||
+ "value,5,-2.345E+68,true\n"
|
||||
+ "value2,6,-8.345E+68,false";
|
||||
|
||||
|
||||
//We obtain a JSONArray with toJSONOBject()
|
||||
|
||||
JSONArray output = CDL.toJSONArray(string);
|
||||
@@ -377,8 +377,8 @@ private static void CDLFromExampleConversion() {
|
||||
```
|
||||
<h3>Properties Conversions</h3>
|
||||
|
||||
```
|
||||
private static Properties PropertyToExampleConversion() {
|
||||
```java
|
||||
private static Properties PropertyToExampleConversion() {
|
||||
|
||||
//We start with a JSONObject
|
||||
|
||||
@@ -391,43 +391,43 @@ private static void CDLFromExampleConversion() {
|
||||
System.out.println("Final Properties: " + output);
|
||||
|
||||
return output;
|
||||
}
|
||||
}
|
||||
```
|
||||
```
|
||||
private static void PropertyFromExampleConversion() {
|
||||
```java
|
||||
private static void PropertyFromExampleConversion() {
|
||||
|
||||
//We start with a Properties object
|
||||
//We start with a Properties object
|
||||
|
||||
Properties input = PropertyToExampleConversion();
|
||||
Properties input = PropertyToExampleConversion();
|
||||
|
||||
//We obtain a JSONObject with toJSONOBject()
|
||||
//We obtain a JSONObject with toJSONOBject()
|
||||
|
||||
JSONObject output = Property.toJSONObject(input);
|
||||
System.out.println("Final JSONObject: " + output);
|
||||
}
|
||||
JSONObject output = Property.toJSONObject(input);
|
||||
System.out.println("Final JSONObject: " + output);
|
||||
}
|
||||
```
|
||||
<h2>List of all examples methods</h2>
|
||||
<h2>List of all examples methods</h2>
|
||||
|
||||
```
|
||||
public static void main(String[] args) {
|
||||
//JSONObjectToArray();
|
||||
//JSONExampleArray1();
|
||||
//JSONExampleArray2();
|
||||
//JSONExampleStringer();
|
||||
//JSONExampleObject1();
|
||||
//JSONExampleObject2();
|
||||
//JSONExampleObject3();
|
||||
//JSONExamplWriter();
|
||||
//XMLToExampleConversion();
|
||||
//XMLFromExampleConversion();
|
||||
//CookieToExampleConversion();
|
||||
//CookieFromExampleConversion();
|
||||
//HTTPToExampleConversion();
|
||||
//HTTPFromExampleConversion();
|
||||
//CDLToExampleConversion();
|
||||
//CDLFromExampleConversion();
|
||||
//PropertyToExampleConversion();
|
||||
//PropertyFromExampleConversion();
|
||||
}
|
||||
```java
|
||||
public static void main(String[] args) {
|
||||
//JSONObjectToArray();
|
||||
//JSONExampleArray1();
|
||||
//JSONExampleArray2();
|
||||
//JSONExampleStringer();
|
||||
//JSONExampleObject1();
|
||||
//JSONExampleObject2();
|
||||
//JSONExampleObject3();
|
||||
//JSONExamplWriter();
|
||||
//XMLToExampleConversion();
|
||||
//XMLFromExampleConversion();
|
||||
//CookieToExampleConversion();
|
||||
//CookieFromExampleConversion();
|
||||
//HTTPToExampleConversion();
|
||||
//HTTPFromExampleConversion();
|
||||
//CDLToExampleConversion();
|
||||
//CDLFromExampleConversion();
|
||||
//PropertyToExampleConversion();
|
||||
//PropertyFromExampleConversion();
|
||||
}
|
||||
```
|
||||
|
||||
|
||||
@@ -8,7 +8,7 @@ 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://search.maven.org/remotecontent?filepath=org/json/json/20220320/json-20220320.jar)**
|
||||
**[Click here if you just want the latest release jar file.](https://search.maven.org/remotecontent?filepath=org/json/json/20230618/json-20230618.jar)**
|
||||
|
||||
|
||||
# Overview
|
||||
|
||||
@@ -5,6 +5,12 @@ and artifactId "json". For example:
|
||||
[https://search.maven.org/search?q=g:org.json%20AND%20a:json&core=gav](https://search.maven.org/search?q=g:org.json%20AND%20a:json&core=gav)
|
||||
|
||||
~~~
|
||||
20230618 Final release with Java 1.6 compatibility. Future releases will require Java 1.8 or greater.
|
||||
|
||||
20230227 Fix for CVE-2022-45688 and recent commits
|
||||
|
||||
20220924 New License - public domain, and some minor updates
|
||||
|
||||
20220320 Wrap StackOverflow with JSONException
|
||||
|
||||
20211205 Recent commits and some bug fixes for similar()
|
||||
@@ -18,6 +24,7 @@ and artifactId "json". For example:
|
||||
20190722 Recent commits
|
||||
|
||||
20180813 POM change to include Automatic-Module-Name (#431)
|
||||
JSONObject(Map) now throws an exception if any of a map keys are null (#405)
|
||||
|
||||
20180130 Recent commits
|
||||
|
||||
|
||||
2
pom.xml
2
pom.xml
@@ -3,7 +3,7 @@
|
||||
|
||||
<groupId>org.json</groupId>
|
||||
<artifactId>json</artifactId>
|
||||
<version>20220320</version>
|
||||
<version>20230618</version>
|
||||
<packaging>bundle</packaging>
|
||||
|
||||
<name>JSON in Java</name>
|
||||
|
||||
@@ -27,7 +27,32 @@ public class JSONML {
|
||||
XMLTokener x,
|
||||
boolean arrayForm,
|
||||
JSONArray ja,
|
||||
boolean keepStrings
|
||||
boolean keepStrings,
|
||||
int currentNestingDepth
|
||||
) throws JSONException {
|
||||
return parse(x,arrayForm, ja,
|
||||
keepStrings ? JSONMLParserConfiguration.KEEP_STRINGS : JSONMLParserConfiguration.ORIGINAL,
|
||||
currentNestingDepth);
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse XML values and store them in a JSONArray.
|
||||
* @param x The XMLTokener containing the source string.
|
||||
* @param arrayForm true if array form, false if object form.
|
||||
* @param ja The JSONArray that is containing the current tag or null
|
||||
* if we are at the outermost level.
|
||||
* @param config The parser configuration:
|
||||
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||
* JSONMLParserConfiguration.KEEP_STRINGS means Don't type-convert text nodes and attribute values.
|
||||
* @return A JSONArray if the value is the outermost tag, otherwise null.
|
||||
* @throws JSONException if a parsing error occurs
|
||||
*/
|
||||
private static Object parse(
|
||||
XMLTokener x,
|
||||
boolean arrayForm,
|
||||
JSONArray ja,
|
||||
JSONMLParserConfiguration config,
|
||||
int currentNestingDepth
|
||||
) throws JSONException {
|
||||
String attribute;
|
||||
char c;
|
||||
@@ -152,7 +177,7 @@ public class JSONML {
|
||||
if (!(token instanceof String)) {
|
||||
throw x.syntaxError("Missing value");
|
||||
}
|
||||
newjo.accumulate(attribute, keepStrings ? ((String)token) :XML.stringToValue((String)token));
|
||||
newjo.accumulate(attribute, config.isKeepStrings() ? ((String)token) :XML.stringToValue((String)token));
|
||||
token = null;
|
||||
} else {
|
||||
newjo.accumulate(attribute, "");
|
||||
@@ -181,7 +206,12 @@ public class JSONML {
|
||||
if (token != XML.GT) {
|
||||
throw x.syntaxError("Misshaped tag");
|
||||
}
|
||||
closeTag = (String)parse(x, arrayForm, newja, keepStrings);
|
||||
|
||||
if (currentNestingDepth == config.getMaxNestingDepth()) {
|
||||
throw x.syntaxError("Maximum nesting depth of " + config.getMaxNestingDepth() + " reached");
|
||||
}
|
||||
|
||||
closeTag = (String)parse(x, arrayForm, newja, config, currentNestingDepth + 1);
|
||||
if (closeTag != null) {
|
||||
if (!closeTag.equals(tagName)) {
|
||||
throw x.syntaxError("Mismatched '" + tagName +
|
||||
@@ -203,7 +233,7 @@ public class JSONML {
|
||||
} else {
|
||||
if (ja != null) {
|
||||
ja.put(token instanceof String
|
||||
? keepStrings ? XML.unescape((String)token) :XML.stringToValue((String)token)
|
||||
? (config.isKeepStrings() ? XML.unescape((String)token) : XML.stringToValue((String)token))
|
||||
: token);
|
||||
}
|
||||
}
|
||||
@@ -224,7 +254,7 @@ public class JSONML {
|
||||
* @throws JSONException Thrown on error converting to a JSONArray
|
||||
*/
|
||||
public static JSONArray toJSONArray(String string) throws JSONException {
|
||||
return (JSONArray)parse(new XMLTokener(string), true, null, false);
|
||||
return (JSONArray)parse(new XMLTokener(string), true, null, JSONMLParserConfiguration.ORIGINAL, 0);
|
||||
}
|
||||
|
||||
|
||||
@@ -235,8 +265,8 @@ public class JSONML {
|
||||
* attributes, then the second element will be JSONObject containing the
|
||||
* name/value pairs. If the tag contains children, then strings and
|
||||
* JSONArrays will represent the child tags.
|
||||
* As opposed to toJSONArray this method does not attempt to convert
|
||||
* any text node or attribute value to any type
|
||||
* As opposed to toJSONArray this method does not attempt to convert
|
||||
* any text node or attribute value to any type
|
||||
* but just leaves it as a string.
|
||||
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||
* @param string The source string.
|
||||
@@ -246,7 +276,32 @@ public class JSONML {
|
||||
* @throws JSONException Thrown on error converting to a JSONArray
|
||||
*/
|
||||
public static JSONArray toJSONArray(String string, boolean keepStrings) throws JSONException {
|
||||
return (JSONArray)parse(new XMLTokener(string), true, null, keepStrings);
|
||||
return (JSONArray)parse(new XMLTokener(string), true, null, keepStrings, 0);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||
* JSONArray using the JsonML transform. Each XML tag is represented as
|
||||
* a JSONArray in which the first element is the tag name. If the tag has
|
||||
* attributes, then the second element will be JSONObject containing the
|
||||
* name/value pairs. If the tag contains children, then strings and
|
||||
* JSONArrays will represent the child tags.
|
||||
* As opposed to toJSONArray this method does not attempt to convert
|
||||
* any text node or attribute value to any type
|
||||
* but just leaves it as a string.
|
||||
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||
* @param string The source string.
|
||||
* @param config The parser configuration:
|
||||
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||
* JSONMLParserConfiguration.KEEP_STRINGS means values will not be coerced into boolean
|
||||
* or numeric values and will instead be left as strings
|
||||
* @return A JSONArray containing the structured data from the XML string.
|
||||
* @throws JSONException Thrown on error converting to a JSONArray
|
||||
*/
|
||||
public static JSONArray toJSONArray(String string, JSONMLParserConfiguration config) throws JSONException {
|
||||
return (JSONArray)parse(new XMLTokener(string), true, null, config, 0);
|
||||
}
|
||||
|
||||
|
||||
@@ -257,8 +312,32 @@ public class JSONML {
|
||||
* attributes, then the second element will be JSONObject containing the
|
||||
* name/value pairs. If the tag contains children, then strings and
|
||||
* JSONArrays will represent the child content and tags.
|
||||
* As opposed to toJSONArray this method does not attempt to convert
|
||||
* any text node or attribute value to any type
|
||||
* As opposed to toJSONArray this method does not attempt to convert
|
||||
* any text node or attribute value to any type
|
||||
* but just leaves it as a string.
|
||||
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||
* @param x An XMLTokener.
|
||||
* @param config The parser configuration:
|
||||
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||
* JSONMLParserConfiguration.KEEP_STRINGS means values will not be coerced into boolean
|
||||
* or numeric values and will instead be left as strings
|
||||
* @return A JSONArray containing the structured data from the XML string.
|
||||
* @throws JSONException Thrown on error converting to a JSONArray
|
||||
*/
|
||||
public static JSONArray toJSONArray(XMLTokener x, JSONMLParserConfiguration config) throws JSONException {
|
||||
return (JSONArray)parse(x, true, null, config, 0);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||
* JSONArray using the JsonML transform. Each XML tag is represented as
|
||||
* a JSONArray in which the first element is the tag name. If the tag has
|
||||
* attributes, then the second element will be JSONObject containing the
|
||||
* name/value pairs. If the tag contains children, then strings and
|
||||
* JSONArrays will represent the child content and tags.
|
||||
* As opposed to toJSONArray this method does not attempt to convert
|
||||
* any text node or attribute value to any type
|
||||
* but just leaves it as a string.
|
||||
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||
* @param x An XMLTokener.
|
||||
@@ -268,7 +347,7 @@ public class JSONML {
|
||||
* @throws JSONException Thrown on error converting to a JSONArray
|
||||
*/
|
||||
public static JSONArray toJSONArray(XMLTokener x, boolean keepStrings) throws JSONException {
|
||||
return (JSONArray)parse(x, true, null, keepStrings);
|
||||
return (JSONArray)parse(x, true, null, keepStrings, 0);
|
||||
}
|
||||
|
||||
|
||||
@@ -285,7 +364,7 @@ public class JSONML {
|
||||
* @throws JSONException Thrown on error converting to a JSONArray
|
||||
*/
|
||||
public static JSONArray toJSONArray(XMLTokener x) throws JSONException {
|
||||
return (JSONArray)parse(x, true, null, false);
|
||||
return (JSONArray)parse(x, true, null, false, 0);
|
||||
}
|
||||
|
||||
|
||||
@@ -303,10 +382,10 @@ public class JSONML {
|
||||
* @throws JSONException Thrown on error converting to a JSONObject
|
||||
*/
|
||||
public static JSONObject toJSONObject(String string) throws JSONException {
|
||||
return (JSONObject)parse(new XMLTokener(string), false, null, false);
|
||||
return (JSONObject)parse(new XMLTokener(string), false, null, false, 0);
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||
* JSONObject using the JsonML transform. Each XML tag is represented as
|
||||
@@ -323,10 +402,32 @@ public class JSONML {
|
||||
* @throws JSONException Thrown on error converting to a JSONObject
|
||||
*/
|
||||
public static JSONObject toJSONObject(String string, boolean keepStrings) throws JSONException {
|
||||
return (JSONObject)parse(new XMLTokener(string), false, null, keepStrings);
|
||||
return (JSONObject)parse(new XMLTokener(string), false, null, keepStrings, 0);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||
* JSONObject using the JsonML transform. Each XML tag is represented as
|
||||
* a JSONObject with a "tagName" property. If the tag has attributes, then
|
||||
* the attributes will be in the JSONObject as properties. If the tag
|
||||
* contains children, the object will have a "childNodes" property which
|
||||
* will be an array of strings and JsonML JSONObjects.
|
||||
|
||||
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||
* @param string The XML source text.
|
||||
* @param config The parser configuration:
|
||||
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||
* JSONMLParserConfiguration.KEEP_STRINGS means values will not be coerced into boolean
|
||||
* or numeric values and will instead be left as strings
|
||||
* @return A JSONObject containing the structured data from the XML string.
|
||||
* @throws JSONException Thrown on error converting to a JSONObject
|
||||
*/
|
||||
public static JSONObject toJSONObject(String string, JSONMLParserConfiguration config) throws JSONException {
|
||||
return (JSONObject)parse(new XMLTokener(string), false, null, config, 0);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||
* JSONObject using the JsonML transform. Each XML tag is represented as
|
||||
@@ -341,7 +442,7 @@ public class JSONML {
|
||||
* @throws JSONException Thrown on error converting to a JSONObject
|
||||
*/
|
||||
public static JSONObject toJSONObject(XMLTokener x) throws JSONException {
|
||||
return (JSONObject)parse(x, false, null, false);
|
||||
return (JSONObject)parse(x, false, null, false, 0);
|
||||
}
|
||||
|
||||
|
||||
@@ -361,7 +462,29 @@ public class JSONML {
|
||||
* @throws JSONException Thrown on error converting to a JSONObject
|
||||
*/
|
||||
public static JSONObject toJSONObject(XMLTokener x, boolean keepStrings) throws JSONException {
|
||||
return (JSONObject)parse(x, false, null, keepStrings);
|
||||
return (JSONObject)parse(x, false, null, keepStrings, 0);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||
* JSONObject using the JsonML transform. Each XML tag is represented as
|
||||
* a JSONObject with a "tagName" property. If the tag has attributes, then
|
||||
* the attributes will be in the JSONObject as properties. If the tag
|
||||
* contains children, the object will have a "childNodes" property which
|
||||
* will be an array of strings and JsonML JSONObjects.
|
||||
|
||||
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||
* @param x An XMLTokener of the XML source text.
|
||||
* @param config The parser configuration:
|
||||
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||
* JSONMLParserConfiguration.KEEP_STRINGS means values will not be coerced into boolean
|
||||
* or numeric values and will instead be left as strings
|
||||
* @return A JSONObject containing the structured data from the XML string.
|
||||
* @throws JSONException Thrown on error converting to a JSONObject
|
||||
*/
|
||||
public static JSONObject toJSONObject(XMLTokener x, JSONMLParserConfiguration config) throws JSONException {
|
||||
return (JSONObject)parse(x, false, null, config, 0);
|
||||
}
|
||||
|
||||
|
||||
@@ -442,6 +565,7 @@ public class JSONML {
|
||||
return sb.toString();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Reverse the JSONML transformation, making an XML text from a JSONObject.
|
||||
* The JSONObject must contain a "tagName" property. If it has children,
|
||||
|
||||
67
src/main/java/org/json/JSONMLParserConfiguration.java
Normal file
67
src/main/java/org/json/JSONMLParserConfiguration.java
Normal file
@@ -0,0 +1,67 @@
|
||||
package org.json;
|
||||
/*
|
||||
Public Domain.
|
||||
*/
|
||||
|
||||
/**
|
||||
* Configuration object for the XML to JSONML parser. The configuration is immutable.
|
||||
*/
|
||||
@SuppressWarnings({""})
|
||||
public class JSONMLParserConfiguration extends ParserConfiguration {
|
||||
|
||||
/**
|
||||
* We can override the default maximum nesting depth if needed.
|
||||
*/
|
||||
public static final int DEFAULT_MAXIMUM_NESTING_DEPTH = ParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH;
|
||||
|
||||
/** Original Configuration of the XML to JSONML Parser. */
|
||||
public static final JSONMLParserConfiguration ORIGINAL
|
||||
= new JSONMLParserConfiguration();
|
||||
/** Original configuration of the XML to JSONML Parser except that values are kept as strings. */
|
||||
public static final JSONMLParserConfiguration KEEP_STRINGS
|
||||
= new JSONMLParserConfiguration().withKeepStrings(true);
|
||||
|
||||
/**
|
||||
* Default parser configuration. Does not keep strings (tries to implicitly convert values).
|
||||
*/
|
||||
public JSONMLParserConfiguration() {
|
||||
super();
|
||||
this.maxNestingDepth = DEFAULT_MAXIMUM_NESTING_DEPTH;
|
||||
}
|
||||
|
||||
/**
|
||||
* Configure the parser string processing and use the default CDATA Tag Name as "content".
|
||||
* @param keepStrings <code>true</code> to parse all values as string.
|
||||
* <code>false</code> to try and convert XML string values into a JSON value.
|
||||
* @param maxNestingDepth <code>int</code> to limit the nesting depth
|
||||
*/
|
||||
protected JSONMLParserConfiguration(final boolean keepStrings, final int maxNestingDepth) {
|
||||
super(keepStrings, maxNestingDepth);
|
||||
}
|
||||
|
||||
/**
|
||||
* Provides a new instance of the same configuration.
|
||||
*/
|
||||
@Override
|
||||
protected JSONMLParserConfiguration clone() {
|
||||
// future modifications to this method should always ensure a "deep"
|
||||
// clone in the case of collections. i.e. if a Map is added as a configuration
|
||||
// item, a new map instance should be created and if possible each value in the
|
||||
// map should be cloned as well. If the values of the map are known to also
|
||||
// be immutable, then a shallow clone of the map is acceptable.
|
||||
return new JSONMLParserConfiguration(
|
||||
this.keepStrings,
|
||||
this.maxNestingDepth
|
||||
);
|
||||
}
|
||||
|
||||
@Override
|
||||
public JSONMLParserConfiguration withKeepStrings(final boolean newVal) {
|
||||
return super.withKeepStrings(newVal);
|
||||
}
|
||||
|
||||
@Override
|
||||
public JSONMLParserConfiguration withMaxNestingDepth(int maxNestingDepth) {
|
||||
return super.withMaxNestingDepth(maxNestingDepth);
|
||||
}
|
||||
}
|
||||
@@ -1290,11 +1290,7 @@ public class JSONObject {
|
||||
if (val == null) {
|
||||
return defaultValue;
|
||||
}
|
||||
final double doubleValue = val.doubleValue();
|
||||
// if (Double.isNaN(doubleValue) || Double.isInfinite(doubleValue)) {
|
||||
// return defaultValue;
|
||||
// }
|
||||
return doubleValue;
|
||||
return val.doubleValue();
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@@ -11,13 +11,13 @@ 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.
|
||||
*/
|
||||
@Documented
|
||||
@Retention(RUNTIME)
|
||||
@Target({METHOD})
|
||||
public @interface JSONPropertyIgnore { }
|
||||
|
||||
@@ -11,14 +11,14 @@ 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.
|
||||
*/
|
||||
@Documented
|
||||
@Retention(RUNTIME)
|
||||
@Target({METHOD})
|
||||
public @interface JSONPropertyName {
|
||||
/**
|
||||
* @return The name of the property as to be used in the JSON Object.
|
||||
|
||||
@@ -1,11 +1,7 @@
|
||||
package org.json;
|
||||
|
||||
import java.io.BufferedReader;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.io.InputStreamReader;
|
||||
import java.io.Reader;
|
||||
import java.io.StringReader;
|
||||
import java.io.*;
|
||||
import java.nio.charset.Charset;
|
||||
|
||||
/*
|
||||
Public Domain.
|
||||
@@ -61,7 +57,7 @@ public class JSONTokener {
|
||||
* @param inputStream The source.
|
||||
*/
|
||||
public JSONTokener(InputStream inputStream) {
|
||||
this(new InputStreamReader(inputStream));
|
||||
this(new InputStreamReader(inputStream, Charset.forName("UTF-8")));
|
||||
}
|
||||
|
||||
|
||||
@@ -125,7 +121,7 @@ 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
|
||||
*/
|
||||
public boolean end() {
|
||||
@@ -189,7 +185,7 @@ public class JSONTokener {
|
||||
this.previous = (char) c;
|
||||
return this.previous;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Get the last character read from the input or '\0' if nothing has been read yet.
|
||||
* @return the last character read from the input.
|
||||
@@ -522,4 +518,10 @@ public class JSONTokener {
|
||||
return " at " + this.index + " [character " + this.character + " line " +
|
||||
this.line + "]";
|
||||
}
|
||||
|
||||
public void close() throws IOException {
|
||||
if(reader!=null){
|
||||
reader.close();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
112
src/main/java/org/json/ParserConfiguration.java
Normal file
112
src/main/java/org/json/ParserConfiguration.java
Normal file
@@ -0,0 +1,112 @@
|
||||
package org.json;
|
||||
/*
|
||||
Public Domain.
|
||||
*/
|
||||
|
||||
/**
|
||||
* Configuration base object for parsers. The configuration is immutable.
|
||||
*/
|
||||
@SuppressWarnings({""})
|
||||
public class ParserConfiguration {
|
||||
/**
|
||||
* Used to indicate there's no defined limit to the maximum nesting depth when parsing a document.
|
||||
*/
|
||||
public static final int UNDEFINED_MAXIMUM_NESTING_DEPTH = -1;
|
||||
|
||||
/**
|
||||
* The default maximum nesting depth when parsing a document.
|
||||
*/
|
||||
public static final int DEFAULT_MAXIMUM_NESTING_DEPTH = 512;
|
||||
|
||||
/**
|
||||
* Specifies if values should be kept as strings (<code>true</code>), or if
|
||||
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||
*/
|
||||
protected boolean keepStrings;
|
||||
|
||||
/**
|
||||
* The maximum nesting depth when parsing a document.
|
||||
*/
|
||||
protected int maxNestingDepth;
|
||||
|
||||
public ParserConfiguration() {
|
||||
this.keepStrings = false;
|
||||
this.maxNestingDepth = DEFAULT_MAXIMUM_NESTING_DEPTH;
|
||||
}
|
||||
|
||||
protected ParserConfiguration(final boolean keepStrings, final int maxNestingDepth) {
|
||||
this.keepStrings = keepStrings;
|
||||
this.maxNestingDepth = maxNestingDepth;
|
||||
}
|
||||
|
||||
/**
|
||||
* Provides a new instance of the same configuration.
|
||||
*/
|
||||
@Override
|
||||
protected ParserConfiguration clone() {
|
||||
// future modifications to this method should always ensure a "deep"
|
||||
// clone in the case of collections. i.e. if a Map is added as a configuration
|
||||
// item, a new map instance should be created and if possible each value in the
|
||||
// map should be cloned as well. If the values of the map are known to also
|
||||
// be immutable, then a shallow clone of the map is acceptable.
|
||||
return new ParserConfiguration(
|
||||
this.keepStrings,
|
||||
this.maxNestingDepth
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* When parsing the XML into JSONML, specifies if values should be kept as strings (<code>true</code>), or if
|
||||
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||
*
|
||||
* @return The <code>keepStrings</code> configuration value.
|
||||
*/
|
||||
public boolean isKeepStrings() {
|
||||
return this.keepStrings;
|
||||
}
|
||||
|
||||
/**
|
||||
* When parsing the XML into JSONML, specifies if values should be kept as strings (<code>true</code>), or if
|
||||
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||
*
|
||||
* @param newVal
|
||||
* new value to use for the <code>keepStrings</code> configuration option.
|
||||
*
|
||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||
*/
|
||||
public <T extends ParserConfiguration> T withKeepStrings(final boolean newVal) {
|
||||
T newConfig = (T)this.clone();
|
||||
newConfig.keepStrings = newVal;
|
||||
return newConfig;
|
||||
}
|
||||
|
||||
/**
|
||||
* The maximum nesting depth that the parser will descend before throwing an exception
|
||||
* when parsing the XML into JSONML.
|
||||
* @return the maximum nesting depth set for this configuration
|
||||
*/
|
||||
public int getMaxNestingDepth() {
|
||||
return maxNestingDepth;
|
||||
}
|
||||
|
||||
/**
|
||||
* Defines the maximum nesting depth that the parser will descend before throwing an exception
|
||||
* when parsing the XML into JSONML. The default max nesting depth is 512, which means the parser
|
||||
* will throw a JsonException if the maximum depth is reached.
|
||||
* Using any negative value as a parameter is equivalent to setting no limit to the nesting depth,
|
||||
* which means the parses will go as deep as the maximum call stack size allows.
|
||||
* @param maxNestingDepth the maximum nesting depth allowed to the XML parser
|
||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||
*/
|
||||
public <T extends ParserConfiguration> T withMaxNestingDepth(int maxNestingDepth) {
|
||||
T newConfig = (T)this.clone();
|
||||
|
||||
if (maxNestingDepth > UNDEFINED_MAXIMUM_NESTING_DEPTH) {
|
||||
newConfig.maxNestingDepth = maxNestingDepth;
|
||||
} else {
|
||||
newConfig.maxNestingDepth = UNDEFINED_MAXIMUM_NESTING_DEPTH;
|
||||
}
|
||||
|
||||
return newConfig;
|
||||
}
|
||||
}
|
||||
@@ -98,7 +98,7 @@ public class XML {
|
||||
/**
|
||||
* Replace special characters with XML escapes:
|
||||
*
|
||||
* <pre>{@code
|
||||
* <pre>{@code
|
||||
* & (ampersand) is replaced by &amp;
|
||||
* < (less than) is replaced by &lt;
|
||||
* > (greater than) is replaced by &gt;
|
||||
@@ -229,10 +229,14 @@ public class XML {
|
||||
* The JSONObject that will include the new material.
|
||||
* @param name
|
||||
* The tag name.
|
||||
* @param config
|
||||
* The XML parser configuration.
|
||||
* @param currentNestingDepth
|
||||
* The current nesting depth.
|
||||
* @return true if the close tag is processed.
|
||||
* @throws JSONException
|
||||
* @throws JSONException Thrown if any parsing error occurs.
|
||||
*/
|
||||
private static boolean parse(XMLTokener x, JSONObject context, String name, XMLParserConfiguration config)
|
||||
private static boolean parse(XMLTokener x, JSONObject context, String name, XMLParserConfiguration config, int currentNestingDepth)
|
||||
throws JSONException {
|
||||
char c;
|
||||
int i;
|
||||
@@ -402,7 +406,11 @@ public class XML {
|
||||
|
||||
} else if (token == LT) {
|
||||
// Nested element
|
||||
if (parse(x, jsonObject, tagName, config)) {
|
||||
if (currentNestingDepth == config.getMaxNestingDepth()) {
|
||||
throw x.syntaxError("Maximum nesting depth of " + config.getMaxNestingDepth() + " reached");
|
||||
}
|
||||
|
||||
if (parse(x, jsonObject, tagName, config, currentNestingDepth + 1)) {
|
||||
if (config.getForceList().contains(tagName)) {
|
||||
// Force the value to be an array
|
||||
if (jsonObject.length() == 0) {
|
||||
@@ -423,7 +431,7 @@ public class XML {
|
||||
context.accumulate(tagName, jsonObject);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
@@ -487,7 +495,7 @@ public class XML {
|
||||
}
|
||||
return string;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* direct copy of {@link JSONObject#stringToNumber(String)} to maintain Android support.
|
||||
*/
|
||||
@@ -534,7 +542,7 @@ public class XML {
|
||||
// integer representation.
|
||||
// This will narrow any values to the smallest reasonable Object representation
|
||||
// (Integer, Long, or BigInteger)
|
||||
|
||||
|
||||
// BigInteger down conversion: We use a similar bitLength compare as
|
||||
// BigInteger#intValueExact uses. Increases GC, but objects hold
|
||||
// only what they need. i.e. Less runtime overhead if the value is
|
||||
@@ -550,7 +558,7 @@ public class XML {
|
||||
}
|
||||
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* direct copy of {@link JSONObject#isDecimalNotation(String)} to maintain Android support.
|
||||
*/
|
||||
@@ -568,7 +576,7 @@ public class XML {
|
||||
* 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 <pre>{@code
|
||||
* "content" member. Comments, prologs, DTDs, and <pre>{@code
|
||||
* <[ [ ]]>}</pre>
|
||||
* are ignored.
|
||||
*
|
||||
@@ -589,7 +597,7 @@ public class XML {
|
||||
* 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 <pre>{@code
|
||||
* "content" member. Comments, prologs, DTDs, and <pre>{@code
|
||||
* <[ [ ]]>}</pre>
|
||||
* are ignored.
|
||||
*
|
||||
@@ -655,7 +663,7 @@ public class XML {
|
||||
while (x.more()) {
|
||||
x.skipPast("<");
|
||||
if(x.more()) {
|
||||
parse(x, jo, null, config);
|
||||
parse(x, jo, null, config, 0);
|
||||
}
|
||||
}
|
||||
return jo;
|
||||
@@ -669,7 +677,7 @@ public class XML {
|
||||
* 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 <pre>{@code
|
||||
* "content" member. Comments, prologs, DTDs, and <pre>{@code
|
||||
* <[ [ ]]>}</pre>
|
||||
* are ignored.
|
||||
*
|
||||
@@ -695,7 +703,7 @@ public class XML {
|
||||
* 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 <pre>{@code
|
||||
* "content" member. Comments, prologs, DTDs, and <pre>{@code
|
||||
* <[ [ ]]>}</pre>
|
||||
* are ignored.
|
||||
*
|
||||
@@ -752,6 +760,28 @@ public class XML {
|
||||
*/
|
||||
public static String toString(final Object object, final String tagName, final XMLParserConfiguration config)
|
||||
throws JSONException {
|
||||
return toString(object, tagName, config, 0, 0);
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert a JSONObject into a well-formed, element-normal XML string,
|
||||
* either pretty print or single-lined depending on indent factor.
|
||||
*
|
||||
* @param object
|
||||
* A JSONObject.
|
||||
* @param tagName
|
||||
* The optional name of the enclosing tag.
|
||||
* @param config
|
||||
* Configuration that can control output to XML.
|
||||
* @param indentFactor
|
||||
* The number of spaces to add to each level of indentation.
|
||||
* @param indent
|
||||
* The current ident level in spaces.
|
||||
* @return
|
||||
* @throws JSONException
|
||||
*/
|
||||
private static String toString(final Object object, final String tagName, final XMLParserConfiguration config, int indentFactor, int indent)
|
||||
throws JSONException {
|
||||
StringBuilder sb = new StringBuilder();
|
||||
JSONArray ja;
|
||||
JSONObject jo;
|
||||
@@ -761,9 +791,14 @@ public class XML {
|
||||
|
||||
// Emit <tagName>
|
||||
if (tagName != null) {
|
||||
sb.append(indent(indent));
|
||||
sb.append('<');
|
||||
sb.append(tagName);
|
||||
sb.append('>');
|
||||
if(indentFactor > 0){
|
||||
sb.append("\n");
|
||||
indent += indentFactor;
|
||||
}
|
||||
}
|
||||
|
||||
// Loop thru the keys.
|
||||
@@ -806,31 +841,39 @@ public class XML {
|
||||
sb.append('<');
|
||||
sb.append(key);
|
||||
sb.append('>');
|
||||
sb.append(toString(val, null, config));
|
||||
sb.append(toString(val, null, config, indentFactor, indent));
|
||||
sb.append("</");
|
||||
sb.append(key);
|
||||
sb.append('>');
|
||||
} else {
|
||||
sb.append(toString(val, key, config));
|
||||
sb.append(toString(val, key, config, indentFactor, indent));
|
||||
}
|
||||
}
|
||||
} else if ("".equals(value)) {
|
||||
sb.append(indent(indent));
|
||||
sb.append('<');
|
||||
sb.append(key);
|
||||
sb.append("/>");
|
||||
if(indentFactor > 0){
|
||||
sb.append("\n");
|
||||
}
|
||||
|
||||
// Emit a new tag <k>
|
||||
|
||||
} else {
|
||||
sb.append(toString(value, key, config));
|
||||
sb.append(toString(value, key, config, indentFactor, indent));
|
||||
}
|
||||
}
|
||||
if (tagName != null) {
|
||||
|
||||
// Emit the </tagName> close tag
|
||||
sb.append(indent(indent - indentFactor));
|
||||
sb.append("</");
|
||||
sb.append(tagName);
|
||||
sb.append('>');
|
||||
if(indentFactor > 0){
|
||||
sb.append("\n");
|
||||
}
|
||||
}
|
||||
return sb.toString();
|
||||
|
||||
@@ -849,15 +892,85 @@ public class XML {
|
||||
// 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));
|
||||
sb.append(toString(val, tagName == null ? "array" : tagName, config, indentFactor, indent));
|
||||
}
|
||||
return sb.toString();
|
||||
}
|
||||
|
||||
string = (object == null) ? "null" : escape(object.toString());
|
||||
return (tagName == null) ? "\"" + string + "\""
|
||||
: (string.length() == 0) ? "<" + tagName + "/>" : "<" + tagName
|
||||
+ ">" + string + "</" + tagName + ">";
|
||||
|
||||
string = (object == null) ? "null" : escape(object.toString());
|
||||
|
||||
if(tagName == null){
|
||||
return indent(indent) + "\"" + string + "\"" + ((indentFactor > 0) ? "\n" : "");
|
||||
} else if(string.length() == 0){
|
||||
return indent(indent) + "<" + tagName + "/>" + ((indentFactor > 0) ? "\n" : "");
|
||||
} else {
|
||||
return indent(indent) + "<" + tagName
|
||||
+ ">" + string + "</" + tagName + ">" + ((indentFactor > 0) ? "\n" : "");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert a JSONObject into a well-formed, pretty printed element-normal XML string.
|
||||
*
|
||||
* @param object
|
||||
* A JSONObject.
|
||||
* @param indentFactor
|
||||
* The number of spaces to add to each level of indentation.
|
||||
* @return A string.
|
||||
* @throws JSONException Thrown if there is an error parsing the string
|
||||
*/
|
||||
public static String toString(Object object, int indentFactor){
|
||||
return toString(object, null, XMLParserConfiguration.ORIGINAL, indentFactor);
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert a JSONObject into a well-formed, pretty printed element-normal XML string.
|
||||
*
|
||||
* @param object
|
||||
* A JSONObject.
|
||||
* @param tagName
|
||||
* The optional name of the enclosing tag.
|
||||
* @param indentFactor
|
||||
* The number of spaces to add to each level of indentation.
|
||||
* @return A string.
|
||||
* @throws JSONException Thrown if there is an error parsing the string
|
||||
*/
|
||||
public static String toString(final Object object, final String tagName, int indentFactor) {
|
||||
return toString(object, tagName, XMLParserConfiguration.ORIGINAL, indentFactor);
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert a JSONObject into a well-formed, pretty printed element-normal XML string.
|
||||
*
|
||||
* @param object
|
||||
* A JSONObject.
|
||||
* @param tagName
|
||||
* The optional name of the enclosing tag.
|
||||
* @param config
|
||||
* Configuration that can control output to XML.
|
||||
* @param indentFactor
|
||||
* The number of spaces to add to each level of indentation.
|
||||
* @return A string.
|
||||
* @throws JSONException Thrown if there is an error parsing the string
|
||||
*/
|
||||
public static String toString(final Object object, final String tagName, final XMLParserConfiguration config, int indentFactor)
|
||||
throws JSONException {
|
||||
return toString(object, tagName, config, indentFactor, 0);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return a String consisting of a number of space characters specified by indent
|
||||
*
|
||||
* @param indent
|
||||
* The number of spaces to be appended to the String.
|
||||
* @return
|
||||
*/
|
||||
private static final String indent(int indent) {
|
||||
StringBuilder sb = new StringBuilder();
|
||||
for (int i = 0; i < indent; i++) {
|
||||
sb.append(' ');
|
||||
}
|
||||
return sb.toString();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -15,7 +15,13 @@ import java.util.Set;
|
||||
* @author AylwardJ
|
||||
*/
|
||||
@SuppressWarnings({""})
|
||||
public class XMLParserConfiguration {
|
||||
public class XMLParserConfiguration extends ParserConfiguration {
|
||||
|
||||
/**
|
||||
* The default maximum nesting depth when parsing a XML document to JSON.
|
||||
*/
|
||||
// public static final int DEFAULT_MAXIMUM_NESTING_DEPTH = 512; // We could override
|
||||
|
||||
/** Original Configuration of the XML Parser. */
|
||||
public static final XMLParserConfiguration ORIGINAL
|
||||
= new XMLParserConfiguration();
|
||||
@@ -23,19 +29,13 @@ public class XMLParserConfiguration {
|
||||
public static final XMLParserConfiguration KEEP_STRINGS
|
||||
= new XMLParserConfiguration().withKeepStrings(true);
|
||||
|
||||
/**
|
||||
* When parsing the XML into JSON, specifies if values should be kept as strings (<code>true</code>), or if
|
||||
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||
*/
|
||||
private boolean keepStrings;
|
||||
|
||||
/**
|
||||
* The name of the key in a JSON Object that indicates a CDATA section. Historically this has
|
||||
* been the value "content" but can be changed. Use <code>null</code> to indicate no CDATA
|
||||
* processing.
|
||||
*/
|
||||
private String cDataTagName;
|
||||
|
||||
|
||||
/**
|
||||
* When parsing the XML into JSON, specifies if values with attribute xsi:nil="true"
|
||||
* should be kept as attribute(<code>false</code>), or they should be converted to
|
||||
@@ -59,7 +59,7 @@ public class XMLParserConfiguration {
|
||||
* values), and the CDATA Tag Name is "content".
|
||||
*/
|
||||
public XMLParserConfiguration () {
|
||||
this.keepStrings = false;
|
||||
super();
|
||||
this.cDataTagName = "content";
|
||||
this.convertNilAttributeToNull = false;
|
||||
this.xsiTypeMap = Collections.emptyMap();
|
||||
@@ -106,7 +106,7 @@ public class XMLParserConfiguration {
|
||||
*/
|
||||
@Deprecated
|
||||
public XMLParserConfiguration (final boolean keepStrings, final String cDataTagName) {
|
||||
this.keepStrings = keepStrings;
|
||||
super(keepStrings, DEFAULT_MAXIMUM_NESTING_DEPTH);
|
||||
this.cDataTagName = cDataTagName;
|
||||
this.convertNilAttributeToNull = false;
|
||||
}
|
||||
@@ -125,7 +125,7 @@ public class XMLParserConfiguration {
|
||||
*/
|
||||
@Deprecated
|
||||
public XMLParserConfiguration (final boolean keepStrings, final String cDataTagName, final boolean convertNilAttributeToNull) {
|
||||
this.keepStrings = keepStrings;
|
||||
super(keepStrings, DEFAULT_MAXIMUM_NESTING_DEPTH);
|
||||
this.cDataTagName = cDataTagName;
|
||||
this.convertNilAttributeToNull = convertNilAttributeToNull;
|
||||
}
|
||||
@@ -140,11 +140,13 @@ public class XMLParserConfiguration {
|
||||
* <code>false</code> to parse values with attribute xsi:nil="true" as {"xsi:nil":true}.
|
||||
* @param xsiTypeMap <code>new HashMap<String, XMLXsiTypeConverter<?>>()</code> to parse values with attribute
|
||||
* xsi:type="integer" as integer, xsi:type="string" as string
|
||||
* @param forceList <code>new HashSet<String>()</code> to parse the provided tags' values as arrays
|
||||
* @param forceList <code>new HashSet<String>()</code> to parse the provided tags' values as arrays
|
||||
* @param maxNestingDepth <code>int</code> to limit the nesting depth
|
||||
*/
|
||||
private XMLParserConfiguration (final boolean keepStrings, final String cDataTagName,
|
||||
final boolean convertNilAttributeToNull, final Map<String, XMLXsiTypeConverter<?>> xsiTypeMap, final Set<String> forceList ) {
|
||||
this.keepStrings = keepStrings;
|
||||
final boolean convertNilAttributeToNull, final Map<String, XMLXsiTypeConverter<?>> xsiTypeMap, final Set<String> forceList,
|
||||
final int maxNestingDepth) {
|
||||
super(keepStrings, maxNestingDepth);
|
||||
this.cDataTagName = cDataTagName;
|
||||
this.convertNilAttributeToNull = convertNilAttributeToNull;
|
||||
this.xsiTypeMap = Collections.unmodifiableMap(xsiTypeMap);
|
||||
@@ -166,40 +168,30 @@ public class XMLParserConfiguration {
|
||||
this.cDataTagName,
|
||||
this.convertNilAttributeToNull,
|
||||
this.xsiTypeMap,
|
||||
this.forceList
|
||||
this.forceList,
|
||||
this.maxNestingDepth
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* When parsing the XML into JSON, specifies if values should be kept as strings (<code>true</code>), or if
|
||||
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||
*
|
||||
* @return The <code>keepStrings</code> configuration value.
|
||||
*/
|
||||
public boolean isKeepStrings() {
|
||||
return this.keepStrings;
|
||||
}
|
||||
|
||||
/**
|
||||
* When parsing the XML into JSON, specifies if values should be kept as strings (<code>true</code>), or if
|
||||
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||
*
|
||||
*
|
||||
* @param newVal
|
||||
* new value to use for the <code>keepStrings</code> configuration option.
|
||||
*
|
||||
*
|
||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||
*/
|
||||
@Override
|
||||
public XMLParserConfiguration withKeepStrings(final boolean newVal) {
|
||||
XMLParserConfiguration newConfig = this.clone();
|
||||
newConfig.keepStrings = newVal;
|
||||
return newConfig;
|
||||
return super.withKeepStrings(newVal);
|
||||
}
|
||||
|
||||
/**
|
||||
* 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.
|
||||
*
|
||||
*
|
||||
* @return The <code>cDataTagName</code> configuration value.
|
||||
*/
|
||||
public String getcDataTagName() {
|
||||
@@ -210,10 +202,10 @@ public class XMLParserConfiguration {
|
||||
* 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.
|
||||
*
|
||||
*
|
||||
* @param newVal
|
||||
* new value to use for the <code>cDataTagName</code> configuration option.
|
||||
*
|
||||
*
|
||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||
*/
|
||||
public XMLParserConfiguration withcDataTagName(final String newVal) {
|
||||
@@ -226,7 +218,7 @@ public class XMLParserConfiguration {
|
||||
* When parsing the XML into JSON, specifies if values with attribute xsi:nil="true"
|
||||
* should be kept as attribute(<code>false</code>), or they should be converted to
|
||||
* <code>null</code>(<code>true</code>)
|
||||
*
|
||||
*
|
||||
* @return The <code>convertNilAttributeToNull</code> configuration value.
|
||||
*/
|
||||
public boolean isConvertNilAttributeToNull() {
|
||||
@@ -237,10 +229,10 @@ public class XMLParserConfiguration {
|
||||
* When parsing the XML into JSON, specifies if values with attribute xsi:nil="true"
|
||||
* should be kept as attribute(<code>false</code>), or they should be converted to
|
||||
* <code>null</code>(<code>true</code>)
|
||||
*
|
||||
*
|
||||
* @param newVal
|
||||
* new value to use for the <code>convertNilAttributeToNull</code> configuration option.
|
||||
*
|
||||
*
|
||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||
*/
|
||||
public XMLParserConfiguration withConvertNilAttributeToNull(final boolean newVal) {
|
||||
@@ -278,7 +270,7 @@ public class XMLParserConfiguration {
|
||||
|
||||
/**
|
||||
* When parsing the XML into JSON, specifies that tags that will be converted to arrays
|
||||
* in this configuration {@code Set<String>} to parse the provided tags' values as arrays
|
||||
* in this configuration {@code Set<String>} to parse the provided tags' values as arrays
|
||||
* @return <code>forceList</code> unmodifiable configuration set.
|
||||
*/
|
||||
public Set<String> getForceList() {
|
||||
@@ -287,8 +279,8 @@ public class XMLParserConfiguration {
|
||||
|
||||
/**
|
||||
* When parsing the XML into JSON, specifies that tags that will be converted to arrays
|
||||
* in this configuration {@code Set<String>} to parse the provided tags' values as arrays
|
||||
* @param forceList {@code new HashSet<String>()} to parse the provided tags' values as arrays
|
||||
* in this configuration {@code Set<String>} to parse the provided tags' values as arrays
|
||||
* @param forceList {@code new HashSet<String>()} to parse the provided tags' values as arrays
|
||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||
*/
|
||||
public XMLParserConfiguration withForceList(final Set<String> forceList) {
|
||||
@@ -297,4 +289,18 @@ public class XMLParserConfiguration {
|
||||
newConfig.forceList = Collections.unmodifiableSet(cloneForceList);
|
||||
return newConfig;
|
||||
}
|
||||
|
||||
/**
|
||||
* Defines the maximum nesting depth that the parser will descend before throwing an exception
|
||||
* when parsing the XML into JSON. The default max nesting depth is 512, which means the parser
|
||||
* will throw a JsonException if the maximum depth is reached.
|
||||
* Using any negative value as a parameter is equivalent to setting no limit to the nesting depth,
|
||||
* which means the parses will go as deep as the maximum call stack size allows.
|
||||
* @param maxNestingDepth the maximum nesting depth allowed to the XML parser
|
||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||
*/
|
||||
@Override
|
||||
public XMLParserConfiguration withMaxNestingDepth(int maxNestingDepth) {
|
||||
return super.withMaxNestingDepth(maxNestingDepth);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -6,7 +6,6 @@ Public Domain.
|
||||
|
||||
import static org.junit.Assert.assertEquals;
|
||||
import static org.junit.Assert.assertFalse;
|
||||
import static org.junit.Assert.assertNotEquals;
|
||||
import static org.junit.Assert.assertNotNull;
|
||||
import static org.junit.Assert.assertNull;
|
||||
import static org.junit.Assert.assertTrue;
|
||||
|
||||
@@ -11,19 +11,19 @@ import org.junit.Test;
|
||||
|
||||
/**
|
||||
* Tests for org.json.JSONML.java
|
||||
*
|
||||
*
|
||||
* Certain inputs are expected to result in exceptions. These tests are
|
||||
* executed first. JSONML provides an API to:
|
||||
* Convert an XML string into a JSONArray or a JSONObject.
|
||||
* Convert an XML string into a JSONArray or a JSONObject.
|
||||
* Convert a JSONArray or JSONObject into an XML string.
|
||||
* Both fromstring and tostring operations operations should be symmetrical
|
||||
* within the limits of JSONML.
|
||||
* within the limits of JSONML.
|
||||
* It should be possible to perform the following operations, which should
|
||||
* result in the original string being recovered, within the limits of the
|
||||
* underlying classes:
|
||||
* Convert a string -> JSONArray -> string -> JSONObject -> string
|
||||
* Convert a string -> JSONObject -> string -> JSONArray -> string
|
||||
*
|
||||
*
|
||||
*/
|
||||
public class JSONMLTest {
|
||||
|
||||
@@ -56,7 +56,7 @@ public class JSONMLTest {
|
||||
|
||||
/**
|
||||
* Attempts to call JSONML.toString() with a null JSONArray.
|
||||
* Expects a NullPointerException.
|
||||
* Expects a NullPointerException.
|
||||
*/
|
||||
@Test(expected=NullPointerException.class)
|
||||
public void nullJSONXMLException() {
|
||||
@@ -69,7 +69,7 @@ public class JSONMLTest {
|
||||
|
||||
/**
|
||||
* Attempts to call JSONML.toString() with a null JSONArray.
|
||||
* Expects a JSONException.
|
||||
* Expects a JSONException.
|
||||
*/
|
||||
@Test
|
||||
public void emptyJSONXMLException() {
|
||||
@@ -125,7 +125,7 @@ public class JSONMLTest {
|
||||
"[\"addresses\","+
|
||||
"{\"xsi:noNamespaceSchemaLocation\":\"test.xsd\","+
|
||||
"\"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\"},"+
|
||||
// this array has no name
|
||||
// this array has no name
|
||||
"["+
|
||||
"[\"name\"],"+
|
||||
"[\"nocontent\"],"+
|
||||
@@ -180,7 +180,7 @@ public class JSONMLTest {
|
||||
}
|
||||
|
||||
/**
|
||||
* Attempts to transform a malformed XML document
|
||||
* Attempts to transform a malformed XML document
|
||||
* (element tag has a frontslash) to a JSONArray.\
|
||||
* Expects a JSONException
|
||||
*/
|
||||
@@ -191,7 +191,7 @@ public class JSONMLTest {
|
||||
* In this case, the XML is invalid because the 'name' element
|
||||
* contains an invalid frontslash.
|
||||
*/
|
||||
String xmlStr =
|
||||
String xmlStr =
|
||||
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||
@@ -216,7 +216,7 @@ public class JSONMLTest {
|
||||
*/
|
||||
@Test
|
||||
public void invalidBangInTagException() {
|
||||
String xmlStr =
|
||||
String xmlStr =
|
||||
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||
@@ -246,7 +246,7 @@ public class JSONMLTest {
|
||||
* In this case, the XML is invalid because an element
|
||||
* starts with '!' and has no closing tag
|
||||
*/
|
||||
String xmlStr =
|
||||
String xmlStr =
|
||||
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||
@@ -276,7 +276,7 @@ public class JSONMLTest {
|
||||
* In this case, the XML is invalid because an element
|
||||
* has no closing '>'.
|
||||
*/
|
||||
String xmlStr =
|
||||
String xmlStr =
|
||||
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||
@@ -306,7 +306,7 @@ public class JSONMLTest {
|
||||
* In this case, the XML is invalid because an element
|
||||
* has no name after the closing tag '</'.
|
||||
*/
|
||||
String xmlStr =
|
||||
String xmlStr =
|
||||
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||
@@ -336,7 +336,7 @@ public class JSONMLTest {
|
||||
* In this case, the XML is invalid because an element
|
||||
* has '>' after the closing tag '</' and name.
|
||||
*/
|
||||
String xmlStr =
|
||||
String xmlStr =
|
||||
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||
" xsi:noNamespaceSchemaLocation=\"test.xsd\">\n"+
|
||||
@@ -364,9 +364,9 @@ public class JSONMLTest {
|
||||
/**
|
||||
* xmlStr contains XML text which is transformed into a JSONArray.
|
||||
* In this case, the XML is invalid because an element
|
||||
* does not have a complete CDATA string.
|
||||
* does not have a complete CDATA string.
|
||||
*/
|
||||
String xmlStr =
|
||||
String xmlStr =
|
||||
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||
@@ -388,7 +388,7 @@ public class JSONMLTest {
|
||||
/**
|
||||
* Convert an XML document into a JSONArray, then use JSONML.toString()
|
||||
* to convert it into a string. This string is then converted back into
|
||||
* a JSONArray. Both JSONArrays are compared against a control to
|
||||
* a JSONArray. Both JSONArrays are compared against a control to
|
||||
* confirm the contents.
|
||||
*/
|
||||
@Test
|
||||
@@ -405,7 +405,7 @@ public class JSONMLTest {
|
||||
* which is used to create a final JSONArray, which is also compared
|
||||
* against the expected JSONArray.
|
||||
*/
|
||||
String xmlStr =
|
||||
String xmlStr =
|
||||
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||
"xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||
@@ -414,7 +414,7 @@ public class JSONMLTest {
|
||||
"<nocontent/>>\n"+
|
||||
"</address>\n"+
|
||||
"</addresses>";
|
||||
String expectedStr =
|
||||
String expectedStr =
|
||||
"[\"addresses\","+
|
||||
"{\"xsi:noNamespaceSchemaLocation\":\"test.xsd\","+
|
||||
"\"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\"},"+
|
||||
@@ -434,12 +434,12 @@ public class JSONMLTest {
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert an XML document into a JSONObject. Use JSONML.toString() to
|
||||
* Convert an XML document into a JSONObject. Use JSONML.toString() to
|
||||
* convert it back into a string, and then re-convert it into a JSONObject.
|
||||
* Both JSONObjects are compared against a control JSONObject to confirm
|
||||
* the contents.
|
||||
* <p>
|
||||
* Next convert the XML document into a JSONArray. Use JSONML.toString() to
|
||||
* Next convert the XML document into a JSONArray. Use JSONML.toString() to
|
||||
* convert it back into a string, and then re-convert it into a JSONArray.
|
||||
* Both JSONArrays are compared against a control JSONArray to confirm
|
||||
* the contents.
|
||||
@@ -452,23 +452,23 @@ public class JSONMLTest {
|
||||
/**
|
||||
* xmlStr contains XML text which is transformed into a JSONObject,
|
||||
* restored to XML, transformed into a JSONArray, and then restored
|
||||
* to XML again. Both JSONObject and JSONArray should contain the same
|
||||
* to XML again. Both JSONObject and JSONArray should contain the same
|
||||
* information and should produce the same XML, allowing for non-ordered
|
||||
* attributes.
|
||||
*
|
||||
*
|
||||
* Transformation to JSONObject:
|
||||
* The elementName is stored as a string where key="tagName"
|
||||
* Attributes are simply stored as key/value pairs
|
||||
* If the element has either content or child elements, they are stored
|
||||
* in a jsonArray with key="childNodes".
|
||||
*
|
||||
*
|
||||
* Transformation to JSONArray:
|
||||
* 1st entry = elementname
|
||||
* 2nd entry = attributes object (if present)
|
||||
* 3rd entry = content (if present)
|
||||
* 4th entry = child element JSONArrays (if present)
|
||||
*/
|
||||
String xmlStr =
|
||||
String xmlStr =
|
||||
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||
"xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||
@@ -585,7 +585,7 @@ public class JSONMLTest {
|
||||
"\"tagName\":\"addresses\""+
|
||||
"}";
|
||||
|
||||
String expectedJSONArrayStr =
|
||||
String expectedJSONArrayStr =
|
||||
"["+
|
||||
"\"addresses\","+
|
||||
"{"+
|
||||
@@ -645,12 +645,12 @@ public class JSONMLTest {
|
||||
JSONObject finalJsonObject = JSONML.toJSONObject(jsonObjectXmlToStr);
|
||||
Util.compareActualVsExpectedJsonObjects(finalJsonObject, expectedJsonObject);
|
||||
|
||||
// create a JSON array from the original string and make sure it
|
||||
// create a JSON array from the original string and make sure it
|
||||
// looks as expected
|
||||
JSONArray jsonArray = JSONML.toJSONArray(xmlStr);
|
||||
JSONArray expectedJsonArray = new JSONArray(expectedJSONArrayStr);
|
||||
Util.compareActualVsExpectedJsonArrays(jsonArray,expectedJsonArray);
|
||||
|
||||
|
||||
// restore the XML, then make another JSONArray and make sure it
|
||||
// looks as expected
|
||||
String jsonArrayXmlToStr = JSONML.toString(jsonArray);
|
||||
@@ -668,14 +668,14 @@ public class JSONMLTest {
|
||||
* Convert an XML document which contains embedded comments into
|
||||
* a JSONArray. Use JSONML.toString() to turn it into a string, then
|
||||
* reconvert it into a JSONArray. Compare both JSONArrays to a control
|
||||
* JSONArray to confirm the contents.
|
||||
* JSONArray to confirm the contents.
|
||||
* <p>
|
||||
* This test shows how XML comments are handled.
|
||||
*/
|
||||
@Test
|
||||
public void commentsInXML() {
|
||||
|
||||
String xmlStr =
|
||||
String xmlStr =
|
||||
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||
"<!-- this is a comment -->\n"+
|
||||
"<addresses>\n"+
|
||||
@@ -734,7 +734,7 @@ public class JSONMLTest {
|
||||
final String expectedJsonString = "[\"root\",[\"id\",\"01\"],[\"id\",\"1\"],[\"id\",\"00\"],[\"id\",\"0\"],[\"item\",{\"id\":\"01\"}],[\"title\",\"True\"]]";
|
||||
final JSONArray json = JSONML.toJSONArray(originalXml,true);
|
||||
assertEquals(expectedJsonString, json.toString());
|
||||
|
||||
|
||||
final String reverseXml = JSONML.toString(json);
|
||||
assertEquals(originalXml, reverseXml);
|
||||
}
|
||||
@@ -749,7 +749,7 @@ public class JSONMLTest {
|
||||
final String revertedXml = JSONML.toString(jsonArray);
|
||||
assertEquals(revertedXml, originalXml);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* JSON string cannot be reverted to original xml. See test result in
|
||||
* comment below.
|
||||
@@ -770,7 +770,7 @@ public class JSONMLTest {
|
||||
// 1. Our XML parser does not handle generic HTML entities, only valid XML entities. Hence
|
||||
// or other HTML specific entities would fail on reversability
|
||||
// 2. Our JSON implementation for storing the XML attributes uses the standard unordered map.
|
||||
// This means that <tag attr1="v1" attr2="v2" /> can not be reversed reliably.
|
||||
// This means that <tag attr1="v1" attr2="v2" /> can not be reversed reliably.
|
||||
//
|
||||
// /**
|
||||
// * Test texts taken from jsonml.org. Currently our implementation FAILS this conversion but shouldn't.
|
||||
@@ -783,13 +783,13 @@ public class JSONMLTest {
|
||||
// final String expectedJsonString = "[\"table\",{\"class\" : \"MyTable\",\"style\" : \"background-color:yellow\"},[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#550758\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:red\"},\"Example text here\"]],[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#993101\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:green\"},\"127624015\"]],[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#E33D87\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:blue\"},\"\u00A0\",[\"span\",{ \"style\" : \"background-color:maroon\" },\"\u00A9\"],\"\u00A0\"]]]";
|
||||
// final JSONArray json = JSONML.toJSONArray(originalXml,true);
|
||||
// final String actualJsonString = json.toString();
|
||||
//
|
||||
//
|
||||
// final String reverseXml = JSONML.toString(json);
|
||||
// assertNotEquals(originalXml, reverseXml);
|
||||
//
|
||||
// assertNotEquals(expectedJsonString, actualJsonString);
|
||||
// }
|
||||
//
|
||||
//
|
||||
// /**
|
||||
// * Test texts taken from jsonml.org but modified to have XML entities only.
|
||||
// */
|
||||
@@ -799,15 +799,15 @@ public class JSONMLTest {
|
||||
// final String expectedJsonString = "[\"table\",{\"class\" : \"MyTable\",\"style\" : \"background-color:yellow\"},[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#550758\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:red\"},\"Example text here\"]],[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#993101\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:green\"},\"127624015\"]],[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#E33D87\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:blue\"},\"&\",[\"span\",{ \"style\" : \"background-color:maroon\" },\">\"],\"<\"]]]";
|
||||
// final JSONArray jsonML = JSONML.toJSONArray(originalXml,true);
|
||||
// final String actualJsonString = jsonML.toString();
|
||||
//
|
||||
//
|
||||
// final String reverseXml = JSONML.toString(jsonML);
|
||||
// // currently not equal because the hashing of the attribute objects makes the attribute
|
||||
// // order not happen the same way twice
|
||||
// // order not happen the same way twice
|
||||
// assertEquals(originalXml, reverseXml);
|
||||
//
|
||||
// assertEquals(expectedJsonString, actualJsonString);
|
||||
// }
|
||||
|
||||
|
||||
@Test (timeout = 6000)
|
||||
public void testIssue484InfinteLoop1() {
|
||||
try {
|
||||
@@ -819,11 +819,11 @@ public class JSONMLTest {
|
||||
ex.getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Test (timeout = 6000)
|
||||
public void testIssue484InfinteLoop2() {
|
||||
try {
|
||||
String input = "??*\n" +
|
||||
String input = "??*\n" +
|
||||
"??|?CglR??`??>?w??PIlr??D?$?-?o??O?*??{OD?Y??`2a????NM?bq?:O?>S$?J?B.gUK?m\b??zE???!v]???????c??????h???s???g???`?qbi??:Zl?)?}1^??k?0??:$V?$?Ovs(}J??????2;gQ????Tg?K?`?h%c?hmGA?<!C*?9?~?t?)??,zA???S}?Q??.q?j????]";
|
||||
JSONML.toJSONObject(input);
|
||||
fail("Exception expected for invalid JSON.");
|
||||
@@ -833,4 +833,157 @@ public class JSONMLTest {
|
||||
ex.getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testToJSONArrayMaxNestingDepthOf42IsRespected() {
|
||||
final String wayTooLongMalformedXML = new String(new char[6000]).replace("\0", "<a>");
|
||||
|
||||
final int maxNestingDepth = 42;
|
||||
|
||||
try {
|
||||
JSONML.toJSONArray(wayTooLongMalformedXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||
|
||||
fail("Expecting a JSONException");
|
||||
} catch (JSONException e) {
|
||||
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Test
|
||||
public void testToJSONArrayMaxNestingDepthIsRespectedWithValidXML() {
|
||||
final String perfectlyFineXML = "<Test>\n" +
|
||||
" <employee>\n" +
|
||||
" <name>sonoo</name>\n" +
|
||||
" <salary>56000</salary>\n" +
|
||||
" <married>true</married>\n" +
|
||||
" </employee>\n" +
|
||||
"</Test>\n";
|
||||
|
||||
final int maxNestingDepth = 1;
|
||||
|
||||
try {
|
||||
JSONML.toJSONArray(perfectlyFineXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||
|
||||
fail("Expecting a JSONException");
|
||||
} catch (JSONException e) {
|
||||
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testToJSONArrayMaxNestingDepthWithValidFittingXML() {
|
||||
final String perfectlyFineXML = "<Test>\n" +
|
||||
" <employee>\n" +
|
||||
" <name>sonoo</name>\n" +
|
||||
" <salary>56000</salary>\n" +
|
||||
" <married>true</married>\n" +
|
||||
" </employee>\n" +
|
||||
"</Test>\n";
|
||||
|
||||
final int maxNestingDepth = 3;
|
||||
|
||||
try {
|
||||
JSONML.toJSONArray(perfectlyFineXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||
} catch (JSONException e) {
|
||||
e.printStackTrace();
|
||||
fail("XML document should be parsed as its maximum depth fits the maxNestingDepth " +
|
||||
"parameter of the JSONMLParserConfiguration used");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Test
|
||||
public void testToJSONObjectMaxDefaultNestingDepthIsRespected() {
|
||||
final String wayTooLongMalformedXML = new String(new char[6000]).replace("\0", "<a>");
|
||||
|
||||
try {
|
||||
JSONML.toJSONObject(wayTooLongMalformedXML, JSONMLParserConfiguration.ORIGINAL);
|
||||
|
||||
fail("Expecting a JSONException");
|
||||
} catch (JSONException e) {
|
||||
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||
e.getMessage().startsWith("Maximum nesting depth of " + JSONMLParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH));
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testToJSONObjectUnlimitedNestingDepthIsPossible() {
|
||||
int actualDepth = JSONMLParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH +10;
|
||||
final String deeperThanDefaultMax = new String(new char[actualDepth]).replace("\0", "<a>") +
|
||||
"value" +
|
||||
new String(new char[actualDepth]).replace("\0", "</a>");
|
||||
|
||||
try {
|
||||
JSONML.toJSONObject(deeperThanDefaultMax, JSONMLParserConfiguration.ORIGINAL
|
||||
.withMaxNestingDepth(JSONMLParserConfiguration.UNDEFINED_MAXIMUM_NESTING_DEPTH));
|
||||
} catch (JSONException e) {
|
||||
e.printStackTrace();
|
||||
fail("XML document should be parsed beyond the default maximum depth if maxNestingDepth " +
|
||||
"parameter is set to -1 in JSONMLParserConfiguration");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Test
|
||||
public void testToJSONObjectMaxNestingDepthOf42IsRespected() {
|
||||
final String wayTooLongMalformedXML = new String(new char[6000]).replace("\0", "<a>");
|
||||
|
||||
final int maxNestingDepth = 42;
|
||||
|
||||
try {
|
||||
JSONML.toJSONObject(wayTooLongMalformedXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||
|
||||
fail("Expecting a JSONException");
|
||||
} catch (JSONException e) {
|
||||
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testToJSONObjectMaxNestingDepthIsRespectedWithValidXML() {
|
||||
final String perfectlyFineXML = "<Test>\n" +
|
||||
" <employee>\n" +
|
||||
" <name>sonoo</name>\n" +
|
||||
" <salary>56000</salary>\n" +
|
||||
" <married>true</married>\n" +
|
||||
" </employee>\n" +
|
||||
"</Test>\n";
|
||||
|
||||
final int maxNestingDepth = 1;
|
||||
|
||||
try {
|
||||
JSONML.toJSONObject(perfectlyFineXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||
|
||||
fail("Expecting a JSONException");
|
||||
} catch (JSONException e) {
|
||||
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testToJSONObjectMaxNestingDepthWithValidFittingXML() {
|
||||
final String perfectlyFineXML = "<Test>\n" +
|
||||
" <employee>\n" +
|
||||
" <name>sonoo</name>\n" +
|
||||
" <salary>56000</salary>\n" +
|
||||
" <married>true</married>\n" +
|
||||
" </employee>\n" +
|
||||
"</Test>\n";
|
||||
|
||||
final int maxNestingDepth = 3;
|
||||
|
||||
try {
|
||||
JSONML.toJSONObject(perfectlyFineXML, JSONMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||
} catch (JSONException e) {
|
||||
e.printStackTrace();
|
||||
fail("XML document should be parsed as its maximum depth fits the maxNestingDepth " +
|
||||
"parameter of the JSONMLParserConfiguration used");
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
126
src/test/java/org/json/junit/JSONObjectNumberTest.java
Normal file
126
src/test/java/org/json/junit/JSONObjectNumberTest.java
Normal file
@@ -0,0 +1,126 @@
|
||||
package org.json.junit;
|
||||
|
||||
import static org.junit.Assert.assertEquals;
|
||||
import static org.junit.Assert.assertTrue;
|
||||
|
||||
import java.math.BigDecimal;
|
||||
import java.math.BigInteger;
|
||||
import java.util.Arrays;
|
||||
import java.util.Collection;
|
||||
|
||||
import org.json.JSONObject;
|
||||
import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
import org.junit.runner.RunWith;
|
||||
import org.junit.runners.Parameterized;
|
||||
import org.junit.runners.Parameterized.Parameters;
|
||||
|
||||
@RunWith(value = Parameterized.class)
|
||||
public class JSONObjectNumberTest {
|
||||
private final String objectString;
|
||||
private Integer value = 50;
|
||||
|
||||
@Parameters(name = "{index}: {0}")
|
||||
public static Collection<Object[]> data() {
|
||||
return Arrays.asList(new Object[][]{
|
||||
{"{value:50}", 1},
|
||||
{"{value:50.0}", 1},
|
||||
{"{value:5e1}", 1},
|
||||
{"{value:5E1}", 1},
|
||||
{"{value:5e1}", 1},
|
||||
{"{value:'50'}", 1},
|
||||
{"{value:-50}", -1},
|
||||
{"{value:-50.0}", -1},
|
||||
{"{value:-5e1}", -1},
|
||||
{"{value:-5E1}", -1},
|
||||
{"{value:-5e1}", -1},
|
||||
{"{value:'-50'}", -1}
|
||||
// JSON does not support octal or hex numbers;
|
||||
// see https://stackoverflow.com/a/52671839/6323312
|
||||
// "{value:062}", // octal 50
|
||||
// "{value:0x32}" // hex 50
|
||||
});
|
||||
}
|
||||
|
||||
public JSONObjectNumberTest(String objectString, int resultIsNegative) {
|
||||
this.objectString = objectString;
|
||||
this.value *= resultIsNegative;
|
||||
}
|
||||
|
||||
private JSONObject object;
|
||||
|
||||
@Before
|
||||
public void setJsonObject() {
|
||||
object = new JSONObject(objectString);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testGetNumber() {
|
||||
assertEquals(value.intValue(), object.getNumber("value").intValue());
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testGetBigDecimal() {
|
||||
assertTrue(BigDecimal.valueOf(value).compareTo(object.getBigDecimal("value")) == 0);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testGetBigInteger() {
|
||||
assertEquals(BigInteger.valueOf(value), object.getBigInteger("value"));
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testGetFloat() {
|
||||
assertEquals(value.floatValue(), object.getFloat("value"), 0.0f);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testGetDouble() {
|
||||
assertEquals(value.doubleValue(), object.getDouble("value"), 0.0d);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testGetInt() {
|
||||
assertEquals(value.intValue(), object.getInt("value"));
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testGetLong() {
|
||||
assertEquals(value.longValue(), object.getLong("value"));
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOptNumber() {
|
||||
assertEquals(value.intValue(), object.optNumber("value").intValue());
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOptBigDecimal() {
|
||||
assertTrue(BigDecimal.valueOf(value).compareTo(object.optBigDecimal("value", null)) == 0);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOptBigInteger() {
|
||||
assertEquals(BigInteger.valueOf(value), object.optBigInteger("value", null));
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOptFloat() {
|
||||
assertEquals(value.floatValue(), object.optFloat("value"), 0.0f);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOptDouble() {
|
||||
assertEquals(value.doubleValue(), object.optDouble("value"), 0.0d);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOptInt() {
|
||||
assertEquals(value.intValue(), object.optInt("value"));
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOptLong() {
|
||||
assertEquals(value.longValue(), object.optLong("value"));
|
||||
}
|
||||
}
|
||||
@@ -72,8 +72,10 @@ public class JSONPointerTest {
|
||||
|
||||
@Test
|
||||
public void queryByEmptyKeySubObject() {
|
||||
assertEquals( "{\"\":\"empty key of an object with an empty key\",\"subKey\":\"Some" +
|
||||
" other value\"}", query("/obj/").toString());
|
||||
JSONObject json = new JSONObject("{\"\":\"empty key of an object with an empty key\",\"subKey\":\"Some" +
|
||||
" other value\"}");
|
||||
JSONObject obj = (JSONObject) query("/obj/");
|
||||
assertTrue(json.similar(obj));
|
||||
}
|
||||
|
||||
@Test
|
||||
|
||||
@@ -313,4 +313,16 @@ public class JSONTokenerTest {
|
||||
assertEquals(0, t2.next());
|
||||
assertFalse(t2.more());
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testAutoClose(){
|
||||
Reader reader = new StringReader("some test string");
|
||||
try {
|
||||
JSONTokener tokener = new JSONTokener(reader);
|
||||
tokener.close();
|
||||
tokener.next();
|
||||
} catch (Exception exception){
|
||||
assertEquals("Stream closed", exception.getMessage());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1051,6 +1051,29 @@ public class XMLConfigurationTest {
|
||||
|
||||
Util.compareActualVsExpectedJsonObjects(jsonObject, expetedJsonObject);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testMaxNestingDepthIsSet() {
|
||||
XMLParserConfiguration xmlParserConfiguration = XMLParserConfiguration.ORIGINAL;
|
||||
|
||||
assertEquals(xmlParserConfiguration.getMaxNestingDepth(), XMLParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH);
|
||||
|
||||
xmlParserConfiguration = xmlParserConfiguration.withMaxNestingDepth(42);
|
||||
|
||||
assertEquals(xmlParserConfiguration.getMaxNestingDepth(), 42);
|
||||
|
||||
xmlParserConfiguration = xmlParserConfiguration.withMaxNestingDepth(0);
|
||||
|
||||
assertEquals(xmlParserConfiguration.getMaxNestingDepth(), 0);
|
||||
|
||||
xmlParserConfiguration = xmlParserConfiguration.withMaxNestingDepth(-31415926);
|
||||
|
||||
assertEquals(xmlParserConfiguration.getMaxNestingDepth(), XMLParserConfiguration.UNDEFINED_MAXIMUM_NESTING_DEPTH);
|
||||
|
||||
xmlParserConfiguration = xmlParserConfiguration.withMaxNestingDepth(Integer.MIN_VALUE);
|
||||
|
||||
assertEquals(xmlParserConfiguration.getMaxNestingDepth(), XMLParserConfiguration.UNDEFINED_MAXIMUM_NESTING_DEPTH);
|
||||
}
|
||||
|
||||
/**
|
||||
* Convenience method, given an input string and expected result,
|
||||
|
||||
@@ -21,13 +21,7 @@ import java.io.StringReader;
|
||||
import java.util.HashMap;
|
||||
import java.util.Map;
|
||||
|
||||
import org.json.JSONArray;
|
||||
import org.json.JSONException;
|
||||
import org.json.JSONObject;
|
||||
import org.json.JSONTokener;
|
||||
import org.json.XML;
|
||||
import org.json.XMLParserConfiguration;
|
||||
import org.json.XMLXsiTypeConverter;
|
||||
import org.json.*;
|
||||
import org.junit.Rule;
|
||||
import org.junit.Test;
|
||||
import org.junit.rules.TemporaryFolder;
|
||||
@@ -1049,4 +1043,276 @@ public class XMLTest {
|
||||
fail("Expected to be unable to modify the config");
|
||||
} catch (Exception ignored) { }
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testIndentComplicatedJsonObject(){
|
||||
String str = "{\n" +
|
||||
" \"success\": true,\n" +
|
||||
" \"error\": null,\n" +
|
||||
" \"response\": [\n" +
|
||||
" {\n" +
|
||||
" \"timestamp\": 1664917200,\n" +
|
||||
" \"dateTimeISO\": \"2022-10-05T00:00:00+03:00\",\n" +
|
||||
" \"loc\": {\n" +
|
||||
" \"lat\": 39.91987,\n" +
|
||||
" \"long\": 32.85427\n" +
|
||||
" },\n" +
|
||||
" \"place\": {\n" +
|
||||
" \"name\": \"ankara\",\n" +
|
||||
" \"state\": \"an\",\n" +
|
||||
" \"country\": \"tr\"\n" +
|
||||
" },\n" +
|
||||
" \"profile\": {\n" +
|
||||
" \"tz\": \"Europe/Istanbul\"\n" +
|
||||
" },\n" +
|
||||
" \"sun\": {\n" +
|
||||
" \"rise\": 1664941721,\n" +
|
||||
" \"riseISO\": \"2022-10-05T06:48:41+03:00\",\n" +
|
||||
" \"set\": 1664983521,\n" +
|
||||
" \"setISO\": \"2022-10-05T18:25:21+03:00\",\n" +
|
||||
" \"transit\": 1664962621,\n" +
|
||||
" \"transitISO\": \"2022-10-05T12:37:01+03:00\",\n" +
|
||||
" \"midnightSun\": false,\n" +
|
||||
" \"polarNight\": false,\n" +
|
||||
" \"twilight\": {\n" +
|
||||
" \"civilBegin\": 1664940106,\n" +
|
||||
" \"civilBeginISO\": \"2022-10-05T06:21:46+03:00\",\n" +
|
||||
" \"civilEnd\": 1664985136,\n" +
|
||||
" \"civilEndISO\": \"2022-10-05T18:52:16+03:00\",\n" +
|
||||
" \"nauticalBegin\": 1664938227,\n" +
|
||||
" \"nauticalBeginISO\": \"2022-10-05T05:50:27+03:00\",\n" +
|
||||
" \"nauticalEnd\": 1664987015,\n" +
|
||||
" \"nauticalEndISO\": \"2022-10-05T19:23:35+03:00\",\n" +
|
||||
" \"astronomicalBegin\": 1664936337,\n" +
|
||||
" \"astronomicalBeginISO\": \"2022-10-05T05:18:57+03:00\",\n" +
|
||||
" \"astronomicalEnd\": 1664988905,\n" +
|
||||
" \"astronomicalEndISO\": \"2022-10-05T19:55:05+03:00\"\n" +
|
||||
" }\n" +
|
||||
" },\n" +
|
||||
" \"moon\": {\n" +
|
||||
" \"rise\": 1664976480,\n" +
|
||||
" \"riseISO\": \"2022-10-05T16:28:00+03:00\",\n" +
|
||||
" \"set\": 1664921520,\n" +
|
||||
" \"setISO\": \"2022-10-05T01:12:00+03:00\",\n" +
|
||||
" \"transit\": 1664994240,\n" +
|
||||
" \"transitISO\": \"2022-10-05T21:24:00+03:00\",\n" +
|
||||
" \"underfoot\": 1664949360,\n" +
|
||||
" \"underfootISO\": \"2022-10-05T08:56:00+03:00\",\n" +
|
||||
" \"phase\": {\n" +
|
||||
" \"phase\": 0.3186,\n" +
|
||||
" \"name\": \"waxing gibbous\",\n" +
|
||||
" \"illum\": 71,\n" +
|
||||
" \"age\": 9.41,\n" +
|
||||
" \"angle\": 0.55\n" +
|
||||
" }\n" +
|
||||
" }\n" +
|
||||
" }\n" +
|
||||
" ]\n" +
|
||||
"}" ;
|
||||
JSONObject jsonObject = new JSONObject(str);
|
||||
String actualIndentedXmlString = XML.toString(jsonObject, 1);
|
||||
JSONObject actualJsonObject = XML.toJSONObject(actualIndentedXmlString);
|
||||
String expected = "<success>true</success>\n" +
|
||||
"<response>\n" +
|
||||
" <dateTimeISO>2022-10-05T00:00:00+03:00</dateTimeISO>\n" +
|
||||
" <loc>\n" +
|
||||
" <lat>39.91987</lat>\n" +
|
||||
" <long>32.85427</long>\n" +
|
||||
" </loc>\n" +
|
||||
" <moon>\n" +
|
||||
" <phase>\n" +
|
||||
" <phase>0.3186</phase>\n" +
|
||||
" <name>waxing gibbous</name>\n" +
|
||||
" <angle>0.55</angle>\n" +
|
||||
" <illum>71</illum>\n" +
|
||||
" <age>9.41</age>\n" +
|
||||
" </phase>\n" +
|
||||
" <setISO>2022-10-05T01:12:00+03:00</setISO>\n" +
|
||||
" <underfoot>1664949360</underfoot>\n" +
|
||||
" <set>1664921520</set>\n" +
|
||||
" <transit>1664994240</transit>\n" +
|
||||
" <transitISO>2022-10-05T21:24:00+03:00</transitISO>\n" +
|
||||
" <riseISO>2022-10-05T16:28:00+03:00</riseISO>\n" +
|
||||
" <rise>1664976480</rise>\n" +
|
||||
" <underfootISO>2022-10-05T08:56:00+03:00</underfootISO>\n" +
|
||||
" </moon>\n" +
|
||||
" <profile>\n" +
|
||||
" <tz>Europe/Istanbul</tz>\n" +
|
||||
" </profile>\n" +
|
||||
" <place>\n" +
|
||||
" <country>tr</country>\n" +
|
||||
" <name>ankara</name>\n" +
|
||||
" <state>an</state>\n" +
|
||||
" </place>\n" +
|
||||
" <sun>\n" +
|
||||
" <setISO>2022-10-05T18:25:21+03:00</setISO>\n" +
|
||||
" <midnightSun>false</midnightSun>\n" +
|
||||
" <set>1664983521</set>\n" +
|
||||
" <transit>1664962621</transit>\n" +
|
||||
" <polarNight>false</polarNight>\n" +
|
||||
" <transitISO>2022-10-05T12:37:01+03:00</transitISO>\n" +
|
||||
" <riseISO>2022-10-05T06:48:41+03:00</riseISO>\n" +
|
||||
" <rise>1664941721</rise>\n" +
|
||||
" <twilight>\n" +
|
||||
" <civilEnd>1664985136</civilEnd>\n" +
|
||||
" <astronomicalBegin>1664936337</astronomicalBegin>\n" +
|
||||
" <astronomicalEnd>1664988905</astronomicalEnd>\n" +
|
||||
" <astronomicalBeginISO>2022-10-05T05:18:57+03:00</astronomicalBeginISO>\n" +
|
||||
" <civilBegin>1664940106</civilBegin>\n" +
|
||||
" <nauticalEndISO>2022-10-05T19:23:35+03:00</nauticalEndISO>\n" +
|
||||
" <astronomicalEndISO>2022-10-05T19:55:05+03:00</astronomicalEndISO>\n" +
|
||||
" <nauticalBegin>1664938227</nauticalBegin>\n" +
|
||||
" <nauticalEnd>1664987015</nauticalEnd>\n" +
|
||||
" <nauticalBeginISO>2022-10-05T05:50:27+03:00</nauticalBeginISO>\n" +
|
||||
" <civilBeginISO>2022-10-05T06:21:46+03:00</civilBeginISO>\n" +
|
||||
" <civilEndISO>2022-10-05T18:52:16+03:00</civilEndISO>\n" +
|
||||
" </twilight>\n" +
|
||||
" </sun>\n" +
|
||||
" <timestamp>1664917200</timestamp>\n" +
|
||||
"</response>\n" +
|
||||
"<error>null</error>\n";
|
||||
JSONObject expectedJsonObject = XML.toJSONObject(expected);
|
||||
assertTrue(expectedJsonObject.similar(actualJsonObject));
|
||||
|
||||
|
||||
}
|
||||
@Test
|
||||
public void testIndentSimpleJsonObject(){
|
||||
String str = "{ \"employee\": { \n" +
|
||||
" \"name\": \"sonoo\", \n" +
|
||||
" \"salary\": 56000, \n" +
|
||||
" \"married\": true \n" +
|
||||
" }}";
|
||||
JSONObject jsonObject = new JSONObject(str);
|
||||
String actual = XML.toString(jsonObject, "Test", 2);
|
||||
JSONObject actualJsonObject = XML.toJSONObject(actual);
|
||||
String expected = "<Test>\n" +
|
||||
" <employee>\n" +
|
||||
" <name>sonoo</name>\n" +
|
||||
" <salary>56000</salary>\n" +
|
||||
" <married>true</married>\n" +
|
||||
" </employee>\n" +
|
||||
"</Test>\n";
|
||||
JSONObject expectedJsonObject = XML.toJSONObject(expected);
|
||||
assertTrue(expectedJsonObject.similar(actualJsonObject));
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testIndentSimpleJsonArray(){
|
||||
String str = "[ \n" +
|
||||
" {\"name\":\"Ram\", \"email\":\"Ram@gmail.com\"}, \n" +
|
||||
" {\"name\":\"Bob\", \"email\":\"bob32@gmail.com\"} \n" +
|
||||
"] ";
|
||||
JSONArray jsonObject = new JSONArray(str);
|
||||
String actual = XML.toString(jsonObject, 2);
|
||||
JSONObject actualJsonObject = XML.toJSONObject(actual);
|
||||
String expected = "<array>\n" +
|
||||
" <name>Ram</name>\n" +
|
||||
" <email>Ram@gmail.com</email>\n" +
|
||||
"</array>\n" +
|
||||
"<array>\n" +
|
||||
" <name>Bob</name>\n" +
|
||||
" <email>bob32@gmail.com</email>\n" +
|
||||
"</array>\n";
|
||||
JSONObject expectedJsonObject = XML.toJSONObject(expected);
|
||||
assertTrue(expectedJsonObject.similar(actualJsonObject));
|
||||
|
||||
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testIndentComplicatedJsonObjectWithArrayAndWithConfig(){
|
||||
try {
|
||||
InputStream jsonStream = null;
|
||||
try {
|
||||
jsonStream = XMLTest.class.getClassLoader().getResourceAsStream("Issue593.json");
|
||||
final JSONObject object = new JSONObject(new JSONTokener(jsonStream));
|
||||
String actualString = XML.toString(object, null, XMLParserConfiguration.KEEP_STRINGS,2);
|
||||
InputStream xmlStream = null;
|
||||
try {
|
||||
xmlStream = XMLTest.class.getClassLoader().getResourceAsStream("Issue593.xml");
|
||||
int bufferSize = 1024;
|
||||
char[] buffer = new char[bufferSize];
|
||||
StringBuilder expected = new StringBuilder();
|
||||
Reader in = new InputStreamReader(xmlStream, "UTF-8");
|
||||
for (int numRead; (numRead = in.read(buffer, 0, buffer.length)) > 0; ) {
|
||||
expected.append(buffer, 0, numRead);
|
||||
}
|
||||
assertEquals(expected.toString(), actualString.replaceAll("\\n|\\r\\n", System.getProperty("line.separator")));
|
||||
} finally {
|
||||
if (xmlStream != null) {
|
||||
xmlStream.close();
|
||||
}
|
||||
}
|
||||
} finally {
|
||||
if (jsonStream != null) {
|
||||
jsonStream.close();
|
||||
}
|
||||
}
|
||||
} catch (IOException e) {
|
||||
fail("file writer error: " +e.getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testMaxNestingDepthOf42IsRespected() {
|
||||
final String wayTooLongMalformedXML = new String(new char[6000]).replace("\0", "<a>");
|
||||
|
||||
final int maxNestingDepth = 42;
|
||||
|
||||
try {
|
||||
XML.toJSONObject(wayTooLongMalformedXML, XMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||
|
||||
fail("Expecting a JSONException");
|
||||
} catch (JSONException e) {
|
||||
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testMaxNestingDepthIsRespectedWithValidXML() {
|
||||
final String perfectlyFineXML = "<Test>\n" +
|
||||
" <employee>\n" +
|
||||
" <name>sonoo</name>\n" +
|
||||
" <salary>56000</salary>\n" +
|
||||
" <married>true</married>\n" +
|
||||
" </employee>\n" +
|
||||
"</Test>\n";
|
||||
|
||||
final int maxNestingDepth = 1;
|
||||
|
||||
try {
|
||||
XML.toJSONObject(perfectlyFineXML, XMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||
|
||||
fail("Expecting a JSONException");
|
||||
} catch (JSONException e) {
|
||||
assertTrue("Wrong throwable thrown: not expecting message <" + e.getMessage() + ">",
|
||||
e.getMessage().startsWith("Maximum nesting depth of " + maxNestingDepth));
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testMaxNestingDepthWithValidFittingXML() {
|
||||
final String perfectlyFineXML = "<Test>\n" +
|
||||
" <employee>\n" +
|
||||
" <name>sonoo</name>\n" +
|
||||
" <salary>56000</salary>\n" +
|
||||
" <married>true</married>\n" +
|
||||
" </employee>\n" +
|
||||
"</Test>\n";
|
||||
|
||||
final int maxNestingDepth = 3;
|
||||
|
||||
try {
|
||||
XML.toJSONObject(perfectlyFineXML, XMLParserConfiguration.ORIGINAL.withMaxNestingDepth(maxNestingDepth));
|
||||
} catch (JSONException e) {
|
||||
e.printStackTrace();
|
||||
fail("XML document should be parsed as its maximum depth fits the maxNestingDepth " +
|
||||
"parameter of the XMLParserConfiguration used");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
704
src/test/resources/Issue593.json
Normal file
704
src/test/resources/Issue593.json
Normal file
@@ -0,0 +1,704 @@
|
||||
{
|
||||
"success": true,
|
||||
"error": null,
|
||||
"response": [
|
||||
{
|
||||
"loc": {
|
||||
"long": 31.25,
|
||||
"lat": 30.063
|
||||
},
|
||||
"interval": "day",
|
||||
"place": {
|
||||
"name": "cairo",
|
||||
"state": "qh",
|
||||
"country": "eg"
|
||||
},
|
||||
"periods": [
|
||||
{
|
||||
"timestamp": 1665032400,
|
||||
"validTime": "2022-10-06T07:00:00+02:00",
|
||||
"dateTimeISO": "2022-10-06T07:00:00+02:00",
|
||||
"maxTempC": 32,
|
||||
"maxTempF": 90,
|
||||
"minTempC": 19,
|
||||
"minTempF": 66,
|
||||
"avgTempC": 25,
|
||||
"avgTempF": 78,
|
||||
"tempC": null,
|
||||
"tempF": null,
|
||||
"maxFeelslikeC": 32,
|
||||
"maxFeelslikeF": 89,
|
||||
"minFeelslikeC": 21,
|
||||
"minFeelslikeF": 70,
|
||||
"avgFeelslikeC": 26,
|
||||
"avgFeelslikeF": 80,
|
||||
"feelslikeC": 21,
|
||||
"feelslikeF": 70,
|
||||
"maxDewpointC": 17,
|
||||
"maxDewpointF": 63,
|
||||
"minDewpointC": 11,
|
||||
"minDewpointF": 52,
|
||||
"avgDewpointC": 14,
|
||||
"avgDewpointF": 58,
|
||||
"dewpointC": 17,
|
||||
"dewpointF": 63,
|
||||
"maxHumidity": 77,
|
||||
"minHumidity": 29,
|
||||
"humidity": 77,
|
||||
"pop": 0,
|
||||
"precipMM": 0,
|
||||
"precipIN": 0,
|
||||
"iceaccum": null,
|
||||
"iceaccumMM": null,
|
||||
"iceaccumIN": null,
|
||||
"snowCM": 0,
|
||||
"snowIN": 0,
|
||||
"pressureMB": 1015,
|
||||
"pressureIN": 29.97,
|
||||
"windDir": "N",
|
||||
"windDirDEG": 353,
|
||||
"windSpeedKTS": 5,
|
||||
"windSpeedKPH": 9,
|
||||
"windSpeedMPH": 6,
|
||||
"windGustKTS": 21,
|
||||
"windGustKPH": 40,
|
||||
"windGustMPH": 25,
|
||||
"windDirMax": "NNW",
|
||||
"windDirMaxDEG": 342,
|
||||
"windSpeedMaxKTS": 9,
|
||||
"windSpeedMaxKPH": 16,
|
||||
"windSpeedMaxMPH": 10,
|
||||
"windDirMin": "N",
|
||||
"windDirMinDEG": 353,
|
||||
"windSpeedMinKTS": 1,
|
||||
"windSpeedMinKPH": 2,
|
||||
"windSpeedMinMPH": 1,
|
||||
"windDir80m": "N",
|
||||
"windDir80mDEG": 11,
|
||||
"windSpeed80mKTS": 12,
|
||||
"windSpeed80mKPH": 22,
|
||||
"windSpeed80mMPH": 13,
|
||||
"windGust80mKTS": 22,
|
||||
"windGust80mKPH": 41,
|
||||
"windGust80mMPH": 25,
|
||||
"windDirMax80m": "NNW",
|
||||
"windDirMax80mDEG": 343,
|
||||
"windSpeedMax80mKTS": 22,
|
||||
"windSpeedMax80mKPH": 41,
|
||||
"windSpeedMax80mMPH": 25,
|
||||
"windDirMin80m": "E",
|
||||
"windDirMin80mDEG": 95,
|
||||
"windSpeedMin80mKTS": 8,
|
||||
"windSpeedMin80mKPH": 15,
|
||||
"windSpeedMin80mMPH": 10,
|
||||
"sky": 22,
|
||||
"cloudsCoded": "FW",
|
||||
"weather": "Mostly Sunny",
|
||||
"weatherCoded": [],
|
||||
"weatherPrimary": "Mostly Sunny",
|
||||
"weatherPrimaryCoded": "::FW",
|
||||
"icon": "fair.png",
|
||||
"visibilityKM": 24.135,
|
||||
"visibilityMI": 15,
|
||||
"uvi": 6,
|
||||
"solradWM2": 5608,
|
||||
"solradMinWM2": 0,
|
||||
"solradMaxWM2": 778,
|
||||
"isDay": true,
|
||||
"maxCoverage": "",
|
||||
"sunrise": 1665028274,
|
||||
"sunset": 1665070502,
|
||||
"sunriseISO": "2022-10-06T05:51:14+02:00",
|
||||
"sunsetISO": "2022-10-06T17:35:02+02:00"
|
||||
},
|
||||
{
|
||||
"timestamp": 1665118800,
|
||||
"validTime": "2022-10-07T07:00:00+02:00",
|
||||
"dateTimeISO": "2022-10-07T07:00:00+02:00",
|
||||
"maxTempC": 30,
|
||||
"maxTempF": 86,
|
||||
"minTempC": 19,
|
||||
"minTempF": 66,
|
||||
"avgTempC": 24,
|
||||
"avgTempF": 76,
|
||||
"tempC": null,
|
||||
"tempF": null,
|
||||
"maxFeelslikeC": 29,
|
||||
"maxFeelslikeF": 85,
|
||||
"minFeelslikeC": 19,
|
||||
"minFeelslikeF": 67,
|
||||
"avgFeelslikeC": 24,
|
||||
"avgFeelslikeF": 76,
|
||||
"feelslikeC": 19,
|
||||
"feelslikeF": 67,
|
||||
"maxDewpointC": 15,
|
||||
"maxDewpointF": 60,
|
||||
"minDewpointC": 10,
|
||||
"minDewpointF": 50,
|
||||
"avgDewpointC": 12,
|
||||
"avgDewpointF": 54,
|
||||
"dewpointC": 15,
|
||||
"dewpointF": 60,
|
||||
"maxHumidity": 77,
|
||||
"minHumidity": 30,
|
||||
"humidity": 77,
|
||||
"pop": 0,
|
||||
"precipMM": 0,
|
||||
"precipIN": 0,
|
||||
"iceaccum": null,
|
||||
"iceaccumMM": null,
|
||||
"iceaccumIN": null,
|
||||
"snowCM": 0,
|
||||
"snowIN": 0,
|
||||
"pressureMB": 1014,
|
||||
"pressureIN": 29.95,
|
||||
"windDir": "NW",
|
||||
"windDirDEG": 325,
|
||||
"windSpeedKTS": 1,
|
||||
"windSpeedKPH": 2,
|
||||
"windSpeedMPH": 1,
|
||||
"windGustKTS": 16,
|
||||
"windGustKPH": 29,
|
||||
"windGustMPH": 18,
|
||||
"windDirMax": "WNW",
|
||||
"windDirMaxDEG": 298,
|
||||
"windSpeedMaxKTS": 7,
|
||||
"windSpeedMaxKPH": 13,
|
||||
"windSpeedMaxMPH": 8,
|
||||
"windDirMin": "NW",
|
||||
"windDirMinDEG": 325,
|
||||
"windSpeedMinKTS": 1,
|
||||
"windSpeedMinKPH": 2,
|
||||
"windSpeedMinMPH": 1,
|
||||
"windDir80m": "NNW",
|
||||
"windDir80mDEG": 347,
|
||||
"windSpeed80mKTS": 6,
|
||||
"windSpeed80mKPH": 10,
|
||||
"windSpeed80mMPH": 6,
|
||||
"windGust80mKTS": 20,
|
||||
"windGust80mKPH": 37,
|
||||
"windGust80mMPH": 23,
|
||||
"windDirMax80m": "NW",
|
||||
"windDirMax80mDEG": 316,
|
||||
"windSpeedMax80mKTS": 20,
|
||||
"windSpeedMax80mKPH": 37,
|
||||
"windSpeedMax80mMPH": 23,
|
||||
"windDirMin80m": "NNW",
|
||||
"windDirMin80mDEG": 347,
|
||||
"windSpeedMin80mKTS": 6,
|
||||
"windSpeedMin80mKPH": 10,
|
||||
"windSpeedMin80mMPH": 6,
|
||||
"sky": 30,
|
||||
"cloudsCoded": "FW",
|
||||
"weather": "Mostly Sunny",
|
||||
"weatherCoded": [],
|
||||
"weatherPrimary": "Mostly Sunny",
|
||||
"weatherPrimaryCoded": "::FW",
|
||||
"icon": "fair.png",
|
||||
"visibilityKM": 24.135,
|
||||
"visibilityMI": 15,
|
||||
"uvi": 6,
|
||||
"solradWM2": 5486,
|
||||
"solradMinWM2": 0,
|
||||
"solradMaxWM2": 742,
|
||||
"isDay": true,
|
||||
"maxCoverage": "",
|
||||
"sunrise": 1665114710,
|
||||
"sunset": 1665156831,
|
||||
"sunriseISO": "2022-10-07T05:51:50+02:00",
|
||||
"sunsetISO": "2022-10-07T17:33:51+02:00"
|
||||
},
|
||||
{
|
||||
"timestamp": 1665205200,
|
||||
"validTime": "2022-10-08T07:00:00+02:00",
|
||||
"dateTimeISO": "2022-10-08T07:00:00+02:00",
|
||||
"maxTempC": 30,
|
||||
"maxTempF": 87,
|
||||
"minTempC": 19,
|
||||
"minTempF": 66,
|
||||
"avgTempC": 25,
|
||||
"avgTempF": 76,
|
||||
"tempC": null,
|
||||
"tempF": null,
|
||||
"maxFeelslikeC": 30,
|
||||
"maxFeelslikeF": 86,
|
||||
"minFeelslikeC": 19,
|
||||
"minFeelslikeF": 67,
|
||||
"avgFeelslikeC": 25,
|
||||
"avgFeelslikeF": 76,
|
||||
"feelslikeC": 19,
|
||||
"feelslikeF": 67,
|
||||
"maxDewpointC": 15,
|
||||
"maxDewpointF": 59,
|
||||
"minDewpointC": 11,
|
||||
"minDewpointF": 52,
|
||||
"avgDewpointC": 13,
|
||||
"avgDewpointF": 56,
|
||||
"dewpointC": 15,
|
||||
"dewpointF": 59,
|
||||
"maxHumidity": 76,
|
||||
"minHumidity": 32,
|
||||
"humidity": 76,
|
||||
"pop": 0,
|
||||
"precipMM": 0,
|
||||
"precipIN": 0,
|
||||
"iceaccum": null,
|
||||
"iceaccumMM": null,
|
||||
"iceaccumIN": null,
|
||||
"snowCM": 0,
|
||||
"snowIN": 0,
|
||||
"pressureMB": 1014,
|
||||
"pressureIN": 29.94,
|
||||
"windDir": "NNE",
|
||||
"windDirDEG": 21,
|
||||
"windSpeedKTS": 1,
|
||||
"windSpeedKPH": 2,
|
||||
"windSpeedMPH": 1,
|
||||
"windGustKTS": 17,
|
||||
"windGustKPH": 32,
|
||||
"windGustMPH": 20,
|
||||
"windDirMax": "WNW",
|
||||
"windDirMaxDEG": 301,
|
||||
"windSpeedMaxKTS": 7,
|
||||
"windSpeedMaxKPH": 13,
|
||||
"windSpeedMaxMPH": 8,
|
||||
"windDirMin": "NNE",
|
||||
"windDirMinDEG": 21,
|
||||
"windSpeedMinKTS": 1,
|
||||
"windSpeedMinKPH": 2,
|
||||
"windSpeedMinMPH": 1,
|
||||
"windDir80m": "NW",
|
||||
"windDir80mDEG": 309,
|
||||
"windSpeed80mKTS": 5,
|
||||
"windSpeed80mKPH": 9,
|
||||
"windSpeed80mMPH": 5,
|
||||
"windGust80mKTS": 17,
|
||||
"windGust80mKPH": 31,
|
||||
"windGust80mMPH": 19,
|
||||
"windDirMax80m": "NW",
|
||||
"windDirMax80mDEG": 322,
|
||||
"windSpeedMax80mKTS": 17,
|
||||
"windSpeedMax80mKPH": 31,
|
||||
"windSpeedMax80mMPH": 19,
|
||||
"windDirMin80m": "NW",
|
||||
"windDirMin80mDEG": 309,
|
||||
"windSpeedMin80mKTS": 5,
|
||||
"windSpeedMin80mKPH": 9,
|
||||
"windSpeedMin80mMPH": 5,
|
||||
"sky": 47,
|
||||
"cloudsCoded": "SC",
|
||||
"weather": "Partly Cloudy",
|
||||
"weatherCoded": [],
|
||||
"weatherPrimary": "Partly Cloudy",
|
||||
"weatherPrimaryCoded": "::SC",
|
||||
"icon": "pcloudy.png",
|
||||
"visibilityKM": 24.135,
|
||||
"visibilityMI": 15,
|
||||
"uvi": 7,
|
||||
"solradWM2": 4785,
|
||||
"solradMinWM2": 0,
|
||||
"solradMaxWM2": 682,
|
||||
"isDay": true,
|
||||
"maxCoverage": "",
|
||||
"sunrise": 1665201146,
|
||||
"sunset": 1665243161,
|
||||
"sunriseISO": "2022-10-08T05:52:26+02:00",
|
||||
"sunsetISO": "2022-10-08T17:32:41+02:00"
|
||||
},
|
||||
{
|
||||
"timestamp": 1665291600,
|
||||
"validTime": "2022-10-09T07:00:00+02:00",
|
||||
"dateTimeISO": "2022-10-09T07:00:00+02:00",
|
||||
"maxTempC": 31,
|
||||
"maxTempF": 87,
|
||||
"minTempC": 19,
|
||||
"minTempF": 67,
|
||||
"avgTempC": 25,
|
||||
"avgTempF": 77,
|
||||
"tempC": null,
|
||||
"tempF": null,
|
||||
"maxFeelslikeC": 30,
|
||||
"maxFeelslikeF": 86,
|
||||
"minFeelslikeC": 20,
|
||||
"minFeelslikeF": 67,
|
||||
"avgFeelslikeC": 25,
|
||||
"avgFeelslikeF": 77,
|
||||
"feelslikeC": 20,
|
||||
"feelslikeF": 67,
|
||||
"maxDewpointC": 17,
|
||||
"maxDewpointF": 63,
|
||||
"minDewpointC": 11,
|
||||
"minDewpointF": 52,
|
||||
"avgDewpointC": 14,
|
||||
"avgDewpointF": 57,
|
||||
"dewpointC": 17,
|
||||
"dewpointF": 63,
|
||||
"maxHumidity": 86,
|
||||
"minHumidity": 31,
|
||||
"humidity": 86,
|
||||
"pop": 0,
|
||||
"precipMM": 0,
|
||||
"precipIN": 0,
|
||||
"iceaccum": null,
|
||||
"iceaccumMM": null,
|
||||
"iceaccumIN": null,
|
||||
"snowCM": 0,
|
||||
"snowIN": 0,
|
||||
"pressureMB": 1016,
|
||||
"pressureIN": 29.99,
|
||||
"windDir": "N",
|
||||
"windDirDEG": 356,
|
||||
"windSpeedKTS": 2,
|
||||
"windSpeedKPH": 4,
|
||||
"windSpeedMPH": 2,
|
||||
"windGustKTS": 19,
|
||||
"windGustKPH": 36,
|
||||
"windGustMPH": 22,
|
||||
"windDirMax": "NNW",
|
||||
"windDirMaxDEG": 343,
|
||||
"windSpeedMaxKTS": 8,
|
||||
"windSpeedMaxKPH": 14,
|
||||
"windSpeedMaxMPH": 9,
|
||||
"windDirMin": "N",
|
||||
"windDirMinDEG": 356,
|
||||
"windSpeedMinKTS": 2,
|
||||
"windSpeedMinKPH": 4,
|
||||
"windSpeedMinMPH": 2,
|
||||
"windDir80m": "NW",
|
||||
"windDir80mDEG": 316,
|
||||
"windSpeed80mKTS": 5,
|
||||
"windSpeed80mKPH": 9,
|
||||
"windSpeed80mMPH": 6,
|
||||
"windGust80mKTS": 20,
|
||||
"windGust80mKPH": 36,
|
||||
"windGust80mMPH": 23,
|
||||
"windDirMax80m": "N",
|
||||
"windDirMax80mDEG": 354,
|
||||
"windSpeedMax80mKTS": 20,
|
||||
"windSpeedMax80mKPH": 36,
|
||||
"windSpeedMax80mMPH": 23,
|
||||
"windDirMin80m": "NW",
|
||||
"windDirMin80mDEG": 316,
|
||||
"windSpeedMin80mKTS": 5,
|
||||
"windSpeedMin80mKPH": 9,
|
||||
"windSpeedMin80mMPH": 6,
|
||||
"sky": 47,
|
||||
"cloudsCoded": "SC",
|
||||
"weather": "Partly Cloudy",
|
||||
"weatherCoded": [],
|
||||
"weatherPrimary": "Partly Cloudy",
|
||||
"weatherPrimaryCoded": "::SC",
|
||||
"icon": "pcloudy.png",
|
||||
"visibilityKM": 24.135,
|
||||
"visibilityMI": 15,
|
||||
"uvi": 7,
|
||||
"solradWM2": 4768,
|
||||
"solradMinWM2": 0,
|
||||
"solradMaxWM2": 726,
|
||||
"isDay": true,
|
||||
"maxCoverage": "",
|
||||
"sunrise": 1665287583,
|
||||
"sunset": 1665329491,
|
||||
"sunriseISO": "2022-10-09T05:53:03+02:00",
|
||||
"sunsetISO": "2022-10-09T17:31:31+02:00"
|
||||
},
|
||||
{
|
||||
"timestamp": 1665378000,
|
||||
"validTime": "2022-10-10T07:00:00+02:00",
|
||||
"dateTimeISO": "2022-10-10T07:00:00+02:00",
|
||||
"maxTempC": 31,
|
||||
"maxTempF": 87,
|
||||
"minTempC": 21,
|
||||
"minTempF": 70,
|
||||
"avgTempC": 26,
|
||||
"avgTempF": 78,
|
||||
"tempC": null,
|
||||
"tempF": null,
|
||||
"maxFeelslikeC": 30,
|
||||
"maxFeelslikeF": 86,
|
||||
"minFeelslikeC": 21,
|
||||
"minFeelslikeF": 69,
|
||||
"avgFeelslikeC": 25,
|
||||
"avgFeelslikeF": 78,
|
||||
"feelslikeC": 21,
|
||||
"feelslikeF": 69,
|
||||
"maxDewpointC": 16,
|
||||
"maxDewpointF": 61,
|
||||
"minDewpointC": 13,
|
||||
"minDewpointF": 55,
|
||||
"avgDewpointC": 14,
|
||||
"avgDewpointF": 58,
|
||||
"dewpointC": 16,
|
||||
"dewpointF": 61,
|
||||
"maxHumidity": 75,
|
||||
"minHumidity": 35,
|
||||
"humidity": 75,
|
||||
"pop": 0,
|
||||
"precipMM": 0,
|
||||
"precipIN": 0,
|
||||
"iceaccum": null,
|
||||
"iceaccumMM": null,
|
||||
"iceaccumIN": null,
|
||||
"snowCM": 0,
|
||||
"snowIN": 0,
|
||||
"pressureMB": 1017,
|
||||
"pressureIN": 30.03,
|
||||
"windDir": "N",
|
||||
"windDirDEG": 358,
|
||||
"windSpeedKTS": 2,
|
||||
"windSpeedKPH": 4,
|
||||
"windSpeedMPH": 2,
|
||||
"windGustKTS": 16,
|
||||
"windGustKPH": 30,
|
||||
"windGustMPH": 19,
|
||||
"windDirMax": "N",
|
||||
"windDirMaxDEG": 10,
|
||||
"windSpeedMaxKTS": 8,
|
||||
"windSpeedMaxKPH": 15,
|
||||
"windSpeedMaxMPH": 9,
|
||||
"windDirMin": "N",
|
||||
"windDirMinDEG": 358,
|
||||
"windSpeedMinKTS": 2,
|
||||
"windSpeedMinKPH": 4,
|
||||
"windSpeedMinMPH": 2,
|
||||
"windDir80m": "N",
|
||||
"windDir80mDEG": 8,
|
||||
"windSpeed80mKTS": 7,
|
||||
"windSpeed80mKPH": 13,
|
||||
"windSpeed80mMPH": 8,
|
||||
"windGust80mKTS": 19,
|
||||
"windGust80mKPH": 36,
|
||||
"windGust80mMPH": 22,
|
||||
"windDirMax80m": "N",
|
||||
"windDirMax80mDEG": 10,
|
||||
"windSpeedMax80mKTS": 19,
|
||||
"windSpeedMax80mKPH": 36,
|
||||
"windSpeedMax80mMPH": 22,
|
||||
"windDirMin80m": "E",
|
||||
"windDirMin80mDEG": 91,
|
||||
"windSpeedMin80mKTS": 7,
|
||||
"windSpeedMin80mKPH": 13,
|
||||
"windSpeedMin80mMPH": 8,
|
||||
"sky": 64,
|
||||
"cloudsCoded": "SC",
|
||||
"weather": "Partly Cloudy",
|
||||
"weatherCoded": [],
|
||||
"weatherPrimary": "Partly Cloudy",
|
||||
"weatherPrimaryCoded": "::SC",
|
||||
"icon": "pcloudy.png",
|
||||
"visibilityKM": 24.135,
|
||||
"visibilityMI": 15,
|
||||
"uvi": 6,
|
||||
"solradWM2": 4494,
|
||||
"solradMinWM2": 0,
|
||||
"solradMaxWM2": 597,
|
||||
"isDay": true,
|
||||
"maxCoverage": "",
|
||||
"sunrise": 1665374020,
|
||||
"sunset": 1665415821,
|
||||
"sunriseISO": "2022-10-10T05:53:40+02:00",
|
||||
"sunsetISO": "2022-10-10T17:30:21+02:00"
|
||||
},
|
||||
{
|
||||
"timestamp": 1665464400,
|
||||
"validTime": "2022-10-11T07:00:00+02:00",
|
||||
"dateTimeISO": "2022-10-11T07:00:00+02:00",
|
||||
"maxTempC": 31,
|
||||
"maxTempF": 87,
|
||||
"minTempC": 21,
|
||||
"minTempF": 70,
|
||||
"avgTempC": 26,
|
||||
"avgTempF": 78,
|
||||
"tempC": null,
|
||||
"tempF": null,
|
||||
"maxFeelslikeC": 31,
|
||||
"maxFeelslikeF": 87,
|
||||
"minFeelslikeC": 22,
|
||||
"minFeelslikeF": 72,
|
||||
"avgFeelslikeC": 26,
|
||||
"avgFeelslikeF": 79,
|
||||
"feelslikeC": 22,
|
||||
"feelslikeF": 72,
|
||||
"maxDewpointC": 17,
|
||||
"maxDewpointF": 62,
|
||||
"minDewpointC": 11,
|
||||
"minDewpointF": 51,
|
||||
"avgDewpointC": 13,
|
||||
"avgDewpointF": 55,
|
||||
"dewpointC": 17,
|
||||
"dewpointF": 62,
|
||||
"maxHumidity": 71,
|
||||
"minHumidity": 30,
|
||||
"humidity": 71,
|
||||
"pop": 0,
|
||||
"precipMM": 0,
|
||||
"precipIN": 0,
|
||||
"iceaccum": null,
|
||||
"iceaccumMM": null,
|
||||
"iceaccumIN": null,
|
||||
"snowCM": 0,
|
||||
"snowIN": 0,
|
||||
"pressureMB": 1015,
|
||||
"pressureIN": 29.98,
|
||||
"windDir": "NNE",
|
||||
"windDirDEG": 13,
|
||||
"windSpeedKTS": 8,
|
||||
"windSpeedKPH": 15,
|
||||
"windSpeedMPH": 9,
|
||||
"windGustKTS": 15,
|
||||
"windGustKPH": 28,
|
||||
"windGustMPH": 17,
|
||||
"windDirMax": "NNE",
|
||||
"windDirMaxDEG": 28,
|
||||
"windSpeedMaxKTS": 15,
|
||||
"windSpeedMaxKPH": 28,
|
||||
"windSpeedMaxMPH": 18,
|
||||
"windDirMin": "NNE",
|
||||
"windDirMinDEG": 14,
|
||||
"windSpeedMinKTS": 7,
|
||||
"windSpeedMinKPH": 14,
|
||||
"windSpeedMinMPH": 8,
|
||||
"windDir80m": "NNE",
|
||||
"windDir80mDEG": 16,
|
||||
"windSpeed80mKTS": 10,
|
||||
"windSpeed80mKPH": 19,
|
||||
"windSpeed80mMPH": 12,
|
||||
"windGust80mKTS": 17,
|
||||
"windGust80mKPH": 31,
|
||||
"windGust80mMPH": 19,
|
||||
"windDirMax80m": "NNE",
|
||||
"windDirMax80mDEG": 28,
|
||||
"windSpeedMax80mKTS": 17,
|
||||
"windSpeedMax80mKPH": 31,
|
||||
"windSpeedMax80mMPH": 19,
|
||||
"windDirMin80m": "NNE",
|
||||
"windDirMin80mDEG": 13,
|
||||
"windSpeedMin80mKTS": 9,
|
||||
"windSpeedMin80mKPH": 18,
|
||||
"windSpeedMin80mMPH": 11,
|
||||
"sky": 0,
|
||||
"cloudsCoded": "CL",
|
||||
"weather": "Sunny",
|
||||
"weatherCoded": [],
|
||||
"weatherPrimary": "Sunny",
|
||||
"weatherPrimaryCoded": "::CL",
|
||||
"icon": "sunny.png",
|
||||
"visibilityKM": 24.135,
|
||||
"visibilityMI": 15,
|
||||
"uvi": null,
|
||||
"solradWM2": 5450,
|
||||
"solradMinWM2": 0,
|
||||
"solradMaxWM2": 758,
|
||||
"isDay": true,
|
||||
"maxCoverage": "",
|
||||
"sunrise": 1665460458,
|
||||
"sunset": 1665502153,
|
||||
"sunriseISO": "2022-10-11T05:54:18+02:00",
|
||||
"sunsetISO": "2022-10-11T17:29:13+02:00"
|
||||
},
|
||||
{
|
||||
"timestamp": 1665550800,
|
||||
"validTime": "2022-10-12T07:00:00+02:00",
|
||||
"dateTimeISO": "2022-10-12T07:00:00+02:00",
|
||||
"maxTempC": 31,
|
||||
"maxTempF": 88,
|
||||
"minTempC": 21,
|
||||
"minTempF": 69,
|
||||
"avgTempC": 26,
|
||||
"avgTempF": 79,
|
||||
"tempC": null,
|
||||
"tempF": null,
|
||||
"maxFeelslikeC": 31,
|
||||
"maxFeelslikeF": 88,
|
||||
"minFeelslikeC": 22,
|
||||
"minFeelslikeF": 72,
|
||||
"avgFeelslikeC": 26,
|
||||
"avgFeelslikeF": 80,
|
||||
"feelslikeC": 22,
|
||||
"feelslikeF": 72,
|
||||
"maxDewpointC": 16,
|
||||
"maxDewpointF": 60,
|
||||
"minDewpointC": 11,
|
||||
"minDewpointF": 51,
|
||||
"avgDewpointC": 13,
|
||||
"avgDewpointF": 55,
|
||||
"dewpointC": 16,
|
||||
"dewpointF": 60,
|
||||
"maxHumidity": 68,
|
||||
"minHumidity": 29,
|
||||
"humidity": 68,
|
||||
"pop": 0,
|
||||
"precipMM": 0,
|
||||
"precipIN": 0,
|
||||
"iceaccum": null,
|
||||
"iceaccumMM": null,
|
||||
"iceaccumIN": null,
|
||||
"snowCM": 0,
|
||||
"snowIN": 0,
|
||||
"pressureMB": 1014,
|
||||
"pressureIN": 29.95,
|
||||
"windDir": "NNE",
|
||||
"windDirDEG": 12,
|
||||
"windSpeedKTS": 8,
|
||||
"windSpeedKPH": 15,
|
||||
"windSpeedMPH": 9,
|
||||
"windGustKTS": 15,
|
||||
"windGustKPH": 28,
|
||||
"windGustMPH": 17,
|
||||
"windDirMax": "E",
|
||||
"windDirMaxDEG": 96,
|
||||
"windSpeedMaxKTS": 14,
|
||||
"windSpeedMaxKPH": 26,
|
||||
"windSpeedMaxMPH": 16,
|
||||
"windDirMin": "NNE",
|
||||
"windDirMinDEG": 12,
|
||||
"windSpeedMinKTS": 7,
|
||||
"windSpeedMinKPH": 13,
|
||||
"windSpeedMinMPH": 8,
|
||||
"windDir80m": "NNE",
|
||||
"windDir80mDEG": 15,
|
||||
"windSpeed80mKTS": 10,
|
||||
"windSpeed80mKPH": 19,
|
||||
"windSpeed80mMPH": 12,
|
||||
"windGust80mKTS": 18,
|
||||
"windGust80mKPH": 33,
|
||||
"windGust80mMPH": 21,
|
||||
"windDirMax80m": "E",
|
||||
"windDirMax80mDEG": 96,
|
||||
"windSpeedMax80mKTS": 18,
|
||||
"windSpeedMax80mKPH": 33,
|
||||
"windSpeedMax80mMPH": 21,
|
||||
"windDirMin80m": "NNE",
|
||||
"windDirMin80mDEG": 15,
|
||||
"windSpeedMin80mKTS": 10,
|
||||
"windSpeedMin80mKPH": 18,
|
||||
"windSpeedMin80mMPH": 11,
|
||||
"sky": 27,
|
||||
"cloudsCoded": "FW",
|
||||
"weather": "Mostly Sunny",
|
||||
"weatherCoded": [],
|
||||
"weatherPrimary": "Mostly Sunny",
|
||||
"weatherPrimaryCoded": "::FW",
|
||||
"icon": "fair.png",
|
||||
"visibilityKM": 24.135,
|
||||
"visibilityMI": 15,
|
||||
"uvi": null,
|
||||
"solradWM2": 4740,
|
||||
"solradMinWM2": 0,
|
||||
"solradMaxWM2": 743,
|
||||
"isDay": true,
|
||||
"maxCoverage": "",
|
||||
"sunrise": 1665546895,
|
||||
"sunset": 1665588484,
|
||||
"sunriseISO": "2022-10-12T05:54:55+02:00",
|
||||
"sunsetISO": "2022-10-12T17:28:04+02:00"
|
||||
}
|
||||
],
|
||||
"profile": {
|
||||
"tz": "Africa/Cairo",
|
||||
"elevM": 23,
|
||||
"elevFT": 75
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
691
src/test/resources/Issue593.xml
Normal file
691
src/test/resources/Issue593.xml
Normal file
@@ -0,0 +1,691 @@
|
||||
<success>true</success>
|
||||
<response>
|
||||
<loc>
|
||||
<long>31.25</long>
|
||||
<lat>30.063</lat>
|
||||
</loc>
|
||||
<profile>
|
||||
<elevM>23</elevM>
|
||||
<tz>Africa/Cairo</tz>
|
||||
<elevFT>75</elevFT>
|
||||
</profile>
|
||||
<periods>
|
||||
<dateTimeISO>2022-10-06T07:00:00+02:00</dateTimeISO>
|
||||
<windDirMin80m>E</windDirMin80m>
|
||||
<windDirMin80mDEG>95</windDirMin80mDEG>
|
||||
<feelslikeC>21</feelslikeC>
|
||||
<visibilityMI>15</visibilityMI>
|
||||
<windSpeedMaxMPH>10</windSpeedMaxMPH>
|
||||
<windDirDEG>353</windDirDEG>
|
||||
<windDir>N</windDir>
|
||||
<sunriseISO>2022-10-06T05:51:14+02:00</sunriseISO>
|
||||
<iceaccumMM>null</iceaccumMM>
|
||||
<windSpeedMaxKTS>9</windSpeedMaxKTS>
|
||||
<iceaccumIN>null</iceaccumIN>
|
||||
<minTempF>66</minTempF>
|
||||
<snowIN>0</snowIN>
|
||||
<weather>Mostly Sunny</weather>
|
||||
<sunsetISO>2022-10-06T17:35:02+02:00</sunsetISO>
|
||||
<maxFeelslikeC>32</maxFeelslikeC>
|
||||
<humidity>77</humidity>
|
||||
<windDir80m>N</windDir80m>
|
||||
<maxFeelslikeF>89</maxFeelslikeF>
|
||||
<precipMM>0</precipMM>
|
||||
<sky>22</sky>
|
||||
<windGust80mMPH>25</windGust80mMPH>
|
||||
<windSpeedMax80mMPH>25</windSpeedMax80mMPH>
|
||||
<weatherPrimary>Mostly Sunny</weatherPrimary>
|
||||
<windGust80mKPH>41</windGust80mKPH>
|
||||
<avgDewpointF>58</avgDewpointF>
|
||||
<windSpeedMax80mKPH>41</windSpeedMax80mKPH>
|
||||
<windGust80mKTS>22</windGust80mKTS>
|
||||
<avgDewpointC>14</avgDewpointC>
|
||||
<precipIN>0</precipIN>
|
||||
<windSpeedMax80mKTS>22</windSpeedMax80mKTS>
|
||||
<windDirMinDEG>353</windDirMinDEG>
|
||||
<windSpeedMaxKPH>16</windSpeedMaxKPH>
|
||||
<windSpeedMin80mKTS>8</windSpeedMin80mKTS>
|
||||
<feelslikeF>70</feelslikeF>
|
||||
<validTime>2022-10-06T07:00:00+02:00</validTime>
|
||||
<windSpeedMin80mMPH>10</windSpeedMin80mMPH>
|
||||
<solradMaxWM2>778</solradMaxWM2>
|
||||
<avgTempC>25</avgTempC>
|
||||
<windSpeedMin80mKPH>15</windSpeedMin80mKPH>
|
||||
<weatherPrimaryCoded>::FW</weatherPrimaryCoded>
|
||||
<sunrise>1665028274</sunrise>
|
||||
<avgTempF>78</avgTempF>
|
||||
<windDirMin>N</windDirMin>
|
||||
<maxCoverage/>
|
||||
<icon>fair.png</icon>
|
||||
<minFeelslikeC>21</minFeelslikeC>
|
||||
<dewpointC>17</dewpointC>
|
||||
<cloudsCoded>FW</cloudsCoded>
|
||||
<minFeelslikeF>70</minFeelslikeF>
|
||||
<minHumidity>29</minHumidity>
|
||||
<dewpointF>63</dewpointF>
|
||||
<windSpeed80mKTS>12</windSpeed80mKTS>
|
||||
<pop>0</pop>
|
||||
<snowCM>0</snowCM>
|
||||
<windDirMax>NNW</windDirMax>
|
||||
<windSpeed80mMPH>13</windSpeed80mMPH>
|
||||
<windSpeed80mKPH>22</windSpeed80mKPH>
|
||||
<windDir80mDEG>11</windDir80mDEG>
|
||||
<maxTempC>32</maxTempC>
|
||||
<pressureMB>1015</pressureMB>
|
||||
<visibilityKM>24.135</visibilityKM>
|
||||
<timestamp>1665032400</timestamp>
|
||||
<maxTempF>90</maxTempF>
|
||||
<tempF>null</tempF>
|
||||
<minDewpointC>11</minDewpointC>
|
||||
<solradMinWM2>0</solradMinWM2>
|
||||
<windSpeedMinKTS>1</windSpeedMinKTS>
|
||||
<windDirMax80mDEG>343</windDirMax80mDEG>
|
||||
<windGustKTS>21</windGustKTS>
|
||||
<windSpeedMinKPH>2</windSpeedMinKPH>
|
||||
<maxDewpointF>63</maxDewpointF>
|
||||
<windSpeedMinMPH>1</windSpeedMinMPH>
|
||||
<avgFeelslikeC>26</avgFeelslikeC>
|
||||
<uvi>6</uvi>
|
||||
<windDirMax80m>NNW</windDirMax80m>
|
||||
<maxDewpointC>17</maxDewpointC>
|
||||
<pressureIN>29.97</pressureIN>
|
||||
<avgFeelslikeF>80</avgFeelslikeF>
|
||||
<iceaccum>null</iceaccum>
|
||||
<isDay>true</isDay>
|
||||
<minTempC>19</minTempC>
|
||||
<minDewpointF>52</minDewpointF>
|
||||
<windSpeedKTS>5</windSpeedKTS>
|
||||
<sunset>1665070502</sunset>
|
||||
<solradWM2>5608</solradWM2>
|
||||
<windSpeedKPH>9</windSpeedKPH>
|
||||
<windGustMPH>25</windGustMPH>
|
||||
<maxHumidity>77</maxHumidity>
|
||||
<windSpeedMPH>6</windSpeedMPH>
|
||||
<windGustKPH>40</windGustKPH>
|
||||
<windDirMaxDEG>342</windDirMaxDEG>
|
||||
<tempC>null</tempC>
|
||||
</periods>
|
||||
<periods>
|
||||
<dateTimeISO>2022-10-07T07:00:00+02:00</dateTimeISO>
|
||||
<windDirMin80m>NNW</windDirMin80m>
|
||||
<windDirMin80mDEG>347</windDirMin80mDEG>
|
||||
<feelslikeC>19</feelslikeC>
|
||||
<visibilityMI>15</visibilityMI>
|
||||
<windSpeedMaxMPH>8</windSpeedMaxMPH>
|
||||
<windDirDEG>325</windDirDEG>
|
||||
<windDir>NW</windDir>
|
||||
<sunriseISO>2022-10-07T05:51:50+02:00</sunriseISO>
|
||||
<iceaccumMM>null</iceaccumMM>
|
||||
<windSpeedMaxKTS>7</windSpeedMaxKTS>
|
||||
<iceaccumIN>null</iceaccumIN>
|
||||
<minTempF>66</minTempF>
|
||||
<snowIN>0</snowIN>
|
||||
<weather>Mostly Sunny</weather>
|
||||
<sunsetISO>2022-10-07T17:33:51+02:00</sunsetISO>
|
||||
<maxFeelslikeC>29</maxFeelslikeC>
|
||||
<humidity>77</humidity>
|
||||
<windDir80m>NNW</windDir80m>
|
||||
<maxFeelslikeF>85</maxFeelslikeF>
|
||||
<precipMM>0</precipMM>
|
||||
<sky>30</sky>
|
||||
<windGust80mMPH>23</windGust80mMPH>
|
||||
<windSpeedMax80mMPH>23</windSpeedMax80mMPH>
|
||||
<weatherPrimary>Mostly Sunny</weatherPrimary>
|
||||
<windGust80mKPH>37</windGust80mKPH>
|
||||
<avgDewpointF>54</avgDewpointF>
|
||||
<windSpeedMax80mKPH>37</windSpeedMax80mKPH>
|
||||
<windGust80mKTS>20</windGust80mKTS>
|
||||
<avgDewpointC>12</avgDewpointC>
|
||||
<precipIN>0</precipIN>
|
||||
<windSpeedMax80mKTS>20</windSpeedMax80mKTS>
|
||||
<windDirMinDEG>325</windDirMinDEG>
|
||||
<windSpeedMaxKPH>13</windSpeedMaxKPH>
|
||||
<windSpeedMin80mKTS>6</windSpeedMin80mKTS>
|
||||
<feelslikeF>67</feelslikeF>
|
||||
<validTime>2022-10-07T07:00:00+02:00</validTime>
|
||||
<windSpeedMin80mMPH>6</windSpeedMin80mMPH>
|
||||
<solradMaxWM2>742</solradMaxWM2>
|
||||
<avgTempC>24</avgTempC>
|
||||
<windSpeedMin80mKPH>10</windSpeedMin80mKPH>
|
||||
<weatherPrimaryCoded>::FW</weatherPrimaryCoded>
|
||||
<sunrise>1665114710</sunrise>
|
||||
<avgTempF>76</avgTempF>
|
||||
<windDirMin>NW</windDirMin>
|
||||
<maxCoverage/>
|
||||
<icon>fair.png</icon>
|
||||
<minFeelslikeC>19</minFeelslikeC>
|
||||
<dewpointC>15</dewpointC>
|
||||
<cloudsCoded>FW</cloudsCoded>
|
||||
<minFeelslikeF>67</minFeelslikeF>
|
||||
<minHumidity>30</minHumidity>
|
||||
<dewpointF>60</dewpointF>
|
||||
<windSpeed80mKTS>6</windSpeed80mKTS>
|
||||
<pop>0</pop>
|
||||
<snowCM>0</snowCM>
|
||||
<windDirMax>WNW</windDirMax>
|
||||
<windSpeed80mMPH>6</windSpeed80mMPH>
|
||||
<windSpeed80mKPH>10</windSpeed80mKPH>
|
||||
<windDir80mDEG>347</windDir80mDEG>
|
||||
<maxTempC>30</maxTempC>
|
||||
<pressureMB>1014</pressureMB>
|
||||
<visibilityKM>24.135</visibilityKM>
|
||||
<timestamp>1665118800</timestamp>
|
||||
<maxTempF>86</maxTempF>
|
||||
<tempF>null</tempF>
|
||||
<minDewpointC>10</minDewpointC>
|
||||
<solradMinWM2>0</solradMinWM2>
|
||||
<windSpeedMinKTS>1</windSpeedMinKTS>
|
||||
<windDirMax80mDEG>316</windDirMax80mDEG>
|
||||
<windGustKTS>16</windGustKTS>
|
||||
<windSpeedMinKPH>2</windSpeedMinKPH>
|
||||
<maxDewpointF>60</maxDewpointF>
|
||||
<windSpeedMinMPH>1</windSpeedMinMPH>
|
||||
<avgFeelslikeC>24</avgFeelslikeC>
|
||||
<uvi>6</uvi>
|
||||
<windDirMax80m>NW</windDirMax80m>
|
||||
<maxDewpointC>15</maxDewpointC>
|
||||
<pressureIN>29.95</pressureIN>
|
||||
<avgFeelslikeF>76</avgFeelslikeF>
|
||||
<iceaccum>null</iceaccum>
|
||||
<isDay>true</isDay>
|
||||
<minTempC>19</minTempC>
|
||||
<minDewpointF>50</minDewpointF>
|
||||
<windSpeedKTS>1</windSpeedKTS>
|
||||
<sunset>1665156831</sunset>
|
||||
<solradWM2>5486</solradWM2>
|
||||
<windSpeedKPH>2</windSpeedKPH>
|
||||
<windGustMPH>18</windGustMPH>
|
||||
<maxHumidity>77</maxHumidity>
|
||||
<windSpeedMPH>1</windSpeedMPH>
|
||||
<windGustKPH>29</windGustKPH>
|
||||
<windDirMaxDEG>298</windDirMaxDEG>
|
||||
<tempC>null</tempC>
|
||||
</periods>
|
||||
<periods>
|
||||
<dateTimeISO>2022-10-08T07:00:00+02:00</dateTimeISO>
|
||||
<windDirMin80m>NW</windDirMin80m>
|
||||
<windDirMin80mDEG>309</windDirMin80mDEG>
|
||||
<feelslikeC>19</feelslikeC>
|
||||
<visibilityMI>15</visibilityMI>
|
||||
<windSpeedMaxMPH>8</windSpeedMaxMPH>
|
||||
<windDirDEG>21</windDirDEG>
|
||||
<windDir>NNE</windDir>
|
||||
<sunriseISO>2022-10-08T05:52:26+02:00</sunriseISO>
|
||||
<iceaccumMM>null</iceaccumMM>
|
||||
<windSpeedMaxKTS>7</windSpeedMaxKTS>
|
||||
<iceaccumIN>null</iceaccumIN>
|
||||
<minTempF>66</minTempF>
|
||||
<snowIN>0</snowIN>
|
||||
<weather>Partly Cloudy</weather>
|
||||
<sunsetISO>2022-10-08T17:32:41+02:00</sunsetISO>
|
||||
<maxFeelslikeC>30</maxFeelslikeC>
|
||||
<humidity>76</humidity>
|
||||
<windDir80m>NW</windDir80m>
|
||||
<maxFeelslikeF>86</maxFeelslikeF>
|
||||
<precipMM>0</precipMM>
|
||||
<sky>47</sky>
|
||||
<windGust80mMPH>19</windGust80mMPH>
|
||||
<windSpeedMax80mMPH>19</windSpeedMax80mMPH>
|
||||
<weatherPrimary>Partly Cloudy</weatherPrimary>
|
||||
<windGust80mKPH>31</windGust80mKPH>
|
||||
<avgDewpointF>56</avgDewpointF>
|
||||
<windSpeedMax80mKPH>31</windSpeedMax80mKPH>
|
||||
<windGust80mKTS>17</windGust80mKTS>
|
||||
<avgDewpointC>13</avgDewpointC>
|
||||
<precipIN>0</precipIN>
|
||||
<windSpeedMax80mKTS>17</windSpeedMax80mKTS>
|
||||
<windDirMinDEG>21</windDirMinDEG>
|
||||
<windSpeedMaxKPH>13</windSpeedMaxKPH>
|
||||
<windSpeedMin80mKTS>5</windSpeedMin80mKTS>
|
||||
<feelslikeF>67</feelslikeF>
|
||||
<validTime>2022-10-08T07:00:00+02:00</validTime>
|
||||
<windSpeedMin80mMPH>5</windSpeedMin80mMPH>
|
||||
<solradMaxWM2>682</solradMaxWM2>
|
||||
<avgTempC>25</avgTempC>
|
||||
<windSpeedMin80mKPH>9</windSpeedMin80mKPH>
|
||||
<weatherPrimaryCoded>::SC</weatherPrimaryCoded>
|
||||
<sunrise>1665201146</sunrise>
|
||||
<avgTempF>76</avgTempF>
|
||||
<windDirMin>NNE</windDirMin>
|
||||
<maxCoverage/>
|
||||
<icon>pcloudy.png</icon>
|
||||
<minFeelslikeC>19</minFeelslikeC>
|
||||
<dewpointC>15</dewpointC>
|
||||
<cloudsCoded>SC</cloudsCoded>
|
||||
<minFeelslikeF>67</minFeelslikeF>
|
||||
<minHumidity>32</minHumidity>
|
||||
<dewpointF>59</dewpointF>
|
||||
<windSpeed80mKTS>5</windSpeed80mKTS>
|
||||
<pop>0</pop>
|
||||
<snowCM>0</snowCM>
|
||||
<windDirMax>WNW</windDirMax>
|
||||
<windSpeed80mMPH>5</windSpeed80mMPH>
|
||||
<windSpeed80mKPH>9</windSpeed80mKPH>
|
||||
<windDir80mDEG>309</windDir80mDEG>
|
||||
<maxTempC>30</maxTempC>
|
||||
<pressureMB>1014</pressureMB>
|
||||
<visibilityKM>24.135</visibilityKM>
|
||||
<timestamp>1665205200</timestamp>
|
||||
<maxTempF>87</maxTempF>
|
||||
<tempF>null</tempF>
|
||||
<minDewpointC>11</minDewpointC>
|
||||
<solradMinWM2>0</solradMinWM2>
|
||||
<windSpeedMinKTS>1</windSpeedMinKTS>
|
||||
<windDirMax80mDEG>322</windDirMax80mDEG>
|
||||
<windGustKTS>17</windGustKTS>
|
||||
<windSpeedMinKPH>2</windSpeedMinKPH>
|
||||
<maxDewpointF>59</maxDewpointF>
|
||||
<windSpeedMinMPH>1</windSpeedMinMPH>
|
||||
<avgFeelslikeC>25</avgFeelslikeC>
|
||||
<uvi>7</uvi>
|
||||
<windDirMax80m>NW</windDirMax80m>
|
||||
<maxDewpointC>15</maxDewpointC>
|
||||
<pressureIN>29.94</pressureIN>
|
||||
<avgFeelslikeF>76</avgFeelslikeF>
|
||||
<iceaccum>null</iceaccum>
|
||||
<isDay>true</isDay>
|
||||
<minTempC>19</minTempC>
|
||||
<minDewpointF>52</minDewpointF>
|
||||
<windSpeedKTS>1</windSpeedKTS>
|
||||
<sunset>1665243161</sunset>
|
||||
<solradWM2>4785</solradWM2>
|
||||
<windSpeedKPH>2</windSpeedKPH>
|
||||
<windGustMPH>20</windGustMPH>
|
||||
<maxHumidity>76</maxHumidity>
|
||||
<windSpeedMPH>1</windSpeedMPH>
|
||||
<windGustKPH>32</windGustKPH>
|
||||
<windDirMaxDEG>301</windDirMaxDEG>
|
||||
<tempC>null</tempC>
|
||||
</periods>
|
||||
<periods>
|
||||
<dateTimeISO>2022-10-09T07:00:00+02:00</dateTimeISO>
|
||||
<windDirMin80m>NW</windDirMin80m>
|
||||
<windDirMin80mDEG>316</windDirMin80mDEG>
|
||||
<feelslikeC>20</feelslikeC>
|
||||
<visibilityMI>15</visibilityMI>
|
||||
<windSpeedMaxMPH>9</windSpeedMaxMPH>
|
||||
<windDirDEG>356</windDirDEG>
|
||||
<windDir>N</windDir>
|
||||
<sunriseISO>2022-10-09T05:53:03+02:00</sunriseISO>
|
||||
<iceaccumMM>null</iceaccumMM>
|
||||
<windSpeedMaxKTS>8</windSpeedMaxKTS>
|
||||
<iceaccumIN>null</iceaccumIN>
|
||||
<minTempF>67</minTempF>
|
||||
<snowIN>0</snowIN>
|
||||
<weather>Partly Cloudy</weather>
|
||||
<sunsetISO>2022-10-09T17:31:31+02:00</sunsetISO>
|
||||
<maxFeelslikeC>30</maxFeelslikeC>
|
||||
<humidity>86</humidity>
|
||||
<windDir80m>NW</windDir80m>
|
||||
<maxFeelslikeF>86</maxFeelslikeF>
|
||||
<precipMM>0</precipMM>
|
||||
<sky>47</sky>
|
||||
<windGust80mMPH>23</windGust80mMPH>
|
||||
<windSpeedMax80mMPH>23</windSpeedMax80mMPH>
|
||||
<weatherPrimary>Partly Cloudy</weatherPrimary>
|
||||
<windGust80mKPH>36</windGust80mKPH>
|
||||
<avgDewpointF>57</avgDewpointF>
|
||||
<windSpeedMax80mKPH>36</windSpeedMax80mKPH>
|
||||
<windGust80mKTS>20</windGust80mKTS>
|
||||
<avgDewpointC>14</avgDewpointC>
|
||||
<precipIN>0</precipIN>
|
||||
<windSpeedMax80mKTS>20</windSpeedMax80mKTS>
|
||||
<windDirMinDEG>356</windDirMinDEG>
|
||||
<windSpeedMaxKPH>14</windSpeedMaxKPH>
|
||||
<windSpeedMin80mKTS>5</windSpeedMin80mKTS>
|
||||
<feelslikeF>67</feelslikeF>
|
||||
<validTime>2022-10-09T07:00:00+02:00</validTime>
|
||||
<windSpeedMin80mMPH>6</windSpeedMin80mMPH>
|
||||
<solradMaxWM2>726</solradMaxWM2>
|
||||
<avgTempC>25</avgTempC>
|
||||
<windSpeedMin80mKPH>9</windSpeedMin80mKPH>
|
||||
<weatherPrimaryCoded>::SC</weatherPrimaryCoded>
|
||||
<sunrise>1665287583</sunrise>
|
||||
<avgTempF>77</avgTempF>
|
||||
<windDirMin>N</windDirMin>
|
||||
<maxCoverage/>
|
||||
<icon>pcloudy.png</icon>
|
||||
<minFeelslikeC>20</minFeelslikeC>
|
||||
<dewpointC>17</dewpointC>
|
||||
<cloudsCoded>SC</cloudsCoded>
|
||||
<minFeelslikeF>67</minFeelslikeF>
|
||||
<minHumidity>31</minHumidity>
|
||||
<dewpointF>63</dewpointF>
|
||||
<windSpeed80mKTS>5</windSpeed80mKTS>
|
||||
<pop>0</pop>
|
||||
<snowCM>0</snowCM>
|
||||
<windDirMax>NNW</windDirMax>
|
||||
<windSpeed80mMPH>6</windSpeed80mMPH>
|
||||
<windSpeed80mKPH>9</windSpeed80mKPH>
|
||||
<windDir80mDEG>316</windDir80mDEG>
|
||||
<maxTempC>31</maxTempC>
|
||||
<pressureMB>1016</pressureMB>
|
||||
<visibilityKM>24.135</visibilityKM>
|
||||
<timestamp>1665291600</timestamp>
|
||||
<maxTempF>87</maxTempF>
|
||||
<tempF>null</tempF>
|
||||
<minDewpointC>11</minDewpointC>
|
||||
<solradMinWM2>0</solradMinWM2>
|
||||
<windSpeedMinKTS>2</windSpeedMinKTS>
|
||||
<windDirMax80mDEG>354</windDirMax80mDEG>
|
||||
<windGustKTS>19</windGustKTS>
|
||||
<windSpeedMinKPH>4</windSpeedMinKPH>
|
||||
<maxDewpointF>63</maxDewpointF>
|
||||
<windSpeedMinMPH>2</windSpeedMinMPH>
|
||||
<avgFeelslikeC>25</avgFeelslikeC>
|
||||
<uvi>7</uvi>
|
||||
<windDirMax80m>N</windDirMax80m>
|
||||
<maxDewpointC>17</maxDewpointC>
|
||||
<pressureIN>29.99</pressureIN>
|
||||
<avgFeelslikeF>77</avgFeelslikeF>
|
||||
<iceaccum>null</iceaccum>
|
||||
<isDay>true</isDay>
|
||||
<minTempC>19</minTempC>
|
||||
<minDewpointF>52</minDewpointF>
|
||||
<windSpeedKTS>2</windSpeedKTS>
|
||||
<sunset>1665329491</sunset>
|
||||
<solradWM2>4768</solradWM2>
|
||||
<windSpeedKPH>4</windSpeedKPH>
|
||||
<windGustMPH>22</windGustMPH>
|
||||
<maxHumidity>86</maxHumidity>
|
||||
<windSpeedMPH>2</windSpeedMPH>
|
||||
<windGustKPH>36</windGustKPH>
|
||||
<windDirMaxDEG>343</windDirMaxDEG>
|
||||
<tempC>null</tempC>
|
||||
</periods>
|
||||
<periods>
|
||||
<dateTimeISO>2022-10-10T07:00:00+02:00</dateTimeISO>
|
||||
<windDirMin80m>E</windDirMin80m>
|
||||
<windDirMin80mDEG>91</windDirMin80mDEG>
|
||||
<feelslikeC>21</feelslikeC>
|
||||
<visibilityMI>15</visibilityMI>
|
||||
<windSpeedMaxMPH>9</windSpeedMaxMPH>
|
||||
<windDirDEG>358</windDirDEG>
|
||||
<windDir>N</windDir>
|
||||
<sunriseISO>2022-10-10T05:53:40+02:00</sunriseISO>
|
||||
<iceaccumMM>null</iceaccumMM>
|
||||
<windSpeedMaxKTS>8</windSpeedMaxKTS>
|
||||
<iceaccumIN>null</iceaccumIN>
|
||||
<minTempF>70</minTempF>
|
||||
<snowIN>0</snowIN>
|
||||
<weather>Partly Cloudy</weather>
|
||||
<sunsetISO>2022-10-10T17:30:21+02:00</sunsetISO>
|
||||
<maxFeelslikeC>30</maxFeelslikeC>
|
||||
<humidity>75</humidity>
|
||||
<windDir80m>N</windDir80m>
|
||||
<maxFeelslikeF>86</maxFeelslikeF>
|
||||
<precipMM>0</precipMM>
|
||||
<sky>64</sky>
|
||||
<windGust80mMPH>22</windGust80mMPH>
|
||||
<windSpeedMax80mMPH>22</windSpeedMax80mMPH>
|
||||
<weatherPrimary>Partly Cloudy</weatherPrimary>
|
||||
<windGust80mKPH>36</windGust80mKPH>
|
||||
<avgDewpointF>58</avgDewpointF>
|
||||
<windSpeedMax80mKPH>36</windSpeedMax80mKPH>
|
||||
<windGust80mKTS>19</windGust80mKTS>
|
||||
<avgDewpointC>14</avgDewpointC>
|
||||
<precipIN>0</precipIN>
|
||||
<windSpeedMax80mKTS>19</windSpeedMax80mKTS>
|
||||
<windDirMinDEG>358</windDirMinDEG>
|
||||
<windSpeedMaxKPH>15</windSpeedMaxKPH>
|
||||
<windSpeedMin80mKTS>7</windSpeedMin80mKTS>
|
||||
<feelslikeF>69</feelslikeF>
|
||||
<validTime>2022-10-10T07:00:00+02:00</validTime>
|
||||
<windSpeedMin80mMPH>8</windSpeedMin80mMPH>
|
||||
<solradMaxWM2>597</solradMaxWM2>
|
||||
<avgTempC>26</avgTempC>
|
||||
<windSpeedMin80mKPH>13</windSpeedMin80mKPH>
|
||||
<weatherPrimaryCoded>::SC</weatherPrimaryCoded>
|
||||
<sunrise>1665374020</sunrise>
|
||||
<avgTempF>78</avgTempF>
|
||||
<windDirMin>N</windDirMin>
|
||||
<maxCoverage/>
|
||||
<icon>pcloudy.png</icon>
|
||||
<minFeelslikeC>21</minFeelslikeC>
|
||||
<dewpointC>16</dewpointC>
|
||||
<cloudsCoded>SC</cloudsCoded>
|
||||
<minFeelslikeF>69</minFeelslikeF>
|
||||
<minHumidity>35</minHumidity>
|
||||
<dewpointF>61</dewpointF>
|
||||
<windSpeed80mKTS>7</windSpeed80mKTS>
|
||||
<pop>0</pop>
|
||||
<snowCM>0</snowCM>
|
||||
<windDirMax>N</windDirMax>
|
||||
<windSpeed80mMPH>8</windSpeed80mMPH>
|
||||
<windSpeed80mKPH>13</windSpeed80mKPH>
|
||||
<windDir80mDEG>8</windDir80mDEG>
|
||||
<maxTempC>31</maxTempC>
|
||||
<pressureMB>1017</pressureMB>
|
||||
<visibilityKM>24.135</visibilityKM>
|
||||
<timestamp>1665378000</timestamp>
|
||||
<maxTempF>87</maxTempF>
|
||||
<tempF>null</tempF>
|
||||
<minDewpointC>13</minDewpointC>
|
||||
<solradMinWM2>0</solradMinWM2>
|
||||
<windSpeedMinKTS>2</windSpeedMinKTS>
|
||||
<windDirMax80mDEG>10</windDirMax80mDEG>
|
||||
<windGustKTS>16</windGustKTS>
|
||||
<windSpeedMinKPH>4</windSpeedMinKPH>
|
||||
<maxDewpointF>61</maxDewpointF>
|
||||
<windSpeedMinMPH>2</windSpeedMinMPH>
|
||||
<avgFeelslikeC>25</avgFeelslikeC>
|
||||
<uvi>6</uvi>
|
||||
<windDirMax80m>N</windDirMax80m>
|
||||
<maxDewpointC>16</maxDewpointC>
|
||||
<pressureIN>30.03</pressureIN>
|
||||
<avgFeelslikeF>78</avgFeelslikeF>
|
||||
<iceaccum>null</iceaccum>
|
||||
<isDay>true</isDay>
|
||||
<minTempC>21</minTempC>
|
||||
<minDewpointF>55</minDewpointF>
|
||||
<windSpeedKTS>2</windSpeedKTS>
|
||||
<sunset>1665415821</sunset>
|
||||
<solradWM2>4494</solradWM2>
|
||||
<windSpeedKPH>4</windSpeedKPH>
|
||||
<windGustMPH>19</windGustMPH>
|
||||
<maxHumidity>75</maxHumidity>
|
||||
<windSpeedMPH>2</windSpeedMPH>
|
||||
<windGustKPH>30</windGustKPH>
|
||||
<windDirMaxDEG>10</windDirMaxDEG>
|
||||
<tempC>null</tempC>
|
||||
</periods>
|
||||
<periods>
|
||||
<dateTimeISO>2022-10-11T07:00:00+02:00</dateTimeISO>
|
||||
<windDirMin80m>NNE</windDirMin80m>
|
||||
<windDirMin80mDEG>13</windDirMin80mDEG>
|
||||
<feelslikeC>22</feelslikeC>
|
||||
<visibilityMI>15</visibilityMI>
|
||||
<windSpeedMaxMPH>18</windSpeedMaxMPH>
|
||||
<windDirDEG>13</windDirDEG>
|
||||
<windDir>NNE</windDir>
|
||||
<sunriseISO>2022-10-11T05:54:18+02:00</sunriseISO>
|
||||
<iceaccumMM>null</iceaccumMM>
|
||||
<windSpeedMaxKTS>15</windSpeedMaxKTS>
|
||||
<iceaccumIN>null</iceaccumIN>
|
||||
<minTempF>70</minTempF>
|
||||
<snowIN>0</snowIN>
|
||||
<weather>Sunny</weather>
|
||||
<sunsetISO>2022-10-11T17:29:13+02:00</sunsetISO>
|
||||
<maxFeelslikeC>31</maxFeelslikeC>
|
||||
<humidity>71</humidity>
|
||||
<windDir80m>NNE</windDir80m>
|
||||
<maxFeelslikeF>87</maxFeelslikeF>
|
||||
<precipMM>0</precipMM>
|
||||
<sky>0</sky>
|
||||
<windGust80mMPH>19</windGust80mMPH>
|
||||
<windSpeedMax80mMPH>19</windSpeedMax80mMPH>
|
||||
<weatherPrimary>Sunny</weatherPrimary>
|
||||
<windGust80mKPH>31</windGust80mKPH>
|
||||
<avgDewpointF>55</avgDewpointF>
|
||||
<windSpeedMax80mKPH>31</windSpeedMax80mKPH>
|
||||
<windGust80mKTS>17</windGust80mKTS>
|
||||
<avgDewpointC>13</avgDewpointC>
|
||||
<precipIN>0</precipIN>
|
||||
<windSpeedMax80mKTS>17</windSpeedMax80mKTS>
|
||||
<windDirMinDEG>14</windDirMinDEG>
|
||||
<windSpeedMaxKPH>28</windSpeedMaxKPH>
|
||||
<windSpeedMin80mKTS>9</windSpeedMin80mKTS>
|
||||
<feelslikeF>72</feelslikeF>
|
||||
<validTime>2022-10-11T07:00:00+02:00</validTime>
|
||||
<windSpeedMin80mMPH>11</windSpeedMin80mMPH>
|
||||
<solradMaxWM2>758</solradMaxWM2>
|
||||
<avgTempC>26</avgTempC>
|
||||
<windSpeedMin80mKPH>18</windSpeedMin80mKPH>
|
||||
<weatherPrimaryCoded>::CL</weatherPrimaryCoded>
|
||||
<sunrise>1665460458</sunrise>
|
||||
<avgTempF>78</avgTempF>
|
||||
<windDirMin>NNE</windDirMin>
|
||||
<maxCoverage/>
|
||||
<icon>sunny.png</icon>
|
||||
<minFeelslikeC>22</minFeelslikeC>
|
||||
<dewpointC>17</dewpointC>
|
||||
<cloudsCoded>CL</cloudsCoded>
|
||||
<minFeelslikeF>72</minFeelslikeF>
|
||||
<minHumidity>30</minHumidity>
|
||||
<dewpointF>62</dewpointF>
|
||||
<windSpeed80mKTS>10</windSpeed80mKTS>
|
||||
<pop>0</pop>
|
||||
<snowCM>0</snowCM>
|
||||
<windDirMax>NNE</windDirMax>
|
||||
<windSpeed80mMPH>12</windSpeed80mMPH>
|
||||
<windSpeed80mKPH>19</windSpeed80mKPH>
|
||||
<windDir80mDEG>16</windDir80mDEG>
|
||||
<maxTempC>31</maxTempC>
|
||||
<pressureMB>1015</pressureMB>
|
||||
<visibilityKM>24.135</visibilityKM>
|
||||
<timestamp>1665464400</timestamp>
|
||||
<maxTempF>87</maxTempF>
|
||||
<tempF>null</tempF>
|
||||
<minDewpointC>11</minDewpointC>
|
||||
<solradMinWM2>0</solradMinWM2>
|
||||
<windSpeedMinKTS>7</windSpeedMinKTS>
|
||||
<windDirMax80mDEG>28</windDirMax80mDEG>
|
||||
<windGustKTS>15</windGustKTS>
|
||||
<windSpeedMinKPH>14</windSpeedMinKPH>
|
||||
<maxDewpointF>62</maxDewpointF>
|
||||
<windSpeedMinMPH>8</windSpeedMinMPH>
|
||||
<avgFeelslikeC>26</avgFeelslikeC>
|
||||
<uvi>null</uvi>
|
||||
<windDirMax80m>NNE</windDirMax80m>
|
||||
<maxDewpointC>17</maxDewpointC>
|
||||
<pressureIN>29.98</pressureIN>
|
||||
<avgFeelslikeF>79</avgFeelslikeF>
|
||||
<iceaccum>null</iceaccum>
|
||||
<isDay>true</isDay>
|
||||
<minTempC>21</minTempC>
|
||||
<minDewpointF>51</minDewpointF>
|
||||
<windSpeedKTS>8</windSpeedKTS>
|
||||
<sunset>1665502153</sunset>
|
||||
<solradWM2>5450</solradWM2>
|
||||
<windSpeedKPH>15</windSpeedKPH>
|
||||
<windGustMPH>17</windGustMPH>
|
||||
<maxHumidity>71</maxHumidity>
|
||||
<windSpeedMPH>9</windSpeedMPH>
|
||||
<windGustKPH>28</windGustKPH>
|
||||
<windDirMaxDEG>28</windDirMaxDEG>
|
||||
<tempC>null</tempC>
|
||||
</periods>
|
||||
<periods>
|
||||
<dateTimeISO>2022-10-12T07:00:00+02:00</dateTimeISO>
|
||||
<windDirMin80m>NNE</windDirMin80m>
|
||||
<windDirMin80mDEG>15</windDirMin80mDEG>
|
||||
<feelslikeC>22</feelslikeC>
|
||||
<visibilityMI>15</visibilityMI>
|
||||
<windSpeedMaxMPH>16</windSpeedMaxMPH>
|
||||
<windDirDEG>12</windDirDEG>
|
||||
<windDir>NNE</windDir>
|
||||
<sunriseISO>2022-10-12T05:54:55+02:00</sunriseISO>
|
||||
<iceaccumMM>null</iceaccumMM>
|
||||
<windSpeedMaxKTS>14</windSpeedMaxKTS>
|
||||
<iceaccumIN>null</iceaccumIN>
|
||||
<minTempF>69</minTempF>
|
||||
<snowIN>0</snowIN>
|
||||
<weather>Mostly Sunny</weather>
|
||||
<sunsetISO>2022-10-12T17:28:04+02:00</sunsetISO>
|
||||
<maxFeelslikeC>31</maxFeelslikeC>
|
||||
<humidity>68</humidity>
|
||||
<windDir80m>NNE</windDir80m>
|
||||
<maxFeelslikeF>88</maxFeelslikeF>
|
||||
<precipMM>0</precipMM>
|
||||
<sky>27</sky>
|
||||
<windGust80mMPH>21</windGust80mMPH>
|
||||
<windSpeedMax80mMPH>21</windSpeedMax80mMPH>
|
||||
<weatherPrimary>Mostly Sunny</weatherPrimary>
|
||||
<windGust80mKPH>33</windGust80mKPH>
|
||||
<avgDewpointF>55</avgDewpointF>
|
||||
<windSpeedMax80mKPH>33</windSpeedMax80mKPH>
|
||||
<windGust80mKTS>18</windGust80mKTS>
|
||||
<avgDewpointC>13</avgDewpointC>
|
||||
<precipIN>0</precipIN>
|
||||
<windSpeedMax80mKTS>18</windSpeedMax80mKTS>
|
||||
<windDirMinDEG>12</windDirMinDEG>
|
||||
<windSpeedMaxKPH>26</windSpeedMaxKPH>
|
||||
<windSpeedMin80mKTS>10</windSpeedMin80mKTS>
|
||||
<feelslikeF>72</feelslikeF>
|
||||
<validTime>2022-10-12T07:00:00+02:00</validTime>
|
||||
<windSpeedMin80mMPH>11</windSpeedMin80mMPH>
|
||||
<solradMaxWM2>743</solradMaxWM2>
|
||||
<avgTempC>26</avgTempC>
|
||||
<windSpeedMin80mKPH>18</windSpeedMin80mKPH>
|
||||
<weatherPrimaryCoded>::FW</weatherPrimaryCoded>
|
||||
<sunrise>1665546895</sunrise>
|
||||
<avgTempF>79</avgTempF>
|
||||
<windDirMin>NNE</windDirMin>
|
||||
<maxCoverage/>
|
||||
<icon>fair.png</icon>
|
||||
<minFeelslikeC>22</minFeelslikeC>
|
||||
<dewpointC>16</dewpointC>
|
||||
<cloudsCoded>FW</cloudsCoded>
|
||||
<minFeelslikeF>72</minFeelslikeF>
|
||||
<minHumidity>29</minHumidity>
|
||||
<dewpointF>60</dewpointF>
|
||||
<windSpeed80mKTS>10</windSpeed80mKTS>
|
||||
<pop>0</pop>
|
||||
<snowCM>0</snowCM>
|
||||
<windDirMax>E</windDirMax>
|
||||
<windSpeed80mMPH>12</windSpeed80mMPH>
|
||||
<windSpeed80mKPH>19</windSpeed80mKPH>
|
||||
<windDir80mDEG>15</windDir80mDEG>
|
||||
<maxTempC>31</maxTempC>
|
||||
<pressureMB>1014</pressureMB>
|
||||
<visibilityKM>24.135</visibilityKM>
|
||||
<timestamp>1665550800</timestamp>
|
||||
<maxTempF>88</maxTempF>
|
||||
<tempF>null</tempF>
|
||||
<minDewpointC>11</minDewpointC>
|
||||
<solradMinWM2>0</solradMinWM2>
|
||||
<windSpeedMinKTS>7</windSpeedMinKTS>
|
||||
<windDirMax80mDEG>96</windDirMax80mDEG>
|
||||
<windGustKTS>15</windGustKTS>
|
||||
<windSpeedMinKPH>13</windSpeedMinKPH>
|
||||
<maxDewpointF>60</maxDewpointF>
|
||||
<windSpeedMinMPH>8</windSpeedMinMPH>
|
||||
<avgFeelslikeC>26</avgFeelslikeC>
|
||||
<uvi>null</uvi>
|
||||
<windDirMax80m>E</windDirMax80m>
|
||||
<maxDewpointC>16</maxDewpointC>
|
||||
<pressureIN>29.95</pressureIN>
|
||||
<avgFeelslikeF>80</avgFeelslikeF>
|
||||
<iceaccum>null</iceaccum>
|
||||
<isDay>true</isDay>
|
||||
<minTempC>21</minTempC>
|
||||
<minDewpointF>51</minDewpointF>
|
||||
<windSpeedKTS>8</windSpeedKTS>
|
||||
<sunset>1665588484</sunset>
|
||||
<solradWM2>4740</solradWM2>
|
||||
<windSpeedKPH>15</windSpeedKPH>
|
||||
<windGustMPH>17</windGustMPH>
|
||||
<maxHumidity>68</maxHumidity>
|
||||
<windSpeedMPH>9</windSpeedMPH>
|
||||
<windGustKPH>28</windGustKPH>
|
||||
<windDirMaxDEG>96</windDirMaxDEG>
|
||||
<tempC>null</tempC>
|
||||
</periods>
|
||||
<interval>day</interval>
|
||||
<place>
|
||||
<country>eg</country>
|
||||
<name>cairo</name>
|
||||
<state>qh</state>
|
||||
</place>
|
||||
</response>
|
||||
<error>null</error>
|
||||
Reference in New Issue
Block a user