mirror of
https://github.com/stleary/JSON-java.git
synced 2026-01-24 00:03:17 -05:00
Compare commits
64 Commits
6daabb43ab
...
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 | ||
|
|
89f6e7f6a6 | ||
|
|
346fb26f2d | ||
|
|
7dd9e01dae | ||
|
|
89f16ad0af |
42
Examples.md
42
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,7 +14,7 @@ import java.util.Properties;
|
||||
|
||||
<h3>Using JSONArray</h3>
|
||||
|
||||
```
|
||||
```java
|
||||
private static void JSONExampleArray1() {
|
||||
//We create a JSONObject from a String containing an array using JSONArray
|
||||
//Firstly, we declare an Array in a String
|
||||
@@ -57,7 +57,7 @@ import java.util.Properties;
|
||||
return res;
|
||||
}
|
||||
```
|
||||
```
|
||||
```java
|
||||
private static void JSONExampleArray2() {
|
||||
|
||||
//We can also create an Array without a String by creating an empty array and adding elements to it
|
||||
@@ -81,7 +81,7 @@ import java.util.Properties;
|
||||
|
||||
<h3>Using JSONStringer</h3>
|
||||
|
||||
```
|
||||
```java
|
||||
private static void JSONExampleStringer() {
|
||||
|
||||
//We initializate the JSONStringer
|
||||
@@ -116,7 +116,7 @@ import java.util.Properties;
|
||||
```
|
||||
<h3>Using JSONObject</h3>
|
||||
|
||||
```
|
||||
```java
|
||||
private static void JSONExampleObject1() {
|
||||
|
||||
//We can create a JSONObject from a String with the class builder
|
||||
@@ -127,7 +127,7 @@ import java.util.Properties;
|
||||
|
||||
}
|
||||
```
|
||||
```
|
||||
```java
|
||||
private static void JSONExampleObject2() {
|
||||
|
||||
//We can also create a JSONObject directly without messing around with any of the other functions.
|
||||
@@ -151,7 +151,7 @@ import java.util.Properties;
|
||||
System.out.println("Final JSONOBject: " + example);
|
||||
}
|
||||
```
|
||||
```
|
||||
```java
|
||||
private static void JSONExampleObject3() {
|
||||
|
||||
//We can also create a JSONObject with a Java Map
|
||||
@@ -170,7 +170,7 @@ import java.util.Properties;
|
||||
```
|
||||
<h3>Using JSONWriter</h3>
|
||||
|
||||
```
|
||||
```java
|
||||
private static void JSONExamplWriter() {
|
||||
|
||||
//This method works in a very similar way to Object and Stringer in the construction of the JSON.
|
||||
@@ -202,7 +202,7 @@ import java.util.Properties;
|
||||
|
||||
}
|
||||
```
|
||||
```
|
||||
```java
|
||||
private static void JSONExampleTokener() {
|
||||
|
||||
//A partir de una String podemos crear un JSONTokener, que lo podemos usar alternativamente para JSONArray,JSONObject
|
||||
@@ -223,7 +223,7 @@ import java.util.Properties;
|
||||
|
||||
<h3>Extra: Conversion to JSONArray</h3>
|
||||
|
||||
```
|
||||
```java
|
||||
private static void JSONObjectToArray() {
|
||||
//We start with a JSONObject
|
||||
|
||||
@@ -244,7 +244,7 @@ import java.util.Properties;
|
||||
```
|
||||
<h3>XML Conversions</h3>
|
||||
|
||||
```
|
||||
```java
|
||||
private static void XMLToExampleConversion() {
|
||||
|
||||
//We start with a JSONObject
|
||||
@@ -258,7 +258,7 @@ import java.util.Properties;
|
||||
System.out.println("Final XML: " + output);
|
||||
}
|
||||
```
|
||||
```
|
||||
```java
|
||||
private static void XMLFromExampleConversion() {
|
||||
|
||||
//We start with a string with the XML format
|
||||
@@ -274,7 +274,7 @@ import java.util.Properties;
|
||||
```
|
||||
<h3>Cookie Conversions</h3>
|
||||
|
||||
```
|
||||
```java
|
||||
private static void CookieToExampleConversion() {
|
||||
|
||||
//We start with a JSONObject
|
||||
@@ -290,7 +290,7 @@ import java.util.Properties;
|
||||
System.out.println("Final Cookie: " + output);
|
||||
}
|
||||
```
|
||||
```
|
||||
```java
|
||||
private static void CookieFromExampleConversion() {
|
||||
|
||||
//We start with a string with the Cookie format
|
||||
@@ -306,7 +306,7 @@ import java.util.Properties;
|
||||
|
||||
<h3>HTTP Conversions</h3>
|
||||
|
||||
```
|
||||
```java
|
||||
private static void HTTPToExampleConversion() {
|
||||
|
||||
//We start with a JSONObject
|
||||
@@ -322,7 +322,7 @@ import java.util.Properties;
|
||||
System.out.println("Final HTTP: " + output);
|
||||
}
|
||||
```
|
||||
```
|
||||
```java
|
||||
private static void HTTPFromExampleConversion() {
|
||||
|
||||
//We start with a string with the HTTP format
|
||||
@@ -337,7 +337,7 @@ import java.util.Properties;
|
||||
```
|
||||
<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"
|
||||
@@ -360,7 +360,7 @@ private static void CDLToExampleConversion() {
|
||||
System.out.println("Final CDL: \r\n" + output);
|
||||
}
|
||||
```
|
||||
```
|
||||
```java
|
||||
private static void CDLFromExampleConversion() {
|
||||
|
||||
//We start wtih a String with the CDL format
|
||||
@@ -377,7 +377,7 @@ private static void CDLFromExampleConversion() {
|
||||
```
|
||||
<h3>Properties Conversions</h3>
|
||||
|
||||
```
|
||||
```java
|
||||
private static Properties PropertyToExampleConversion() {
|
||||
|
||||
//We start with a JSONObject
|
||||
@@ -393,7 +393,7 @@ private static void CDLFromExampleConversion() {
|
||||
return output;
|
||||
}
|
||||
```
|
||||
```
|
||||
```java
|
||||
private static void PropertyFromExampleConversion() {
|
||||
|
||||
//We start with a Properties object
|
||||
@@ -408,7 +408,7 @@ private static void CDLFromExampleConversion() {
|
||||
```
|
||||
<h2>List of all examples methods</h2>
|
||||
|
||||
```
|
||||
```java
|
||||
public static void main(String[] args) {
|
||||
//JSONObjectToArray();
|
||||
//JSONExampleArray1();
|
||||
|
||||
@@ -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>
|
||||
|
||||
@@ -1366,6 +1366,10 @@ public class JSONArray implements Iterable<Object> {
|
||||
if (!JSONObject.isNumberSimilar((Number)valueThis, (Number)valueOther)) {
|
||||
return false;
|
||||
}
|
||||
} else if (valueThis instanceof JSONString && valueOther instanceof JSONString) {
|
||||
if (!((JSONString) valueThis).toJSONString().equals(((JSONString) valueOther).toJSONString())) {
|
||||
return false;
|
||||
}
|
||||
} else if (!valueThis.equals(valueOther)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
@@ -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);
|
||||
}
|
||||
|
||||
|
||||
@@ -246,7 +276,56 @@ 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);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||
* JSONArray using the JsonML transform. Each XML tag is represented as
|
||||
* a JSONArray in which the first element is the tag name. If the tag has
|
||||
* attributes, then the second element will be JSONObject containing the
|
||||
* name/value pairs. If the tag contains children, then strings and
|
||||
* JSONArrays will represent the child content and tags.
|
||||
* As opposed to toJSONArray this method does not attempt to convert
|
||||
* any text node or attribute value to any type
|
||||
* but just leaves it as a string.
|
||||
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||
* @param x An XMLTokener.
|
||||
* @param config The parser configuration:
|
||||
* JSONMLParserConfiguration.ORIGINAL is the default behaviour;
|
||||
* JSONMLParserConfiguration.KEEP_STRINGS means values will not be coerced into boolean
|
||||
* or numeric values and will instead be left as strings
|
||||
* @return A JSONArray containing the structured data from the XML string.
|
||||
* @throws JSONException Thrown on error converting to a JSONArray
|
||||
*/
|
||||
public static JSONArray toJSONArray(XMLTokener x, JSONMLParserConfiguration config) throws JSONException {
|
||||
return (JSONArray)parse(x, true, null, config, 0);
|
||||
}
|
||||
|
||||
|
||||
@@ -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,7 +382,7 @@ 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);
|
||||
}
|
||||
|
||||
|
||||
@@ -323,7 +402,29 @@ 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);
|
||||
}
|
||||
|
||||
|
||||
@@ -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();
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -2125,6 +2121,10 @@ public class JSONObject {
|
||||
if (!isNumberSimilar((Number)valueThis, (Number)valueOther)) {
|
||||
return false;
|
||||
}
|
||||
} else if (valueThis instanceof JSONString && valueOther instanceof JSONString) {
|
||||
if (!((JSONString) valueThis).toJSONString().equals(((JSONString) valueOther).toJSONString())) {
|
||||
return false;
|
||||
}
|
||||
} else if (!valueThis.equals(valueOther)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
@@ -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")));
|
||||
}
|
||||
|
||||
|
||||
@@ -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;
|
||||
}
|
||||
}
|
||||
@@ -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) {
|
||||
@@ -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;
|
||||
@@ -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,12 +29,6 @@ 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
|
||||
@@ -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;
|
||||
}
|
||||
@@ -141,10 +141,12 @@ public class XMLParserConfiguration {
|
||||
* @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 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,20 +168,11 @@ 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)
|
||||
@@ -189,10 +182,9 @@ public class XMLParserConfiguration {
|
||||
*
|
||||
* @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);
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -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);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -29,7 +29,9 @@ import org.json.JSONArray;
|
||||
import org.json.JSONException;
|
||||
import org.json.JSONObject;
|
||||
import org.json.JSONPointerException;
|
||||
import org.json.JSONString;
|
||||
import org.json.JSONTokener;
|
||||
import org.json.junit.data.MyJsonString;
|
||||
import org.junit.Test;
|
||||
|
||||
import com.jayway.jsonpath.Configuration;
|
||||
@@ -1334,4 +1336,25 @@ public class JSONArrayTest {
|
||||
fail("Excepected Exception.");
|
||||
Util.checkJSONArrayMaps(json_input);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testIssue682SimilarityOfJSONString() {
|
||||
JSONArray ja1 = new JSONArray()
|
||||
.put(new MyJsonString())
|
||||
.put(2);
|
||||
JSONArray ja2 = new JSONArray()
|
||||
.put(new MyJsonString())
|
||||
.put(2);
|
||||
assertTrue(ja1.similar(ja2));
|
||||
|
||||
JSONArray ja3 = new JSONArray()
|
||||
.put(new JSONString() {
|
||||
@Override
|
||||
public String toJSONString() {
|
||||
return "\"different value\"";
|
||||
}
|
||||
})
|
||||
.put(2);
|
||||
assertFalse(ja1.similar(ja3));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -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"));
|
||||
}
|
||||
}
|
||||
@@ -30,6 +30,7 @@ import org.json.JSONArray;
|
||||
import org.json.JSONException;
|
||||
import org.json.JSONObject;
|
||||
import org.json.JSONPointerException;
|
||||
import org.json.JSONString;
|
||||
import org.json.JSONTokener;
|
||||
import org.json.XML;
|
||||
import org.json.junit.data.BrokenToString;
|
||||
@@ -3487,4 +3488,25 @@ public class JSONObjectTest {
|
||||
assertNotNull(json_input);
|
||||
fail("Excepected Exception.");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testIssue682SimilarityOfJSONString() {
|
||||
JSONObject jo1 = new JSONObject()
|
||||
.put("a", new MyJsonString())
|
||||
.put("b", 2);
|
||||
JSONObject jo2 = new JSONObject()
|
||||
.put("a", new MyJsonString())
|
||||
.put("b", 2);
|
||||
assertTrue(jo1.similar(jo2));
|
||||
|
||||
JSONObject jo3 = new JSONObject()
|
||||
.put("a", new JSONString() {
|
||||
@Override
|
||||
public String toJSONString() {
|
||||
return "\"different value\"";
|
||||
}
|
||||
})
|
||||
.put("b", 2);
|
||||
assertFalse(jo1.similar(jo3));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -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());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1052,6 +1052,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,
|
||||
* convert to JSONObject and compare actual to 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